A framework for building convergent cross-platform Nextcloud clients using Flutter.
 
 

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);
},
),
],
],
),
],
),
),
),
),
),
),
);
}
}