// ignore_for_file: import_of_legacy_library_into_null_safe import 'dart:async'; import 'dart:io'; 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/l10n/l10n.dart"; import 'package:ente_auth/models/code.dart'; import 'package:ente_auth/onboarding/view/setup_enter_secret_key_page.dart'; import 'package:ente_auth/services/user_service.dart'; import 'package:ente_auth/store/code_store.dart'; import 'package:ente_auth/ui/code_widget.dart'; import 'package:ente_auth/ui/common/loading_widget.dart'; import 'package:ente_auth/ui/scanner_page.dart'; import 'package:ente_auth/ui/settings_page.dart'; import 'package:flutter/material.dart'; import 'package:flutter_speed_dial/flutter_speed_dial.dart'; import 'package:move_to_background/move_to_background.dart'; class HomePage extends StatefulWidget { const HomePage({Key? key}) : super(key: key); @override State createState() => _HomePageState(); } class _HomePageState extends State { static final _settingsPage = SettingsPage( emailNotifier: UserService.instance.emailValueNotifier, ); bool _hasLoaded = false; bool _isSettingsOpen = false; List _codes = []; StreamSubscription? _streamSubscription; @override void initState() { super.initState(); _loadCodes(); _streamSubscription = Bus.instance.on().listen((event) { _loadCodes(); }); } void _loadCodes() { CodeStore.instance.getAllCodes().then((codes) { _codes = codes; _hasLoaded = true; setState(() {}); }); } @override void dispose() { _streamSubscription?.cancel(); super.dispose(); } Future _redirectToScannerPage() async { final Code? code = await Navigator.of(context).push( MaterialPageRoute( builder: (BuildContext context) { return const ScannerPage(); }, ), ); if (code != null) { CodeStore.instance.addCode(code); } } Future _redirectToManualEntryPage() async { final Code? code = await Navigator.of(context).push( MaterialPageRoute( builder: (BuildContext context) { return SetupEnterSecretKeyPage(); }, ), ); if (code != null) { CodeStore.instance.addCode(code); } } @override Widget build(BuildContext context) { return WillPopScope( onWillPop: () async { if (_isSettingsOpen) { Navigator.pop(context); return false; } if (Platform.isAndroid) { MoveToBackground.moveTaskToBack(); return false; } else { return true; } }, child: Scaffold( drawerEnableOpenDragGesture: true, drawer: ConstrainedBox( constraints: const BoxConstraints(maxWidth: 428), child: Drawer( width: double.infinity, child: _settingsPage, ), ), onDrawerChanged: (isOpened) => _isSettingsOpen = isOpened, body: SafeArea( bottom: false, child: Builder( builder: (context) { return _getBody(); }, ), ), resizeToAvoidBottomInset: false, appBar: AppBar( title: const Text('ente Authenticator'), ), floatingActionButton: !_hasLoaded || _codes.isEmpty ? null : _getFab(), ), ); } Widget _getBody() { if (_hasLoaded) { if (_codes.isEmpty) { return _getEmptyState(); } else { return ListView.builder( itemBuilder: ((context, index) { return CodeWidget(_codes[index]); }), itemCount: _codes.length, ); } } else { return const EnteLoadingWidget(); } } Widget _getFab() { return SpeedDial( icon: Icons.add, activeIcon: Icons.close, spacing: 3, childPadding: const EdgeInsets.all(5), spaceBetweenChildren: 4, tooltip: 'Add Code', 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: const SpeedDialLabelWidget("Scan a QR Code"), onTap: _redirectToScannerPage, ), SpeedDialChild( child: const Icon(Icons.keyboard), foregroundColor: Theme.of(context).colorScheme.fabForegroundColor, backgroundColor: Theme.of(context).colorScheme.fabBackgroundColor, labelWidget: const SpeedDialLabelWidget("Enter details manually"), onTap: _redirectToManualEntryPage, ), ], ); } Widget _getEmptyState() { final l10n = context.l10n; return Center( child: ConstrainedBox( constraints: const BoxConstraints.tightFor(height: 800, width: 450), child: Padding( padding: const EdgeInsets.symmetric(vertical: 40.0, horizontal: 40), child: Column( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ Column( children: [ Image.asset( "assets/wallet-front-gradient.png", width: 200, height: 200, ), Text( l10n.setupFirstAccount, textAlign: TextAlign.center, style: Theme.of(context).textTheme.headline4, ), const SizedBox(height: 64), SizedBox( width: 400, child: OutlinedButton( onPressed: _redirectToScannerPage, child: Text(l10n.importScanQrCode), ), ), const SizedBox(height: 18), SizedBox( width: 400, child: OutlinedButton( onPressed: _redirectToManualEntryPage, child: Text(l10n.importEnterSetupKey), ), ), ], ), ], ), ), ), ); } } class SpeedDialLabelWidget extends StatelessWidget { final String label; const SpeedDialLabelWidget( this.label, { Key? key, }) : super(key: key); @override Widget build(BuildContext context) { return Container( margin: const EdgeInsets.all(4), padding: const EdgeInsets.all(12), decoration: BoxDecoration( borderRadius: BorderRadius.circular(8), color: Theme.of(context).colorScheme.fabBackgroundColor, ), child: Text( label, style: TextStyle( color: Theme.of(context).colorScheme.fabForegroundColor, ), ), ); } }