266 lines
11 KiB
266 lines
11 KiB
part of '../../neon.dart'; |
|
|
|
class LoginPage extends StatefulWidget { |
|
const LoginPage({ |
|
this.serverURL, |
|
super.key, |
|
}); |
|
|
|
final String? serverURL; |
|
|
|
@override |
|
_LoginPageState createState() => _LoginPageState(); |
|
} |
|
|
|
class _LoginPageState extends State<LoginPage> { |
|
WebViewController? _webViewController; |
|
final _formKey = GlobalKey<FormState>(); |
|
final _focusNode = FocusNode(); |
|
late final PackageInfo _packageInfo; |
|
late final LoginBloc _loginBloc; |
|
|
|
@override |
|
void initState() { |
|
super.initState(); |
|
|
|
_packageInfo = Provider.of<PackageInfo>(context, listen: false); |
|
_loginBloc = LoginBloc(_packageInfo); |
|
|
|
if (widget.serverURL != null) { |
|
_loginBloc.setServerURL(widget.serverURL!); |
|
} |
|
|
|
_loginBloc.loginFlowInit.listen((final init) { |
|
if (init != null && !Provider.of<NeonPlatform>(context, listen: false).canUseWebView) { |
|
WidgetsBinding.instance.addPostFrameCallback((final _) async { |
|
await launchUrlString( |
|
init.login!, |
|
mode: LaunchMode.externalApplication, |
|
webViewConfiguration: WebViewConfiguration( |
|
headers: _buildHeaders(context, Provider.of<Env?>(context, listen: false)), |
|
), |
|
); |
|
}); |
|
} |
|
}); |
|
|
|
_loginBloc.loginFlowResult.listen((final result) async { |
|
if (result != null) { |
|
final account = Account( |
|
serverURL: result.server!, |
|
username: result.loginName!, |
|
appPassword: result.appPassword!, |
|
); |
|
|
|
final accountsBloc = RxBlocProvider.of<AccountsBloc>(context); |
|
if (widget.serverURL != null) { |
|
accountsBloc.updateAccount(account); |
|
Navigator.of(context).pop(); |
|
} else { |
|
for (final a in accountsBloc.accounts.value) { |
|
if (a.id == account.id) { |
|
ExceptionWidget.showSnackbar(context, AppLocalizations.of(context).errorAccountAlreadyExists); |
|
_loginBloc.setServerURL(result.server!); |
|
return; |
|
} |
|
} |
|
accountsBloc |
|
..addAccount(account) |
|
..setActiveAccount(account); |
|
} |
|
} |
|
}); |
|
} |
|
|
|
Map<String, String> _buildHeaders( |
|
final BuildContext context, |
|
final Env? env, |
|
) => |
|
{ |
|
HttpHeaders.userAgentHeader: userAgent(_packageInfo), |
|
if (env != null) ...{ |
|
HttpHeaders.authorizationHeader: |
|
'Basic ${base64.encode(utf8.encode('${env.testUsername}:${env.testPassword}'))}', |
|
}, |
|
}; |
|
|
|
@override |
|
void dispose() { |
|
_loginBloc.dispose(); |
|
super.dispose(); |
|
} |
|
|
|
@override |
|
Widget build(final BuildContext context) { |
|
final env = Provider.of<Env?>(context); |
|
|
|
return RxBlocBuilder<AccountsBloc, List<Account>>( |
|
state: (final bloc) => bloc.accounts, |
|
builder: (final context, final accountsSnapshot, final accountsBloc) => WillPopScope( |
|
onWillPop: () async { |
|
if (accountsSnapshot.data?.isNotEmpty ?? false) { |
|
return true; |
|
} |
|
|
|
if ((await _loginBloc.serverURL.first) == null) { |
|
return true; |
|
} |
|
|
|
_loginBloc.setServerURL(null); |
|
return false; |
|
}, |
|
child: RxBlocBuilder<LoginBloc, String?>( |
|
bloc: _loginBloc, |
|
state: (final bloc) => bloc.serverURL, |
|
builder: (final context, final serverURLSnapshot, final _) => |
|
RxBlocBuilder<LoginBloc, ServerConnectionState?>( |
|
bloc: _loginBloc, |
|
state: (final bloc) => bloc.serverConnectionState, |
|
builder: (final context, final serverConnectionStateSnapshot, final _) => Scaffold( |
|
appBar: serverConnectionStateSnapshot.data == ServerConnectionState.success || |
|
(accountsSnapshot.data?.isNotEmpty ?? false) |
|
? AppBar( |
|
leading: IconButton( |
|
onPressed: () { |
|
if (accountsSnapshot.data?.isNotEmpty ?? false) { |
|
Navigator.of(context).pop(); |
|
} else { |
|
_loginBloc.setServerURL(null); |
|
} |
|
}, |
|
icon: const Icon(Icons.arrow_back), |
|
), |
|
actions: [ |
|
if (serverURLSnapshot.hasData && Provider.of<NeonPlatform>(context).canUseWebView) ...[ |
|
IconButton( |
|
onPressed: () { |
|
_loginBloc.setServerURL(serverURLSnapshot.data); |
|
}, |
|
icon: const Icon(Icons.refresh), |
|
), |
|
], |
|
], |
|
) |
|
: null, |
|
body: serverConnectionStateSnapshot.data == ServerConnectionState.success |
|
? Provider.of<NeonPlatform>(context).canUseWebView |
|
? WebView( |
|
javascriptMode: JavascriptMode.unrestricted, |
|
zoomEnabled: false, |
|
userAgent: userAgent(_packageInfo), |
|
onWebViewCreated: (final controller) async { |
|
_webViewController = controller; |
|
final url = |
|
(await _loginBloc.loginFlowInit.firstWhere((final init) => init != null))!.login!; |
|
if (mounted) { |
|
await _webViewController!.loadUrl( |
|
url, |
|
headers: _buildHeaders(context, env), |
|
); |
|
} |
|
}, |
|
onPageFinished: (final url) async { |
|
if (Provider.of<Env?>(context, listen: false) != null) { |
|
// There seems to be a bug where the first try with provided authentication header |
|
// always fails, but all the following ones work. This just restarts the flow if it sees |
|
// an error. |
|
// To be clear: This doesn't happen without pre-provided authentication, |
|
// so it's not a concern while still being a bug somewhere. |
|
final result = await _webViewController! |
|
.runJavascriptReturningResult('document.getElementsByClassName("error").length'); |
|
if (result == '1') { |
|
_loginBloc.setServerURL(serverURLSnapshot.data); |
|
} |
|
} |
|
}, |
|
) |
|
: Center( |
|
child: Column( |
|
mainAxisAlignment: MainAxisAlignment.center, |
|
children: [ |
|
Text(AppLocalizations.of(context).loginSwitchToBrowserWindow), |
|
const SizedBox( |
|
height: 10, |
|
), |
|
ElevatedButton( |
|
onPressed: () { |
|
_loginBloc.setServerURL(serverURLSnapshot.data!); |
|
}, |
|
child: Text(AppLocalizations.of(context).loginOpenAgain), |
|
), |
|
], |
|
), |
|
) |
|
: Center( |
|
child: ListView( |
|
shrinkWrap: true, |
|
padding: const EdgeInsets.symmetric(vertical: 40, horizontal: 20), |
|
children: [ |
|
SizedBox( |
|
height: MediaQuery.of(context).size.height / 2, |
|
child: Column( |
|
children: [ |
|
const NeonLogo(), |
|
const SizedBox( |
|
height: 30, |
|
), |
|
Text(AppLocalizations.of(context).loginWorksWith), |
|
const SizedBox( |
|
height: 20, |
|
), |
|
const NextcloudLogo(), |
|
], |
|
), |
|
), |
|
Form( |
|
key: _formKey, |
|
child: TextFormField( |
|
focusNode: _focusNode, |
|
decoration: const InputDecoration( |
|
hintText: 'https://...', |
|
), |
|
initialValue: |
|
widget.serverURL ?? (env?.testHost != null ? 'http://${env!.testHost}' : null), |
|
validator: (final input) => validateHttpUrl(context, input), |
|
onFieldSubmitted: (final input) { |
|
if (_formKey.currentState!.validate()) { |
|
_loginBloc.setServerURL(input); |
|
} else { |
|
_focusNode.requestFocus(); |
|
} |
|
}, |
|
), |
|
), |
|
Column( |
|
children: [ |
|
CustomLinearProgressIndicator( |
|
visible: serverConnectionStateSnapshot.data == ServerConnectionState.loading, |
|
), |
|
if (serverConnectionStateSnapshot.data == ServerConnectionState.unreachable) ...[ |
|
ExceptionWidget( |
|
AppLocalizations.of(context).errorUnableToReachServer, |
|
onRetry: () { |
|
_loginBloc.setServerURL(serverURLSnapshot.data); |
|
}, |
|
), |
|
], |
|
if (serverConnectionStateSnapshot.data == ServerConnectionState.maintenanceMode) ...[ |
|
ExceptionWidget( |
|
AppLocalizations.of(context).errorServerInMaintenanceMode, |
|
onRetry: () { |
|
_loginBloc.setServerURL(serverURLSnapshot.data); |
|
}, |
|
), |
|
], |
|
], |
|
), |
|
], |
|
), |
|
), |
|
), |
|
), |
|
), |
|
), |
|
); |
|
} |
|
}
|
|
|