🐛 Bug fixes on bad switching account UX

This commit is contained in:
LittleSheep 2024-05-23 23:54:05 +08:00
parent 3e640768c8
commit 15ed75b04e
12 changed files with 149 additions and 103 deletions

View File

@ -9,13 +9,8 @@ import 'package:solian/services.dart';
import 'package:oauth2/oauth2.dart' as oauth2; import 'package:oauth2/oauth2.dart' as oauth2;
class AuthProvider extends GetConnect { class AuthProvider extends GetConnect {
final deviceEndpoint = Uri.parse(
'${ServiceFinder.services['passport']}/api/notifications/subscribe');
final tokenEndpoint = final tokenEndpoint =
Uri.parse('${ServiceFinder.services['passport']}/api/auth/token'); Uri.parse('${ServiceFinder.services['passport']}/api/auth/token');
final userinfoEndpoint =
Uri.parse('${ServiceFinder.services['passport']}/api/users/me');
final redirectUrl = Uri.parse('solian://auth');
static const clientId = 'solian'; static const clientId = 'solian';
static const clientSecret = '_F4%q2Eea3'; static const clientSecret = '_F4%q2Eea3';
@ -25,13 +20,12 @@ class AuthProvider extends GetConnect {
@override @override
void onInit() { void onInit() {
httpClient.baseUrl = ServiceFinder.services['passport']; httpClient.baseUrl = ServiceFinder.services['passport'];
loadCredentials();
applyAuthenticator();
} }
oauth2.Credentials? credentials; oauth2.Credentials? credentials;
Future<Request<T?>> reqAuthenticator<T>(Request<T?> request) async { Future<Request<T?>> requestAuthenticator<T>(Request<T?> request) async {
if (credentials != null && credentials!.isExpired) { if (credentials != null && credentials!.isExpired) {
final resp = await post('/api/auth/token', { final resp = await post('/api/auth/token', {
'refresh_token': credentials!.refreshToken, 'refresh_token': credentials!.refreshToken,
@ -48,7 +42,9 @@ class AuthProvider extends GetConnect {
expiration: DateTime.now().add(const Duration(minutes: 3)), expiration: DateTime.now().add(const Duration(minutes: 3)),
); );
storage.write( storage.write(
key: 'auth_credentials', value: jsonEncode(credentials!.toJson())); key: 'auth_credentials',
value: jsonEncode(credentials!.toJson()),
);
} }
if (credentials != null) { if (credentials != null) {
@ -58,18 +54,20 @@ class AuthProvider extends GetConnect {
return request; return request;
} }
void applyAuthenticator() { Future<void> loadCredentials() async {
isAuthorized.then((status) async { if (await isAuthorized) {
if (status) {
final content = await storage.read(key: 'auth_credentials'); final content = await storage.read(key: 'auth_credentials');
credentials = oauth2.Credentials.fromJson(jsonDecode(content!)); credentials = oauth2.Credentials.fromJson(jsonDecode(content!));
httpClient.addAuthenticator(reqAuthenticator);
} }
});
} }
Future<oauth2.Credentials> signin( Future<oauth2.Credentials> signin(
BuildContext context, String username, String password) async { BuildContext context,
String username,
String password,
) async {
_cacheUserProfileResponse = null;
final resp = await oauth2.resourceOwnerPasswordGrant( final resp = await oauth2.resourceOwnerPasswordGrant(
tokenEndpoint, tokenEndpoint,
username, username,
@ -89,13 +87,16 @@ class AuthProvider extends GetConnect {
); );
storage.write( storage.write(
key: 'auth_credentials', value: jsonEncode(credentials!.toJson())); key: 'auth_credentials',
applyAuthenticator(); value: jsonEncode(credentials!.toJson()),
);
return credentials!; return credentials!;
} }
void signout() { void signout() {
_cacheUserProfileResponse = null;
storage.deleteAll(); storage.deleteAll();
} }
@ -108,7 +109,11 @@ class AuthProvider extends GetConnect {
return _cacheUserProfileResponse!; return _cacheUserProfileResponse!;
} }
final resp = await get('/api/users/me'); final client = GetConnect();
client.httpClient.baseUrl = ServiceFinder.services['passport'];
client.httpClient.addAuthenticator(requestAuthenticator);
final resp = await client.get('/api/users/me');
_cacheUserProfileResponse = resp; _cacheUserProfileResponse = resp;
return resp; return resp;
} }

View File

@ -39,7 +39,7 @@ class AttachmentProvider extends GetConnect {
final client = GetConnect(); final client = GetConnect();
client.httpClient.baseUrl = ServiceFinder.services['paperclip']; client.httpClient.baseUrl = ServiceFinder.services['paperclip'];
client.httpClient.addAuthenticator(auth.reqAuthenticator); client.httpClient.addAuthenticator(auth.requestAuthenticator);
final filePayload = final filePayload =
MultipartFile(await file.readAsBytes(), filename: basename(file.path)); MultipartFile(await file.readAsBytes(), filename: basename(file.path));
@ -78,7 +78,7 @@ class AttachmentProvider extends GetConnect {
final client = GetConnect(); final client = GetConnect();
client.httpClient.baseUrl = ServiceFinder.services['paperclip']; client.httpClient.baseUrl = ServiceFinder.services['paperclip'];
client.httpClient.addAuthenticator(auth.reqAuthenticator); client.httpClient.addAuthenticator(auth.requestAuthenticator);
var resp = await client.put('/api/attachments/$id', { var resp = await client.put('/api/attachments/$id', {
'metadata': { 'metadata': {
@ -102,7 +102,7 @@ class AttachmentProvider extends GetConnect {
final client = GetConnect(); final client = GetConnect();
client.httpClient.baseUrl = ServiceFinder.services['paperclip']; client.httpClient.baseUrl = ServiceFinder.services['paperclip'];
client.httpClient.addAuthenticator(auth.reqAuthenticator); client.httpClient.addAuthenticator(auth.requestAuthenticator);
var resp = await client.delete('/api/attachments/$id'); var resp = await client.delete('/api/attachments/$id');
if (resp.statusCode != 200) { if (resp.statusCode != 200) {

View File

@ -9,7 +9,7 @@ class FriendProvider extends GetConnect {
final AuthProvider auth = Get.find(); final AuthProvider auth = Get.find();
httpClient.baseUrl = ServiceFinder.services['passport']; httpClient.baseUrl = ServiceFinder.services['passport'];
httpClient.addAuthenticator(auth.reqAuthenticator); httpClient.addAuthenticator(auth.requestAuthenticator);
} }
Future<Response> listFriendship() => get('/api/users/me/friends'); Future<Response> listFriendship() => get('/api/users/me/friends');

View File

@ -2,8 +2,6 @@ import 'package:go_router/go_router.dart';
import 'package:solian/screens/account.dart'; import 'package:solian/screens/account.dart';
import 'package:solian/screens/account/friend.dart'; import 'package:solian/screens/account/friend.dart';
import 'package:solian/screens/account/personalize.dart'; import 'package:solian/screens/account/personalize.dart';
import 'package:solian/screens/auth/signin.dart';
import 'package:solian/screens/auth/signup.dart';
import 'package:solian/screens/contact.dart'; import 'package:solian/screens/contact.dart';
import 'package:solian/screens/social.dart'; import 'package:solian/screens/social.dart';
import 'package:solian/screens/posts/publish.dart'; import 'package:solian/screens/posts/publish.dart';
@ -48,16 +46,6 @@ abstract class AppRouter {
name: 'accountPersonalize', name: 'accountPersonalize',
builder: (context, state) => const PersonalizeScreen(), builder: (context, state) => const PersonalizeScreen(),
), ),
GoRoute(
path: '/auth/sign-in',
name: 'signin',
builder: (context, state) => const SignInScreen(),
),
GoRoute(
path: '/auth/sign-up',
name: 'signup',
builder: (context, state) => const SignUpScreen(),
),
], ],
), ),
GoRoute( GoRoute(

View File

@ -2,8 +2,9 @@ import 'package:flutter/material.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:solian/providers/auth.dart'; import 'package:solian/providers/auth.dart';
import 'package:solian/router.dart'; import 'package:solian/router.dart';
import 'package:solian/screens/auth/signin.dart';
import 'package:solian/screens/auth/signup.dart';
import 'package:solian/services.dart'; import 'package:solian/services.dart';
import 'package:solian/theme.dart';
import 'package:solian/widgets/account/account_avatar.dart'; import 'package:solian/widgets/account/account_avatar.dart';
class AccountScreen extends StatefulWidget { class AccountScreen extends StatefulWidget {
@ -42,7 +43,14 @@ class _AccountScreenState extends State<AccountScreen> {
title: 'signin'.tr, title: 'signin'.tr,
caption: 'signinCaption'.tr, caption: 'signinCaption'.tr,
onTap: () { onTap: () {
AppRouter.instance.pushNamed('signin').then((_) { showModalBottomSheet(
useRootNavigator: true,
isDismissible: false,
isScrollControlled: true,
context: context,
builder: (context) => const SignInPopup(),
).then((_) async {
await provider.getProfile(noCache: true);
setState(() {}); setState(() {});
}); });
}, },
@ -52,7 +60,15 @@ class _AccountScreenState extends State<AccountScreen> {
title: 'signup'.tr, title: 'signup'.tr,
caption: 'signupCaption'.tr, caption: 'signupCaption'.tr,
onTap: () { onTap: () {
AppRouter.instance.pushNamed('signup'); showModalBottomSheet(
useRootNavigator: true,
isDismissible: false,
isScrollControlled: true,
context: context,
builder: (context) => const SignUpPopup(),
).then((_) {
setState(() {});
});
}, },
), ),
], ],
@ -69,7 +85,9 @@ class _AccountScreenState extends State<AccountScreen> {
leading: x.$1, leading: x.$1,
title: Text(x.$2), title: Text(x.$2),
onTap: () { onTap: () {
AppRouter.instance.pushNamed(x.$3); AppRouter.instance
.pushNamed(x.$3)
.then((_) => setState(() {}));
}, },
), ),
)), )),
@ -111,6 +129,8 @@ class AccountNameCard extends StatelessWidget {
children: [ children: [
AspectRatio( AspectRatio(
aspectRatio: 16 / 7, aspectRatio: 16 / 7,
child: Container(
color: Theme.of(context).colorScheme.surfaceContainer,
child: Stack( child: Stack(
clipBehavior: Clip.none, clipBehavior: Clip.none,
fit: StackFit.expand, fit: StackFit.expand,
@ -131,6 +151,7 @@ class AccountNameCard extends StatelessWidget {
], ],
), ),
), ),
),
Row( Row(
crossAxisAlignment: CrossAxisAlignment.baseline, crossAxisAlignment: CrossAxisAlignment.baseline,
textBaseline: TextBaseline.alphabetic, textBaseline: TextBaseline.alphabetic,
@ -155,9 +176,11 @@ class AccountNameCard extends StatelessWidget {
child: Card( child: Card(
child: ListTile( child: ListTile(
title: Text('description'.tr), title: Text('description'.tr),
subtitle: Text(prof.body['description']?.isNotEmpty subtitle: Text(
prof.body['description']?.isNotEmpty
? prof.body['description'] ? prof.body['description']
: 'No description yet.'), : 'No description yet.',
),
), ),
), ),
).paddingOnly(left: 24, right: 24, top: 8), ).paddingOnly(left: 24, right: 24, top: 8),

View File

@ -100,7 +100,7 @@ class _PersonalizeScreenState extends State<PersonalizeScreen> {
final client = GetConnect(); final client = GetConnect();
client.httpClient.baseUrl = ServiceFinder.services['passport']; client.httpClient.baseUrl = ServiceFinder.services['passport'];
client.httpClient.addAuthenticator(auth.reqAuthenticator); client.httpClient.addAuthenticator(auth.requestAuthenticator);
final resp = await client.put( final resp = await client.put(
'/api/users/me/$position', '/api/users/me/$position',
@ -124,7 +124,7 @@ class _PersonalizeScreenState extends State<PersonalizeScreen> {
final client = GetConnect(); final client = GetConnect();
client.httpClient.baseUrl = ServiceFinder.services['passport']; client.httpClient.baseUrl = ServiceFinder.services['passport'];
client.httpClient.addAuthenticator(auth.reqAuthenticator); client.httpClient.addAuthenticator(auth.requestAuthenticator);
_birthday?.toIso8601String(); _birthday?.toIso8601String();
final resp = await client.put( final resp = await client.put(

View File

@ -2,18 +2,17 @@ import 'package:flutter/material.dart';
import 'package:get/get.dart'; import 'package:get/get.dart';
import 'package:solian/exts.dart'; import 'package:solian/exts.dart';
import 'package:solian/providers/auth.dart'; import 'package:solian/providers/auth.dart';
import 'package:solian/router.dart';
import 'package:solian/services.dart'; import 'package:solian/services.dart';
import 'package:url_launcher/url_launcher_string.dart'; import 'package:url_launcher/url_launcher_string.dart';
class SignInScreen extends StatefulWidget { class SignInPopup extends StatefulWidget {
const SignInScreen({super.key}); const SignInPopup({super.key});
@override @override
State<SignInScreen> createState() => _SignInScreenState(); State<SignInPopup> createState() => _SignInPopupState();
} }
class _SignInScreenState extends State<SignInScreen> { class _SignInPopupState extends State<SignInPopup> {
final _usernameController = TextEditingController(); final _usernameController = TextEditingController();
final _passwordController = TextEditingController(); final _passwordController = TextEditingController();
@ -23,15 +22,16 @@ class _SignInScreenState extends State<SignInScreen> {
final username = _usernameController.value.text; final username = _usernameController.value.text;
final password = _passwordController.value.text; final password = _passwordController.value.text;
if (username.isEmpty || password.isEmpty) return; if (username.isEmpty || password.isEmpty) return;
provider.signin(context, username, password).then((_) { provider.signin(context, username, password).then((_) async {
AppRouter.instance.pop(true); Navigator.pop(context, true);
}).catchError((e) { }).catchError((e) {
List<String> messages = e.toString().split('\n'); List<String> messages = e.toString().split('\n');
if (messages.last.contains('risk')) { if (messages.last.contains('risk')) {
final ticketId = RegExp(r'ticketId=(\d+)').firstMatch(messages.last); final ticketId = RegExp(r'ticketId=(\d+)').firstMatch(messages.last);
if (ticketId == null) { if (ticketId == null) {
context.showErrorDialog( context.showErrorDialog(
'Requested to multi-factor authenticate, but the ticket id was not found'); 'Requested to multi-factor authenticate, but the ticket id was not found',
);
} }
showDialog( showDialog(
context: context, context: context,
@ -46,9 +46,7 @@ class _SignInScreenState extends State<SignInScreen> {
launchUrlString( launchUrlString(
'${ServiceFinder.services['passport']}/mfa?ticket=${ticketId!.group(1)}', '${ServiceFinder.services['passport']}/mfa?ticket=${ticketId!.group(1)}',
); );
if (Navigator.canPop(context)) {
Navigator.pop(context); Navigator.pop(context);
}
}, },
) )
], ],
@ -56,28 +54,32 @@ class _SignInScreenState extends State<SignInScreen> {
}, },
); );
} else { } else {
ScaffoldMessenger.of(context).showSnackBar(SnackBar( context.showErrorDialog(messages.last);
content: Text(messages.last),
));
} }
}); });
} }
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Material( return SizedBox(
color: Theme.of(context).colorScheme.surface, height: MediaQuery.of(context).size.height * 0.9,
child: Center( child: Center(
child: Container( child: Container(
width: MediaQuery.of(context).size.width * 0.6, width: MediaQuery.of(context).size.width * 0.6,
constraints: const BoxConstraints(maxWidth: 360), constraints: const BoxConstraints(maxWidth: 360),
child: Column( child: Column(
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
Container( Image.asset('assets/logo.png', width: 64, height: 64)
padding: const EdgeInsets.symmetric(vertical: 16), .paddingOnly(bottom: 4),
child: Image.asset('assets/logo.png', width: 72, height: 72), Text(
'signinGreeting'.tr,
style: const TextStyle(
fontSize: 28,
fontWeight: FontWeight.w900,
), ),
).paddingOnly(left: 4, bottom: 16),
TextField( TextField(
autocorrect: false, autocorrect: false,
enableSuggestions: false, enableSuggestions: false,
@ -107,10 +109,19 @@ class _SignInScreenState extends State<SignInScreen> {
FocusManager.instance.primaryFocus?.unfocus(), FocusManager.instance.primaryFocus?.unfocus(),
onSubmitted: (_) => performAction(context), onSubmitted: (_) => performAction(context),
), ),
const SizedBox(height: 16), const SizedBox(height: 12),
ElevatedButton( Align(
child: Text('signin'.tr), alignment: Alignment.centerRight,
child: TextButton(
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
Text('next'.tr),
const Icon(Icons.chevron_right),
],
),
onPressed: () => performAction(context), onPressed: () => performAction(context),
),
) )
], ],
), ),

View File

@ -4,14 +4,14 @@ import 'package:solian/exts.dart';
import 'package:solian/router.dart'; import 'package:solian/router.dart';
import 'package:solian/services.dart'; import 'package:solian/services.dart';
class SignUpScreen extends StatefulWidget { class SignUpPopup extends StatefulWidget {
const SignUpScreen({super.key}); const SignUpPopup({super.key});
@override @override
State<SignUpScreen> createState() => _SignUpScreenState(); State<SignUpPopup> createState() => _SignUpPopupState();
} }
class _SignUpScreenState extends State<SignUpScreen> { class _SignUpPopupState extends State<SignUpPopup> {
final _emailController = TextEditingController(); final _emailController = TextEditingController();
final _usernameController = TextEditingController(); final _usernameController = TextEditingController();
final _nicknameController = TextEditingController(); final _nicknameController = TextEditingController();
@ -61,19 +61,25 @@ class _SignUpScreenState extends State<SignUpScreen> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Material( return SizedBox(
color: Theme.of(context).colorScheme.surface, height: MediaQuery.of(context).size.height * 0.9,
child: Center( child: Center(
child: Container( child: Container(
width: MediaQuery.of(context).size.width * 0.6, width: MediaQuery.of(context).size.width * 0.6,
constraints: const BoxConstraints(maxWidth: 360), constraints: const BoxConstraints(maxWidth: 360),
child: Column( child: Column(
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
Container( Image.asset('assets/logo.png', width: 64, height: 64)
padding: const EdgeInsets.symmetric(vertical: 16), .paddingOnly(bottom: 4),
child: Image.asset('assets/logo.png', width: 72, height: 72), Text(
'signupGreeting'.tr,
style: const TextStyle(
fontSize: 28,
fontWeight: FontWeight.w900,
), ),
).paddingOnly(left: 4, bottom: 16),
TextField( TextField(
autocorrect: false, autocorrect: false,
enableSuggestions: false, enableSuggestions: false,
@ -132,9 +138,18 @@ class _SignUpScreenState extends State<SignUpScreen> {
onSubmitted: (_) => performAction(context), onSubmitted: (_) => performAction(context),
), ),
const SizedBox(height: 16), const SizedBox(height: 16),
ElevatedButton( Align(
child: Text('signup'.tr), alignment: Alignment.centerRight,
child: TextButton(
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
Text('next'.tr),
const Icon(Icons.chevron_right),
],
),
onPressed: () => performAction(context), onPressed: () => performAction(context),
),
) )
], ],
), ),

View File

@ -65,7 +65,7 @@ class _PostPublishingScreenState extends State<PostPublishingScreen> {
final client = GetConnect(); final client = GetConnect();
client.httpClient.baseUrl = ServiceFinder.services['interactive']; client.httpClient.baseUrl = ServiceFinder.services['interactive'];
client.httpClient.addAuthenticator(auth.reqAuthenticator); client.httpClient.addAuthenticator(auth.requestAuthenticator);
final payload = { final payload = {
'content': _contentController.value.text, 'content': _contentController.value.text,

View File

@ -43,11 +43,13 @@ class SolianMessages extends Translations {
'aspectRatioPortrait': 'Portrait', 'aspectRatioPortrait': 'Portrait',
'aspectRatioLandscape': 'Landscape', 'aspectRatioLandscape': 'Landscape',
'signin': 'Sign in', 'signin': 'Sign in',
'signinGreeting': 'Welcome back\nSolar Network',
'signinCaption': 'signinCaption':
'Sign in to create post, start a realm, message your friend and more!', 'Sign in to create post, start a realm, message your friend and more!',
'signinRiskDetected': 'signinRiskDetected':
'Risk detected, click Next to open a webpage and signin through it to pass security check.', 'Risk detected, click Next to open a webpage and signin through it to pass security check.',
'signup': 'Sign up', 'signup': 'Sign up',
'signupGreeting': 'Welcome onboard 👋',
'signupCaption': 'signupCaption':
'Create an account on Solarpass and then get the access of entire Solar Network!', 'Create an account on Solarpass and then get the access of entire Solar Network!',
'signout': 'Sign out', 'signout': 'Sign out',
@ -119,9 +121,11 @@ class SolianMessages extends Translations {
'aspectRatioPortrait': '竖型', 'aspectRatioPortrait': '竖型',
'aspectRatioLandscape': '横型', 'aspectRatioLandscape': '横型',
'signin': '登录', 'signin': '登录',
'signinGreeting': '欢迎回来\nSolar Network',
'signinCaption': '登录以发表帖子、文章、创建领域、和你的朋友聊天,以及获取更多功能!', 'signinCaption': '登录以发表帖子、文章、创建领域、和你的朋友聊天,以及获取更多功能!',
'signinRiskDetected': '检测到风险,点击下一步按钮来打开一个网页,并通过在其上面登录来通过安全检查。', 'signinRiskDetected': '检测到风险,点击下一步按钮来打开一个网页,并通过在其上面登录来通过安全检查。',
'signup': '注册', 'signup': '注册',
'signupGreeting': '欢迎加入\nSolar Network',
'signupCaption': '在 Solarpass 注册一个账号以获得整个 Solar Network 的存取权!', 'signupCaption': '在 Solarpass 注册一个账号以获得整个 Solar Network 的存取权!',
'signout': '登出', 'signout': '登出',
'riskDetection': '检测到风险', 'riskDetection': '检测到风险',

View File

@ -156,7 +156,7 @@ class _PostDeletionDialogState extends State<PostDeletionDialog> {
final client = GetConnect(); final client = GetConnect();
client.httpClient.baseUrl = ServiceFinder.services['interactive']; client.httpClient.baseUrl = ServiceFinder.services['interactive'];
client.httpClient.addAuthenticator(auth.reqAuthenticator); client.httpClient.addAuthenticator(auth.requestAuthenticator);
setState(() => _isBusy = true); setState(() => _isBusy = true);
final resp = await client.delete('/api/posts/${widget.item.id}'); final resp = await client.delete('/api/posts/${widget.item.id}');

View File

@ -50,7 +50,7 @@ class _PostQuickActionState extends State<PostQuickAction> {
final client = GetConnect(); final client = GetConnect();
client.httpClient.baseUrl = ServiceFinder.services['interactive']; client.httpClient.baseUrl = ServiceFinder.services['interactive'];
client.httpClient.addAuthenticator(auth.reqAuthenticator); client.httpClient.addAuthenticator(auth.requestAuthenticator);
setState(() => _isSubmitting = true); setState(() => _isSubmitting = true);