/* * Copyright (C) 2022-2024 Yubico. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import 'dart:async'; import 'dart:convert'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:logging/logging.dart'; import 'package:shared_preferences/shared_preferences.dart'; import '../app/app.dart'; import '../app/features.dart' as features; import '../app/logging.dart'; import '../app/models.dart'; import '../app/state.dart'; import '../app/views/main_page.dart'; import '../core/state.dart'; import '../fido/state.dart'; import '../management/state.dart'; import '../oath/state.dart'; import 'app_methods.dart'; import 'fido/state.dart'; import 'logger.dart'; import 'management/state.dart'; import 'oath/otp_auth_link_handler.dart'; import 'oath/state.dart'; import 'qr_scanner/qr_scanner_provider.dart'; import 'state.dart'; import 'tap_request_dialog.dart'; import 'views/nfc/nfc_activity_command_listener.dart'; import 'window_state_provider.dart'; Future initialize() async { _initSystemUi(); if (kDebugMode) { Logger.root.level = Levels.DEBUG; } _initLicenses(); return ProviderScope( overrides: [ prefProvider.overrideWithValue(await SharedPreferences.getInstance()), logLevelProvider.overrideWith((ref) => AndroidLogger()), attachedDevicesProvider.overrideWith( () => AndroidAttachedDevicesNotifier(), ), currentDeviceDataProvider.overrideWith( (ref) => ref.watch(androidDeviceDataProvider), ), oathStateProvider.overrideWithProvider(androidOathStateProvider.call), credentialListProvider .overrideWithProvider(androidCredentialListProvider.call), currentSectionProvider.overrideWith( (ref) => androidCurrentSectionNotifier(ref), ), managementStateProvider.overrideWithProvider(androidManagementState.call), currentDeviceProvider.overrideWith( () => AndroidCurrentDeviceNotifier(), ), qrScannerProvider .overrideWith(androidQrScannerProvider(await getHasCamera())), windowStateProvider .overrideWith((ref) => ref.watch(androidWindowStateProvider)), clipboardProvider.overrideWith( (ref) => ref.watch(androidClipboardProvider), ), androidSdkVersionProvider.overrideWithValue(await getAndroidSdkVersion()), androidNfcSupportProvider.overrideWithValue(await getHasNfc()), supportedSectionsProvider.overrideWithValue([ Section.home, Section.accounts, Section.fingerprints, Section.passkeys ]), // this specifies the priority of sections to show when // the connected YubiKey does not support current section androidSectionPriority.overrideWithValue( [Section.accounts, Section.fingerprints, Section.passkeys]), supportedThemesProvider.overrideWith( (ref) => ref.watch(androidSupportedThemesProvider), ), defaultColorProvider.overrideWithValue(await getPrimaryColor()), // FIDO fidoStateProvider.overrideWithProvider(androidFidoStateProvider.call), fingerprintProvider.overrideWithProvider(androidFingerprintProvider.call), credentialProvider.overrideWithProvider(androidCredentialProvider.call), ], child: DismissKeyboard( child: YubicoAuthenticatorApp(page: Consumer( builder: (context, ref, child) { ref.read(nfcEventCommandListener).startListener(context); Timer.run(() { ref.read(featureFlagProvider.notifier) // TODO: Load feature flags from file/config? //..loadConfig(config) // Disable unimplemented feature ..setFeature(features.piv, false) ..setFeature(features.otp, false) ..setFeature(features.management, false); }); // activates window state provider ref.read(androidWindowStateProvider); // initializes global handler for dialogs ref.read(androidDialogProvider); // set context which will handle otpauth links setupOtpAuthLinkHandler(context); setupAppMethodsChannel(ref); return const MainPage(); }, )), ), ); } class DismissKeyboard extends StatelessWidget { final Widget child; const DismissKeyboard({super.key, required this.child}); @override Widget build(BuildContext context) { return GestureDetector( onTap: () { // De-select any selected node when tapping outside. FocusScopeNode currentFocus = FocusScope.of(context); if (!currentFocus.hasPrimaryFocus && currentFocus.focusedChild != null) { FocusManager.instance.primaryFocus?.unfocus(); } }, child: child, ); } } void _initSystemUi() async { await SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge, overlays: SystemUiOverlay.values); } void _initLicenses() async { const licenseDir = 'assets/licenses/raw'; final androidProjectsToLicenseUrl = await rootBundle.loadStructuredData( '$licenseDir/android.json', (value) async => jsonDecode(value), ); // mapping from url to license text final fileMap = await rootBundle.loadStructuredData( '$licenseDir/map.json', (value) async => jsonDecode(value), ); final urlToLicense = {}; fileMap.forEach((url, file) async { String licenseText = url; try { licenseText = await rootBundle.loadString('$licenseDir/$file'); urlToLicense[url] = licenseText; } catch (_) { // failed to read license file, will use the url } }); if (androidProjectsToLicenseUrl.isNotEmpty) { LicenseRegistry.addLicense(() async* { for (final e in androidProjectsToLicenseUrl) { var licenseUrl = e['PackageLicense']; var content = licenseUrl; if (urlToLicense.containsKey(licenseUrl)) { content = '${urlToLicense[licenseUrl]}\n\n$licenseUrl\n\n'; } yield LicenseEntryWithLineBreaks([e['PackageName']], content); } }); } }