/* * Copyright (C) 2022 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:io'; import 'package:collection/collection.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 'package:window_manager/window_manager.dart'; import '../app/logging.dart'; import '../app/models.dart'; import '../app/state.dart'; import '../core/state.dart'; import 'models.dart'; import 'rpc.dart'; final _log = Logger('state'); // This must be initialized before use in initialize.dart. final rpcProvider = FutureProvider((ref) { throw UnimplementedError(); }); final rpcStateProvider = StateNotifierProvider<_RpcStateNotifier, RpcState>( (ref) => _RpcStateNotifier(ref.watch(rpcProvider).valueOrNull), ); class _RpcStateNotifier extends StateNotifier { final RpcSession? _rpc; _RpcStateNotifier(this._rpc) : super(const RpcState('unknown', false)) { _init(); } _init() async { final response = await _rpc?.command('get', []); if (mounted && response != null) { state = RpcState.fromJson(response['data']); } } } final desktopWindowStateProvider = StateNotifierProvider( (ref) => DesktopWindowStateNotifier(ref.watch(prefProvider))); const String windowHidden = 'DESKTOP_WINDOW_HIDDEN'; class DesktopWindowStateNotifier extends StateNotifier with WindowListener { final SharedPreferences _prefs; Timer? _idleTimer; DesktopWindowStateNotifier(this._prefs) : super(WindowState( focused: true, visible: true, active: true, hidden: _prefs.getBool(windowHidden) ?? false)) { _init(); } void _init() async { windowManager.addListener(this); // isFocused is not supported on Linux, assume focused if (!Platform.isLinux) { _idleTimer = Timer(const Duration(seconds: 5), () async { final focused = await windowManager.isFocused(); if (mounted && !focused) { state = state.copyWith(active: false); } }); } } void setWindowHidden(bool hidden) async { if (hidden) { await windowManager.hide(); } else { await windowManager.show(); } await _prefs.setBool(windowHidden, hidden); state = state.copyWith(hidden: hidden); } @override void dispose() { windowManager.removeListener(this); super.dispose(); } @override set state(WindowState value) { _log.debug('Window state changed: $value'); super.state = value; } @override @protected void onWindowEvent(String eventName) { if (mounted) { switch (eventName) { case 'blur': state = state.copyWith(focused: false); _idleTimer?.cancel(); _idleTimer = Timer(const Duration(seconds: 5), () async { if (mounted) { state = state.copyWith(active: false); } }); break; case 'focus': state = state.copyWith(focused: true, active: true); _idleTimer?.cancel(); break; case 'minimize': state = state.copyWith(visible: false, active: false); _idleTimer?.cancel(); break; case 'restore': state = state.copyWith(visible: true, active: true); _idleTimer?.cancel(); break; default: _log.traffic('Window event ignored: $eventName'); } } } } final desktopClipboardProvider = Provider( (ref) => _DesktopClipboard(), ); class _DesktopClipboard extends AppClipboard { @override bool platformGivesFeedback() { return false; } @override Future setText(String toClipboard, {bool isSensitive = false}) async { // Wayland requires the window to be focused to copy to clipboard final needsFocus = Platform.isLinux && Platform.environment['XDG_SESSION_TYPE'] == 'wayland'; var hidden = false; try { if (needsFocus && !await windowManager.isFocused()) { if (!await windowManager.isVisible()) { hidden = true; await windowManager.setOpacity(0.0); await windowManager.show(); } await windowManager.focus(); // Window focus isn't immediate, wait until focused with 10s timeout await Future.doWhile(() async => !await windowManager.isFocused()) .timeout(const Duration(seconds: 10)); } await Clipboard.setData(ClipboardData(text: toClipboard)); } finally { if (hidden) { await windowManager.hide(); await windowManager.setOpacity(1.0); } } } } final desktopSupportedThemesProvider = StateProvider>( (ref) => ThemeMode.values, ); class DesktopCurrentDeviceNotifier extends CurrentDeviceNotifier { static const String _lastDevice = 'APP_STATE_LAST_DEVICE'; @override DeviceNode? build() { SharedPreferences prefs = ref.watch(prefProvider); final devices = ref.watch(attachedDevicesProvider); final lastDevice = prefs.getString(_lastDevice) ?? ''; var node = devices.where((dev) => dev.path.key == lastDevice).firstOrNull; if (node == null) { final parts = lastDevice.split('/'); if (parts.firstOrNull == 'pid') { node = devices .whereType() .where((e) => e.pid.value.toString() == parts[1]) .firstOrNull; } } return node ?? devices.whereType().firstOrNull; } @override setCurrentDevice(DeviceNode? device) { state = device; ref.read(prefProvider).setString(_lastDevice, device?.path.key ?? ''); } } CurrentSectionNotifier desktopCurrentSectionNotifier(Ref ref) { final notifier = DesktopCurrentSectionNotifier( ref.watch(supportedSectionsProvider), ref.watch(prefProvider)); ref.listen>(currentDeviceDataProvider, (_, data) { notifier._notifyDeviceChanged(data.whenOrNull(data: ((data) => data))); }, fireImmediately: true); return notifier; } class DesktopCurrentSectionNotifier extends CurrentSectionNotifier { final List
_supportedSections; static const String _key = 'APP_STATE_LAST_SECTION'; final SharedPreferences _prefs; DesktopCurrentSectionNotifier(this._supportedSections, this._prefs) : super(_fromName(_prefs.getString(_key), _supportedSections)); @override void setCurrentSection(Section section) { state = section; _prefs.setString(_key, section.name); } void _notifyDeviceChanged(YubiKeyData? data) { if (data == null) { state = _supportedSections.first; return; } String? lastAppName = _prefs.getString(_key); if (lastAppName != null && lastAppName != state.name) { // Try switching to saved app state = Section.values.firstWhere((app) => app.name == lastAppName); } if (state == Section.passkeys && state.getAvailability(data) != Availability.enabled) { state = Section.securityKey; } if (state == Section.securityKey && state.getAvailability(data) != Availability.enabled) { state = Section.passkeys; } if (state.getAvailability(data) != Availability.unsupported) { // Keep current app return; } state = _supportedSections.firstWhere( (app) => app.getAvailability(data) == Availability.enabled, orElse: () => _supportedSections.first, ); } static Section _fromName(String? name, List
supportedSections) => supportedSections.firstWhere((element) => element.name == name, orElse: () => supportedSections.first); }