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 { WebViewController? _webViewController; final _formKey = GlobalKey(); final _focusNode = FocusNode(); late final PackageInfo _packageInfo; late final LoginBloc _loginBloc; @override void initState() { super.initState(); _packageInfo = Provider.of(context, listen: false); _loginBloc = LoginBloc(_packageInfo); if (widget.serverURL != null) { _loginBloc.setServerURL(widget.serverURL!); } _loginBloc.loginFlowInit.listen((final init) { if (init != null && !Provider.of(context, listen: false).canUseWebView) { WidgetsBinding.instance.addPostFrameCallback((final _) async { await launchUrlString( init.login!, mode: LaunchMode.externalApplication, webViewConfiguration: WebViewConfiguration( headers: _buildHeaders(context, Provider.of(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(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 _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(context); return RxBlocBuilder>( 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( bloc: _loginBloc, state: (final bloc) => bloc.serverURL, builder: (final context, final serverURLSnapshot, final _) => RxBlocBuilder( 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(context).canUseWebView) ...[ IconButton( onPressed: () { _loginBloc.setServerURL(serverURLSnapshot.data); }, icon: const Icon(Icons.refresh), ), ], ], ) : null, body: serverConnectionStateSnapshot.data == ServerConnectionState.success ? Provider.of(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(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); }, ), ], ], ), ], ), ), ), ), ), ), ); } }