mirror of
https://github.com/Yubico/yubioath-flutter.git
synced 2024-12-02 05:50:34 +03:00
e57ebdf1be
Also introduces supportedAppsProvider which can be used to limit the supported apps per platform.
169 lines
4.8 KiB
Dart
Executable File
169 lines
4.8 KiB
Dart
Executable File
import 'package:flutter/material.dart';
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|
import 'package:logging/logging.dart';
|
|
import 'package:shared_preferences/shared_preferences.dart';
|
|
|
|
import '../core/state.dart';
|
|
import '../oath/menu_actions.dart';
|
|
import 'models.dart';
|
|
|
|
final _log = Logger('app.state');
|
|
|
|
// Override this to alter the set of supported apps.
|
|
final supportedAppsProvider =
|
|
Provider<List<Application>>((ref) => Application.values);
|
|
|
|
// Default implementation is always focused, override with platform specific version.
|
|
final windowStateProvider = Provider<WindowState>(
|
|
(ref) => WindowState(focused: true, visible: true, active: true),
|
|
);
|
|
|
|
final themeModeProvider = StateNotifierProvider<ThemeModeNotifier, ThemeMode>(
|
|
(ref) => ThemeModeNotifier(ref.watch(prefProvider)));
|
|
|
|
class ThemeModeNotifier extends StateNotifier<ThemeMode> {
|
|
static const String _key = 'APP_STATE_THEME';
|
|
final SharedPreferences _prefs;
|
|
ThemeModeNotifier(this._prefs) : super(_fromName(_prefs.getString(_key)));
|
|
|
|
void setThemeMode(ThemeMode mode) {
|
|
_log.config('Set theme to $mode');
|
|
state = mode;
|
|
_prefs.setString(_key, mode.name);
|
|
}
|
|
|
|
static ThemeMode _fromName(String? name) {
|
|
switch (name) {
|
|
case 'light':
|
|
return ThemeMode.light;
|
|
case 'dark':
|
|
return ThemeMode.dark;
|
|
default:
|
|
return ThemeMode.system;
|
|
}
|
|
}
|
|
}
|
|
|
|
final searchProvider =
|
|
StateNotifierProvider<SearchNotifier, String>((ref) => SearchNotifier());
|
|
|
|
class SearchNotifier extends StateNotifier<String> {
|
|
SearchNotifier() : super('');
|
|
|
|
setFilter(String value) {
|
|
state = value;
|
|
}
|
|
}
|
|
|
|
// Override with platform implementation
|
|
final attachedDevicesProvider =
|
|
StateNotifierProvider<AttachedDevicesNotifier, List<DeviceNode>>(
|
|
(ref) => AttachedDevicesNotifier([]),
|
|
);
|
|
|
|
class AttachedDevicesNotifier extends StateNotifier<List<DeviceNode>> {
|
|
AttachedDevicesNotifier(List<DeviceNode> state) : super(state);
|
|
|
|
/// Force a refresh of all device data.
|
|
void refresh() {}
|
|
}
|
|
|
|
// Override with platform implementation
|
|
final currentDeviceDataProvider = Provider<YubiKeyData?>(
|
|
(ref) => throw UnimplementedError(),
|
|
);
|
|
|
|
final currentDeviceProvider =
|
|
StateNotifierProvider<CurrentDeviceNotifier, DeviceNode?>((ref) {
|
|
final provider = CurrentDeviceNotifier(ref.watch(prefProvider));
|
|
ref.listen(attachedDevicesProvider, provider._updateAttachedDevices);
|
|
return provider;
|
|
});
|
|
|
|
class CurrentDeviceNotifier extends StateNotifier<DeviceNode?> {
|
|
static const String _lastDevice = 'APP_STATE_LAST_DEVICE';
|
|
final SharedPreferences _prefs;
|
|
CurrentDeviceNotifier(this._prefs) : super(null);
|
|
|
|
_updateAttachedDevices(List<DeviceNode>? previous, List<DeviceNode> devices) {
|
|
if (!devices.contains(state)) {
|
|
final lastDevice = _prefs.getString(_lastDevice) ?? '';
|
|
try {
|
|
state = devices.firstWhere(
|
|
(dev) => dev.when(
|
|
usbYubiKey: (path, name, pid, info) =>
|
|
lastDevice == 'serial:${info?.serial}',
|
|
nfcReader: (path, name) => lastDevice == 'name:$name',
|
|
),
|
|
orElse: () => devices.whereType<UsbYubiKeyNode>().first);
|
|
} on StateError {
|
|
state = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
setCurrentDevice(DeviceNode device) {
|
|
state = device;
|
|
device.when(
|
|
usbYubiKey: (path, name, pid, info) {
|
|
final serial = info?.serial;
|
|
if (serial != null) {
|
|
_prefs.setString(_lastDevice, 'serial:$serial');
|
|
}
|
|
},
|
|
nfcReader: (path, name) {
|
|
_prefs.setString(_lastDevice, 'name:$name');
|
|
},
|
|
);
|
|
}
|
|
}
|
|
|
|
final currentAppProvider =
|
|
StateNotifierProvider<CurrentAppNotifier, Application>((ref) {
|
|
final notifier = CurrentAppNotifier(ref.watch(supportedAppsProvider));
|
|
ref.listen<YubiKeyData?>(currentDeviceDataProvider, (_, data) {
|
|
notifier._notifyDeviceChanged(data);
|
|
}, fireImmediately: true);
|
|
return notifier;
|
|
});
|
|
|
|
class CurrentAppNotifier extends StateNotifier<Application> {
|
|
final List<Application> _supportedApps;
|
|
CurrentAppNotifier(this._supportedApps) : super(_supportedApps.first);
|
|
|
|
void setCurrentApp(Application app) {
|
|
state = app;
|
|
}
|
|
|
|
void _notifyDeviceChanged(YubiKeyData? data) {
|
|
if (data == null ||
|
|
state.getAvailability(data) != Availability.unsupported) {
|
|
// Keep current app
|
|
return;
|
|
}
|
|
|
|
state = _supportedApps.firstWhere(
|
|
(app) => app.getAvailability(data) == Availability.enabled,
|
|
orElse: () => _supportedApps.first,
|
|
);
|
|
}
|
|
}
|
|
|
|
final menuActionsProvider = Provider.autoDispose<List<MenuAction>>((ref) {
|
|
switch (ref.watch(currentAppProvider)) {
|
|
case Application.oath:
|
|
return buildOathMenuActions(ref);
|
|
// TODO: Handle other cases.
|
|
default:
|
|
return [];
|
|
}
|
|
});
|
|
|
|
abstract class QrScanner {
|
|
Future<String> scanQr();
|
|
}
|
|
|
|
final qrScannerProvider = Provider<QrScanner?>(
|
|
(ref) => null,
|
|
);
|