import 'dart:async'; import 'dart:io'; import 'package:app_links/app_links.dart'; import 'package:ente_auth/core/configuration.dart'; import 'package:ente_auth/core/event_bus.dart'; import 'package:ente_auth/ente_theme_data.dart'; import 'package:ente_auth/events/codes_updated_event.dart'; import 'package:ente_auth/events/icons_changed_event.dart'; import 'package:ente_auth/events/trigger_logout_event.dart'; import "package:ente_auth/l10n/l10n.dart"; import 'package:ente_auth/models/code.dart'; import 'package:ente_auth/models/codes.dart'; import 'package:ente_auth/onboarding/view/setup_enter_secret_key_page.dart'; import 'package:ente_auth/services/preference_service.dart'; import 'package:ente_auth/services/user_service.dart'; import 'package:ente_auth/store/code_store.dart'; import 'package:ente_auth/ui/account/logout_dialog.dart'; import 'package:ente_auth/ui/code_widget.dart'; import 'package:ente_auth/ui/common/loading_widget.dart'; import 'package:ente_auth/ui/home/coach_mark_widget.dart'; import 'package:ente_auth/ui/home/home_empty_state.dart'; import 'package:ente_auth/ui/home/speed_dial_label_widget.dart'; import 'package:ente_auth/ui/scanner_page.dart'; import 'package:ente_auth/ui/settings_page.dart'; import 'package:ente_auth/utils/dialog_util.dart'; import 'package:ente_auth/utils/platform_util.dart'; import 'package:ente_auth/utils/totp_util.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_speed_dial/flutter_speed_dial.dart'; import 'package:flutter_staggered_grid_view/flutter_staggered_grid_view.dart'; import 'package:logging/logging.dart'; import 'package:move_to_background/move_to_background.dart'; class HomePage extends StatefulWidget { const HomePage({super.key}); @override State createState() => _HomePageState(); } class _HomePageState extends State { late final _settingsPage = SettingsPage( emailNotifier: UserService.instance.emailValueNotifier, scaffoldKey: scaffoldKey, ); bool _hasLoaded = false; bool _isSettingsOpen = false; final Logger _logger = Logger("HomePage"); final scaffoldKey = GlobalKey(); final TextEditingController _textController = TextEditingController(); final FocusNode searchInputFocusNode = FocusNode(); bool _showSearchBox = false; String _searchText = ""; Codes? _codes; List _filteredCodes = []; StreamSubscription? _streamSubscription; StreamSubscription? _triggerLogoutEvent; StreamSubscription? _iconsChangedEvent; String selectedTag = ""; @override void initState() { super.initState(); _textController.addListener(_applyFilteringAndRefresh); _loadCodes(); _streamSubscription = Bus.instance.on().listen((event) { _loadCodes(); }); _triggerLogoutEvent = Bus.instance.on().listen((event) async { await autoLogoutAlert(context); }); _initDeepLinks(); Future.delayed( const Duration(seconds: 1), () async => await CodeStore.instance.importOfflineCodes(), ); _iconsChangedEvent = Bus.instance.on().listen((event) { setState(() {}); }); _showSearchBox = PreferenceService.instance.shouldAutoFocusOnSearchBar(); if (_showSearchBox) { WidgetsBinding.instance.addPostFrameCallback( (_) { // https://github.com/flutter/flutter/issues/20706#issuecomment-646328652 FocusScope.of(context).unfocus(); Timer(const Duration(milliseconds: 1), () { FocusScope.of(context).requestFocus(searchInputFocusNode); }); }, ); } } void _loadCodes() { CodeStore.instance.getAllCodes().then((codes) { _codes = codes; _hasLoaded = true; _applyFilteringAndRefresh(); }).onError((error, stackTrace) { _logger.severe('Error while loading codes', error, stackTrace); }); } void _applyFilteringAndRefresh() { if (_searchText.isNotEmpty && _showSearchBox && _codes != null) { final String val = _searchText.toLowerCase(); // Prioritize issuer match above account for better UX while searching // for a specific TOTP for email providers. Searching for "emailProvider" like (gmail, proton) should // show the email provider first instead of other accounts where protonmail // is the account name. final List issuerMatch = []; final List accountMatch = []; for (final CodeState codeState in _codes!.allCodes) { if (codeState.error != null) continue; if (selectedTag != "" && !codeState.code!.display.tags.contains(selectedTag)) { continue; } if (codeState.code!.issuer.toLowerCase().contains(val)) { issuerMatch.add(codeState); } else if (codeState.code!.account.toLowerCase().contains(val)) { accountMatch.add(codeState); } } _filteredCodes = issuerMatch; _filteredCodes.addAll(accountMatch); } else { _filteredCodes = _codes?.allCodes .where( (element) => selectedTag == "" || element.code != null && element.code!.display.tags.contains(selectedTag), ) .toList() ?? []; } if (mounted) { setState(() {}); } } @override void dispose() { _streamSubscription?.cancel(); _triggerLogoutEvent?.cancel(); _iconsChangedEvent?.cancel(); _textController.removeListener(_applyFilteringAndRefresh); super.dispose(); } Future _redirectToScannerPage() async { final Code? code = await Navigator.of(context).push( MaterialPageRoute( builder: (BuildContext context) { return const ScannerPage(); }, ), ); if (code != null) { await CodeStore.instance.addCode(code); // Focus the new code by searching if ((_codes?.allCodes.length ?? 0) > 2) { _focusNewCode(code); } } } Future _redirectToManualEntryPage() async { final Code? code = await Navigator.of(context).push( MaterialPageRoute( builder: (BuildContext context) { return SetupEnterSecretKeyPage( tags: _codes?.tags ?? [], ); }, ), ); if (code != null) { await CodeStore.instance.addCode(code); } } @override Widget build(BuildContext context) { final l10n = context.l10n; if (!(_codes?.tags.contains(selectedTag) ?? true)) { selectedTag = ""; setState(() {}); } return PopScope( onPopInvoked: (_) async { if (_isSettingsOpen) { scaffoldKey.currentState!.closeDrawer(); return; } else if (!Platform.isAndroid) { Navigator.of(context).pop(); return; } await MoveToBackground.moveTaskToBack(); }, canPop: false, child: Scaffold( key: scaffoldKey, drawerEnableOpenDragGesture: !Platform.isAndroid, drawer: Drawer( width: 428, child: _settingsPage, ), onDrawerChanged: (isOpened) => _isSettingsOpen = isOpened, body: SafeArea( bottom: false, child: Builder( builder: (context) { return _getBody(); }, ), ), resizeToAvoidBottomInset: false, appBar: AppBar( title: !_showSearchBox ? const Text('Ente Auth') : TextField( focusNode: searchInputFocusNode, autofocus: _searchText.isEmpty, controller: _textController, onChanged: (val) { _searchText = val; _applyFilteringAndRefresh(); }, decoration: InputDecoration( hintText: l10n.searchHint, border: InputBorder.none, focusedBorder: InputBorder.none, ), ), centerTitle: true, actions: [ IconButton( icon: _showSearchBox ? const Icon(Icons.clear) : const Icon(Icons.search), tooltip: l10n.search, onPressed: () { setState( () { _showSearchBox = !_showSearchBox; if (!_showSearchBox) { _textController.clear(); _searchText = ""; } else { _searchText = _textController.text; } _applyFilteringAndRefresh(); }, ); }, ), ], ), floatingActionButton: !_hasLoaded || (_codes?.allCodes.isEmpty ?? true) || !PreferenceService.instance.hasShownCoachMark() ? null : _getFab(), ), ); } Widget _getBody() { final l10n = context.l10n; if (_hasLoaded) { if (_filteredCodes.isEmpty && _searchText.isEmpty) { return HomeEmptyStateWidget( onScanTap: _redirectToScannerPage, onManuallySetupTap: _redirectToManualEntryPage, ); } else { final list = Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ SizedBox( height: 48, child: ListView.separated( scrollDirection: Axis.horizontal, padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 2), separatorBuilder: (context, index) => const SizedBox(width: 8), itemCount: _codes?.tags == null ? 0 : _codes!.tags.length + 1, itemBuilder: (context, index) { if (index == 0) { return TagChip( label: "All", state: selectedTag == "" ? TagChipState.selected : TagChipState.unselected, onTap: () { selectedTag = ""; setState(() {}); _applyFilteringAndRefresh(); }, ); } return TagChip( label: _codes!.tags[index - 1], action: TagChipAction.menu, state: selectedTag == _codes!.tags[index - 1] ? TagChipState.selected : TagChipState.unselected, onTap: () { selectedTag = _codes!.tags[index - 1]; setState(() {}); _applyFilteringAndRefresh(); }, ); }, ), ), Expanded( child: AlignedGridView.count( crossAxisCount: (MediaQuery.sizeOf(context).width ~/ 400) .clamp(1, double.infinity) .toInt(), itemBuilder: ((context, index) { try { if (_filteredCodes[index].error != null) { return Center( child: Padding( padding: const EdgeInsets.all(8.0), child: Text( l10n.sorryUnableToGenCode( _filteredCodes[index].code?.issuer ?? "", ), ), ), ); } return ClipRect( child: CodeWidget( _filteredCodes[index].code!, _codes?.tags ?? [], ), ); } catch (e) { return const Text("Failed"); } }), itemCount: _filteredCodes.length, ), ), ], ); if (!PreferenceService.instance.hasShownCoachMark()) { return Stack( children: [ list, const CoachMarkWidget(), ], ); } else if (_showSearchBox) { return Column( children: [ Expanded( child: _filteredCodes.isNotEmpty ? AlignedGridView.count( crossAxisCount: (MediaQuery.sizeOf(context).width ~/ 400) .clamp(1, double.infinity) .toInt(), itemBuilder: ((context, index) { final codeState = _filteredCodes[index]; if (codeState.code != null) { return CodeWidget( codeState.code!, _codes?.tags ?? [], ); } else { _logger.severe( "code widget error", codeState.error, ); return Center( child: Padding( padding: const EdgeInsets.all(8.0), child: Text( l10n.sorryUnableToGenCode(""), ), ), ); } }), itemCount: _filteredCodes.length, ) : Center(child: (Text(l10n.noResult))), ), ], ); } else { return list; } } } else { return const EnteLoadingWidget(); } } Future _initDeepLinks() async { // Platform messages may fail, so we use a try/catch PlatformException. final appLinks = AppLinks(); try { String? initialLink; initialLink = await appLinks.getInitialAppLinkString(); // Parse the link and warn the user, if it is not correct, // but keep in mind it could be `null`. if (initialLink != null) { _handleDeeplink(context, initialLink); return true; } else { _logger.info("No initial link received."); } } on PlatformException { // Handle exception by warning the user their action did not succeed // return? _logger.severe("PlatformException thrown while getting initial link"); } // Attach a listener to the stream if (!kIsWeb && !Platform.isLinux) { appLinks.stringLinkStream.listen( (link) { _handleDeeplink(context, link); }, onError: (err) { _logger.severe(err); }, ); } return false; } void _handleDeeplink(BuildContext context, String? link) { if (!Configuration.instance.hasConfiguredAccount() || link == null) { return; } if (mounted && link.toLowerCase().startsWith("otpauth://")) { try { final newCode = Code.fromOTPAuthUrl(link); getNextTotp(newCode); CodeStore.instance.addCode(newCode); _focusNewCode(newCode); } catch (e, s) { showGenericErrorDialog(context: context); _logger.severe("error while handling deeplink", e, s); } } } void _focusNewCode(Code newCode) { _showSearchBox = true; _textController.text = newCode.account; _searchText = newCode.account; _applyFilteringAndRefresh(); } Widget _getFab() { if (PlatformUtil.isDesktop()) { return FloatingActionButton( onPressed: () => _redirectToManualEntryPage(), child: const Icon(Icons.add), elevation: 8.0, shape: const CircleBorder(), ); } return SpeedDial( icon: Icons.add, activeIcon: Icons.close, spacing: 3, childPadding: const EdgeInsets.all(5), spaceBetweenChildren: 4, tooltip: context.l10n.addCode, foregroundColor: Theme.of(context).colorScheme.fabForegroundColor, backgroundColor: Theme.of(context).colorScheme.fabBackgroundColor, overlayOpacity: 0.5, overlayColor: Theme.of(context).colorScheme.background, elevation: 8.0, animationCurve: Curves.elasticInOut, children: [ SpeedDialChild( child: const Icon(Icons.qr_code), foregroundColor: Theme.of(context).colorScheme.fabForegroundColor, backgroundColor: Theme.of(context).colorScheme.fabBackgroundColor, labelWidget: SpeedDialLabelWidget(context.l10n.scanAQrCode), onTap: _redirectToScannerPage, ), SpeedDialChild( child: const Icon(Icons.keyboard), foregroundColor: Theme.of(context).colorScheme.fabForegroundColor, backgroundColor: Theme.of(context).colorScheme.fabBackgroundColor, labelWidget: SpeedDialLabelWidget(context.l10n.enterDetailsManually), onTap: _redirectToManualEntryPage, ), ], ); } }