refactor(legacy-settings): migrate timezone, sound, sync_clock to AsyncNotifier

This commit is contained in:
2026-04-22 00:42:33 +02:00
parent fe9476d417
commit 3b57d0e70d
24 changed files with 748 additions and 1396 deletions

View File

@@ -0,0 +1,33 @@
import 'dart:async';
import 'package:legacy_device_state/legacy_device_state.dart';
import 'package:riverpod_annotation/riverpod_annotation.dart';
import 'package:sf_shared/sf_shared.dart';
import 'package:sf_tracking/sf_tracking.dart';
part 'sound_controller.g.dart';
@riverpod
class SoundController extends _$SoundController {
@override
FutureOr<void> build() {}
Future<void> save({
required DeviceEntity device,
required String newMode,
}) async {
if (newMode == device.settings.soundMode) return;
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
final updated = device.settings.copyWith(soundMode: newMode);
await ref
.read(deviceSettingsUpdateProvider)
.updateDeviceSettings(device: device, updatedSettings: updated);
ref.syncDeviceSettings(device, updated);
unawaited(
ref.read(sfTrackingProvider).legacySettingsSoundChanged(mode: newMode),
);
});
}
}

View File

@@ -0,0 +1,55 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'sound_controller.dart';
// **************************************************************************
// RiverpodGenerator
// **************************************************************************
// GENERATED CODE - DO NOT MODIFY BY HAND
// ignore_for_file: type=lint, type=warning
@ProviderFor(SoundController)
const soundControllerProvider = SoundControllerProvider._();
final class SoundControllerProvider
extends $AsyncNotifierProvider<SoundController, void> {
const SoundControllerProvider._()
: super(
from: null,
argument: null,
retry: null,
name: r'soundControllerProvider',
isAutoDispose: true,
dependencies: null,
$allTransitiveDependencies: null,
);
@override
String debugGetCreateSourceHash() => _$soundControllerHash();
@$internal
@override
SoundController create() => SoundController();
}
String _$soundControllerHash() => r'2e33ae828acc061f6ed4bf30de1fc3b0f755c2ba';
abstract class _$SoundController extends $AsyncNotifier<void> {
FutureOr<void> build();
@$mustCallSuper
@override
void runBuild() {
build();
final ref = this.ref as $Ref<AsyncValue<void>, void>;
final element =
ref.element
as $ClassProviderElement<
AnyNotifier<AsyncValue<void>, void>,
AsyncValue<void>,
Object?,
Object?
>;
element.handleValue(ref, null);
}
}

View File

@@ -0,0 +1,11 @@
import 'package:riverpod_annotation/riverpod_annotation.dart';
part 'sound_selection_provider.g.dart';
@riverpod
class SoundSelection extends _$SoundSelection {
@override
String? build() => null;
void select(String value) => state = value;
}

View File

@@ -0,0 +1,63 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'sound_selection_provider.dart';
// **************************************************************************
// RiverpodGenerator
// **************************************************************************
// GENERATED CODE - DO NOT MODIFY BY HAND
// ignore_for_file: type=lint, type=warning
@ProviderFor(SoundSelection)
const soundSelectionProvider = SoundSelectionProvider._();
final class SoundSelectionProvider
extends $NotifierProvider<SoundSelection, String?> {
const SoundSelectionProvider._()
: super(
from: null,
argument: null,
retry: null,
name: r'soundSelectionProvider',
isAutoDispose: true,
dependencies: null,
$allTransitiveDependencies: null,
);
@override
String debugGetCreateSourceHash() => _$soundSelectionHash();
@$internal
@override
SoundSelection create() => SoundSelection();
/// {@macro riverpod.override_with_value}
Override overrideWithValue(String? value) {
return $ProviderOverride(
origin: this,
providerOverride: $SyncValueProvider<String?>(value),
);
}
}
String _$soundSelectionHash() => r'b5ba9a381457a1b10d581dedcf506a1935b47b11';
abstract class _$SoundSelection extends $Notifier<String?> {
String? build();
@$mustCallSuper
@override
void runBuild() {
final created = build();
final ref = this.ref as $Ref<String?, String?>;
final element =
ref.element
as $ClassProviderElement<
AnyNotifier<String?, String?>,
String?,
Object?,
Object?
>;
element.handleValue(ref, created);
}
}

View File

@@ -1,13 +1,14 @@
import 'package:design_system/design_system.dart';
import 'package:legacy_theme/legacy_theme.dart';
import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:legacy_device_state/legacy_device_state.dart';
import 'package:legacy_theme/legacy_theme.dart';
import 'package:legacy_ui/legacy_ui.dart';
import 'package:navigation/navigation.dart';
import 'package:settings/src/features/sound/presentation/providers/sound_controller.dart';
import 'package:settings/src/features/sound/presentation/providers/sound_selection_provider.dart';
import 'package:sf_localizations/sf_localizations.dart';
import 'state/sound_view_model.dart';
import 'package:sf_shared/sf_shared.dart';
class SoundScreen extends ConsumerWidget {
final NavigationContract navigationContract;
@@ -16,26 +17,20 @@ class SoundScreen extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
ref.listen(soundViewModelProvider.select((s) => s.errorMessage), (
_,
errorMessage,
) {
if (errorMessage.isNotEmpty) {
showTopSnackbar(
context,
message: errorMessage,
type: MessageType.error,
);
ref.listen(soundControllerProvider, (prev, next) async {
next.showErrorOn(context);
if (prev != null &&
prev.isLoading &&
!next.isLoading &&
!next.hasError) {
await showSuccessDialog(context, I18n.deviceUpdatedSuccess);
if (context.mounted) navigationContract.goBack();
}
});
ref.listen(soundViewModelProvider.select((s) => s.isComplete), (
_,
isComplete,
) {
if (isComplete) Navigator.pop(context);
});
final device = ref.watch(selectedDeviceProvider).value;
final currentMode = device?.settings.soundMode ?? '';
final selectedMode = ref.watch(soundSelectionProvider) ?? currentMode;
return LegacyPageLayout(
title: context.translate(I18n.sound),
@@ -51,25 +46,23 @@ class SoundScreen extends ConsumerWidget {
),
),
const SizedBox(height: 36),
const _OptionsSection(),
_OptionsSection(selectedMode: selectedMode),
],
),
),
footer: const _SaveSection(),
footer: _SaveSection(device: device, selectedMode: selectedMode),
);
}
}
class _OptionsSection extends ConsumerWidget {
const _OptionsSection();
final String selectedMode;
const _OptionsSection({required this.selectedMode});
@override
Widget build(BuildContext context, WidgetRef ref) {
final vm = ref.read(soundViewModelProvider.notifier);
final soundOption = ref.watch(
soundViewModelProvider.select((s) => s.soundOption),
);
final select = ref.read(soundSelectionProvider.notifier).select;
return SingleChildScrollView(
child: Column(
@@ -77,37 +70,29 @@ class _OptionsSection extends ConsumerWidget {
_SectionButton(
title: context.translate(I18n.soundAndVibration),
icon: Icons.volume_up_outlined,
active: soundOption == 'VIBRATION_AND_RINGING',
onPressed: () {
vm.setSoundOption('VIBRATION_AND_RINGING');
},
active: selectedMode == 'VIBRATION_AND_RINGING',
onPressed: () => select('VIBRATION_AND_RINGING'),
),
SizedBox(height: 12),
const SizedBox(height: 12),
_SectionButton(
title: context.translate(I18n.soundOnly),
icon: Icons.volume_up_outlined,
active: soundOption == 'RINGING',
onPressed: () {
vm.setSoundOption('RINGING');
},
active: selectedMode == 'RINGING',
onPressed: () => select('RINGING'),
),
SizedBox(height: 12),
const SizedBox(height: 12),
_SectionButton(
title: context.translate(I18n.vibrationOnly),
icon: Icons.vibration_outlined,
active: soundOption == 'VIBRATION',
onPressed: () {
vm.setSoundOption('VIBRATION');
},
active: selectedMode == 'VIBRATION',
onPressed: () => select('VIBRATION'),
),
SizedBox(height: 12),
const SizedBox(height: 12),
_SectionButton(
title: context.translate(I18n.silent),
icon: Icons.volume_mute_outlined,
active: soundOption == 'SILENCE',
onPressed: () {
vm.setSoundOption('SILENCE');
},
active: selectedMode == 'SILENCE',
onPressed: () => select('SILENCE'),
),
],
),
@@ -115,7 +100,7 @@ class _OptionsSection extends ConsumerWidget {
}
}
class _SectionButton extends ConsumerWidget {
class _SectionButton extends StatelessWidget {
final String title;
final bool active;
final IconData icon;
@@ -129,15 +114,10 @@ class _SectionButton extends ConsumerWidget {
});
@override
Widget build(BuildContext context, WidgetRef ref) {
Widget build(BuildContext context) {
return SectionButton(
onPressed: onPressed,
icon: Icon(
icon,
color: context.sfColors.legacyPrimary,
size: 36,
),
icon: Icon(icon, color: context.sfColors.legacyPrimary, size: 36),
iconPadding: 8,
body: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
@@ -152,12 +132,7 @@ class _SectionButton extends ConsumerWidget {
: Theme.of(context).colorScheme.outline,
),
),
Switch(
value: active,
onChanged: (_) {
onPressed();
},
),
Switch(value: active, onChanged: (_) => onPressed()),
],
),
);
@@ -165,20 +140,31 @@ class _SectionButton extends ConsumerWidget {
}
class _SaveSection extends ConsumerWidget {
const _SaveSection();
final DeviceEntity? device;
final String selectedMode;
const _SaveSection({required this.device, required this.selectedMode});
@override
Widget build(BuildContext context, WidgetRef ref) {
final vm = ref.read(soundViewModelProvider.notifier);
final isLoading = ref.watch(
soundControllerProvider.select((s) => s.isLoading),
);
return Padding(
padding: EdgeInsets.symmetric(horizontal: 24, vertical: 12),
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 12),
child: PrimaryButton(
onPressed: () async {
if (!await guardDeviceCommand(context, ref)) return;
vm.submit();
},
onPressed: isLoading
? null
: () async {
if (device == null) return;
if (!await guardDeviceCommand(context, ref)) return;
if (!context.mounted) return;
ref.read(soundControllerProvider.notifier).save(
device: device!,
newMode: selectedMode,
);
},
text: context.translate(I18n.save),
color: context.sfColors.legacyPrimary,
),

View File

@@ -1,81 +0,0 @@
import 'dart:async';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:legacy_device_state/legacy_device_state.dart';
import 'package:sf_shared/sf_shared.dart';
import 'package:sf_tracking/sf_tracking.dart';
import 'sound_view_state.dart';
final soundViewModelProvider =
NotifierProvider.autoDispose<SoundViewModel, SoundViewState>(
SoundViewModel.new,
);
class SoundViewModel extends Notifier<SoundViewState> {
late final DeviceSettingsUpdateDatasource _datasource;
late final SfTrackingRepository _tracking;
@override
SoundViewState build() {
_datasource = ref.read(deviceSettingsUpdateProvider);
_tracking = ref.read(sfTrackingProvider);
Future.microtask(() => load());
return const SoundViewState();
}
Future<void> load() async {
final device = ref.read(selectedDeviceProvider).value;
if (device == null) return;
state = state.copyWith(
device: device,
soundOption: device.settings.soundMode,
isLoading: false,
);
}
void setSoundOption(String value) {
if (state.soundOption == value) return;
state = state.copyWith(soundOption: value);
}
Future<void> submit() async {
final device = state.device;
if (device == null) return;
if (state.soundOption == device.settings.soundMode) {
state = state.copyWith(isComplete: true);
return;
}
try {
state = state.copyWith(
isLoading: true,
isComplete: false,
errorMessage: '',
);
final updatedSettings = device.settings.copyWith(
soundMode: state.soundOption,
);
await _datasource.updateDeviceSettings(
device: device,
updatedSettings: updatedSettings,
);
if (!ref.mounted) return;
ref.syncDeviceSettings(device, updatedSettings);
unawaited(
_tracking.legacySettingsSoundChanged(mode: state.soundOption ?? ''),
);
state = state.copyWith(isLoading: false, isComplete: true);
} catch (e) {
if (!ref.mounted) return;
state = state.copyWith(isLoading: false, errorMessage: e.toString());
}
}
}

View File

@@ -1,15 +0,0 @@
import 'package:freezed_annotation/freezed_annotation.dart';
import 'package:sf_shared/sf_shared.dart';
part 'sound_view_state.freezed.dart';
@freezed
abstract class SoundViewState with _$SoundViewState {
const factory SoundViewState({
DeviceEntity? device,
String? soundOption,
@Default(true) bool isLoading,
@Default(false) bool isComplete,
@Default('') String errorMessage,
}) = _SoundViewState;
}

View File

@@ -1,307 +0,0 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
// coverage:ignore-file
// ignore_for_file: type=lint
// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark
part of 'sound_view_state.dart';
// **************************************************************************
// FreezedGenerator
// **************************************************************************
// dart format off
T _$identity<T>(T value) => value;
/// @nodoc
mixin _$SoundViewState {
DeviceEntity? get device; String? get soundOption; bool get isLoading; bool get isComplete; String get errorMessage;
/// Create a copy of SoundViewState
/// with the given fields replaced by the non-null parameter values.
@JsonKey(includeFromJson: false, includeToJson: false)
@pragma('vm:prefer-inline')
$SoundViewStateCopyWith<SoundViewState> get copyWith => _$SoundViewStateCopyWithImpl<SoundViewState>(this as SoundViewState, _$identity);
@override
bool operator ==(Object other) {
return identical(this, other) || (other.runtimeType == runtimeType&&other is SoundViewState&&(identical(other.device, device) || other.device == device)&&(identical(other.soundOption, soundOption) || other.soundOption == soundOption)&&(identical(other.isLoading, isLoading) || other.isLoading == isLoading)&&(identical(other.isComplete, isComplete) || other.isComplete == isComplete)&&(identical(other.errorMessage, errorMessage) || other.errorMessage == errorMessage));
}
@override
int get hashCode => Object.hash(runtimeType,device,soundOption,isLoading,isComplete,errorMessage);
@override
String toString() {
return 'SoundViewState(device: $device, soundOption: $soundOption, isLoading: $isLoading, isComplete: $isComplete, errorMessage: $errorMessage)';
}
}
/// @nodoc
abstract mixin class $SoundViewStateCopyWith<$Res> {
factory $SoundViewStateCopyWith(SoundViewState value, $Res Function(SoundViewState) _then) = _$SoundViewStateCopyWithImpl;
@useResult
$Res call({
DeviceEntity? device, String? soundOption, bool isLoading, bool isComplete, String errorMessage
});
$DeviceEntityCopyWith<$Res>? get device;
}
/// @nodoc
class _$SoundViewStateCopyWithImpl<$Res>
implements $SoundViewStateCopyWith<$Res> {
_$SoundViewStateCopyWithImpl(this._self, this._then);
final SoundViewState _self;
final $Res Function(SoundViewState) _then;
/// Create a copy of SoundViewState
/// with the given fields replaced by the non-null parameter values.
@pragma('vm:prefer-inline') @override $Res call({Object? device = freezed,Object? soundOption = freezed,Object? isLoading = null,Object? isComplete = null,Object? errorMessage = null,}) {
return _then(_self.copyWith(
device: freezed == device ? _self.device : device // ignore: cast_nullable_to_non_nullable
as DeviceEntity?,soundOption: freezed == soundOption ? _self.soundOption : soundOption // ignore: cast_nullable_to_non_nullable
as String?,isLoading: null == isLoading ? _self.isLoading : isLoading // ignore: cast_nullable_to_non_nullable
as bool,isComplete: null == isComplete ? _self.isComplete : isComplete // ignore: cast_nullable_to_non_nullable
as bool,errorMessage: null == errorMessage ? _self.errorMessage : errorMessage // ignore: cast_nullable_to_non_nullable
as String,
));
}
/// Create a copy of SoundViewState
/// with the given fields replaced by the non-null parameter values.
@override
@pragma('vm:prefer-inline')
$DeviceEntityCopyWith<$Res>? get device {
if (_self.device == null) {
return null;
}
return $DeviceEntityCopyWith<$Res>(_self.device!, (value) {
return _then(_self.copyWith(device: value));
});
}
}
/// Adds pattern-matching-related methods to [SoundViewState].
extension SoundViewStatePatterns on SoundViewState {
/// A variant of `map` that fallback to returning `orElse`.
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case final Subclass value:
/// return ...;
/// case _:
/// return orElse();
/// }
/// ```
@optionalTypeArgs TResult maybeMap<TResult extends Object?>(TResult Function( _SoundViewState value)? $default,{required TResult orElse(),}){
final _that = this;
switch (_that) {
case _SoundViewState() when $default != null:
return $default(_that);case _:
return orElse();
}
}
/// A `switch`-like method, using callbacks.
///
/// Callbacks receives the raw object, upcasted.
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case final Subclass value:
/// return ...;
/// case final Subclass2 value:
/// return ...;
/// }
/// ```
@optionalTypeArgs TResult map<TResult extends Object?>(TResult Function( _SoundViewState value) $default,){
final _that = this;
switch (_that) {
case _SoundViewState():
return $default(_that);case _:
throw StateError('Unexpected subclass');
}
}
/// A variant of `map` that fallback to returning `null`.
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case final Subclass value:
/// return ...;
/// case _:
/// return null;
/// }
/// ```
@optionalTypeArgs TResult? mapOrNull<TResult extends Object?>(TResult? Function( _SoundViewState value)? $default,){
final _that = this;
switch (_that) {
case _SoundViewState() when $default != null:
return $default(_that);case _:
return null;
}
}
/// A variant of `when` that fallback to an `orElse` callback.
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case Subclass(:final field):
/// return ...;
/// case _:
/// return orElse();
/// }
/// ```
@optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( DeviceEntity? device, String? soundOption, bool isLoading, bool isComplete, String errorMessage)? $default,{required TResult orElse(),}) {final _that = this;
switch (_that) {
case _SoundViewState() when $default != null:
return $default(_that.device,_that.soundOption,_that.isLoading,_that.isComplete,_that.errorMessage);case _:
return orElse();
}
}
/// A `switch`-like method, using callbacks.
///
/// As opposed to `map`, this offers destructuring.
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case Subclass(:final field):
/// return ...;
/// case Subclass2(:final field2):
/// return ...;
/// }
/// ```
@optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( DeviceEntity? device, String? soundOption, bool isLoading, bool isComplete, String errorMessage) $default,) {final _that = this;
switch (_that) {
case _SoundViewState():
return $default(_that.device,_that.soundOption,_that.isLoading,_that.isComplete,_that.errorMessage);case _:
throw StateError('Unexpected subclass');
}
}
/// A variant of `when` that fallback to returning `null`
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case Subclass(:final field):
/// return ...;
/// case _:
/// return null;
/// }
/// ```
@optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( DeviceEntity? device, String? soundOption, bool isLoading, bool isComplete, String errorMessage)? $default,) {final _that = this;
switch (_that) {
case _SoundViewState() when $default != null:
return $default(_that.device,_that.soundOption,_that.isLoading,_that.isComplete,_that.errorMessage);case _:
return null;
}
}
}
/// @nodoc
class _SoundViewState implements SoundViewState {
const _SoundViewState({this.device, this.soundOption, this.isLoading = true, this.isComplete = false, this.errorMessage = ''});
@override final DeviceEntity? device;
@override final String? soundOption;
@override@JsonKey() final bool isLoading;
@override@JsonKey() final bool isComplete;
@override@JsonKey() final String errorMessage;
/// Create a copy of SoundViewState
/// with the given fields replaced by the non-null parameter values.
@override @JsonKey(includeFromJson: false, includeToJson: false)
@pragma('vm:prefer-inline')
_$SoundViewStateCopyWith<_SoundViewState> get copyWith => __$SoundViewStateCopyWithImpl<_SoundViewState>(this, _$identity);
@override
bool operator ==(Object other) {
return identical(this, other) || (other.runtimeType == runtimeType&&other is _SoundViewState&&(identical(other.device, device) || other.device == device)&&(identical(other.soundOption, soundOption) || other.soundOption == soundOption)&&(identical(other.isLoading, isLoading) || other.isLoading == isLoading)&&(identical(other.isComplete, isComplete) || other.isComplete == isComplete)&&(identical(other.errorMessage, errorMessage) || other.errorMessage == errorMessage));
}
@override
int get hashCode => Object.hash(runtimeType,device,soundOption,isLoading,isComplete,errorMessage);
@override
String toString() {
return 'SoundViewState(device: $device, soundOption: $soundOption, isLoading: $isLoading, isComplete: $isComplete, errorMessage: $errorMessage)';
}
}
/// @nodoc
abstract mixin class _$SoundViewStateCopyWith<$Res> implements $SoundViewStateCopyWith<$Res> {
factory _$SoundViewStateCopyWith(_SoundViewState value, $Res Function(_SoundViewState) _then) = __$SoundViewStateCopyWithImpl;
@override @useResult
$Res call({
DeviceEntity? device, String? soundOption, bool isLoading, bool isComplete, String errorMessage
});
@override $DeviceEntityCopyWith<$Res>? get device;
}
/// @nodoc
class __$SoundViewStateCopyWithImpl<$Res>
implements _$SoundViewStateCopyWith<$Res> {
__$SoundViewStateCopyWithImpl(this._self, this._then);
final _SoundViewState _self;
final $Res Function(_SoundViewState) _then;
/// Create a copy of SoundViewState
/// with the given fields replaced by the non-null parameter values.
@override @pragma('vm:prefer-inline') $Res call({Object? device = freezed,Object? soundOption = freezed,Object? isLoading = null,Object? isComplete = null,Object? errorMessage = null,}) {
return _then(_SoundViewState(
device: freezed == device ? _self.device : device // ignore: cast_nullable_to_non_nullable
as DeviceEntity?,soundOption: freezed == soundOption ? _self.soundOption : soundOption // ignore: cast_nullable_to_non_nullable
as String?,isLoading: null == isLoading ? _self.isLoading : isLoading // ignore: cast_nullable_to_non_nullable
as bool,isComplete: null == isComplete ? _self.isComplete : isComplete // ignore: cast_nullable_to_non_nullable
as bool,errorMessage: null == errorMessage ? _self.errorMessage : errorMessage // ignore: cast_nullable_to_non_nullable
as String,
));
}
/// Create a copy of SoundViewState
/// with the given fields replaced by the non-null parameter values.
@override
@pragma('vm:prefer-inline')
$DeviceEntityCopyWith<$Res>? get device {
if (_self.device == null) {
return null;
}
return $DeviceEntityCopyWith<$Res>(_self.device!, (value) {
return _then(_self.copyWith(device: value));
});
}
}
// dart format on

View File

@@ -0,0 +1,21 @@
import 'dart:async';
import 'package:riverpod_annotation/riverpod_annotation.dart';
import 'package:sf_tracking/sf_tracking.dart';
part 'sync_clock_controller.g.dart';
@riverpod
class SyncClockController extends _$SyncClockController {
@override
FutureOr<void> build() {}
Future<void> sync() async {
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
unawaited(
ref.read(sfTrackingProvider).legacySettingsSyncClockTriggered(),
);
});
}
}

View File

@@ -0,0 +1,56 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'sync_clock_controller.dart';
// **************************************************************************
// RiverpodGenerator
// **************************************************************************
// GENERATED CODE - DO NOT MODIFY BY HAND
// ignore_for_file: type=lint, type=warning
@ProviderFor(SyncClockController)
const syncClockControllerProvider = SyncClockControllerProvider._();
final class SyncClockControllerProvider
extends $AsyncNotifierProvider<SyncClockController, void> {
const SyncClockControllerProvider._()
: super(
from: null,
argument: null,
retry: null,
name: r'syncClockControllerProvider',
isAutoDispose: true,
dependencies: null,
$allTransitiveDependencies: null,
);
@override
String debugGetCreateSourceHash() => _$syncClockControllerHash();
@$internal
@override
SyncClockController create() => SyncClockController();
}
String _$syncClockControllerHash() =>
r'4969dcad124bfea5ea7d704951b972b914dd31db';
abstract class _$SyncClockController extends $AsyncNotifier<void> {
FutureOr<void> build();
@$mustCallSuper
@override
void runBuild() {
build();
final ref = this.ref as $Ref<AsyncValue<void>, void>;
final element =
ref.element
as $ClassProviderElement<
AnyNotifier<AsyncValue<void>, void>,
AsyncValue<void>,
Object?,
Object?
>;
element.handleValue(ref, null);
}
}

View File

@@ -1,46 +0,0 @@
import 'dart:async';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:sf_shared/sf_shared.dart';
import 'package:sf_tracking/sf_tracking.dart';
import 'sync_clock_view_state.dart';
final syncClockViewModelProvider =
NotifierProvider.autoDispose<SyncClockViewModel, SyncClockViewState>(
SyncClockViewModel.new,
);
class SyncClockViewModel extends Notifier<SyncClockViewState> {
late final SfTrackingRepository _tracking;
@override
SyncClockViewState build() {
_tracking = ref.read(sfTrackingProvider);
Future.microtask(() => load());
return const SyncClockViewState();
}
Future<void> load() async {
final device = ref.read(selectedDeviceProvider).value;
setDevice(device!);
}
void setDevice(DeviceEntity device) {
state = state.copyWith(deviceId: device.identificator, isLoading: false);
}
Future<void> syncClock() async {
if (state.isLoading) return;
try {
state = state.copyWith(isLoading: true);
// Stub: real call is ref.read(settingsRepositoryProvider).syncClock(deviceId: state.deviceId)
unawaited(_tracking.legacySettingsSyncClockTriggered());
} catch (e) {
state = state.copyWith(isLoading: false, errorMessage: e.toString());
}
}
}

View File

@@ -1,12 +0,0 @@
import 'package:freezed_annotation/freezed_annotation.dart';
part 'sync_clock_view_state.freezed.dart';
@freezed
abstract class SyncClockViewState with _$SyncClockViewState {
const factory SyncClockViewState({
@Default('') String deviceId,
@Default(true) bool isLoading,
@Default('') String errorMessage,
}) = _SyncClockViewState;
}

View File

@@ -1,277 +0,0 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
// coverage:ignore-file
// ignore_for_file: type=lint
// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark
part of 'sync_clock_view_state.dart';
// **************************************************************************
// FreezedGenerator
// **************************************************************************
// dart format off
T _$identity<T>(T value) => value;
/// @nodoc
mixin _$SyncClockViewState {
String get deviceId; bool get isLoading; String get errorMessage;
/// Create a copy of SyncClockViewState
/// with the given fields replaced by the non-null parameter values.
@JsonKey(includeFromJson: false, includeToJson: false)
@pragma('vm:prefer-inline')
$SyncClockViewStateCopyWith<SyncClockViewState> get copyWith => _$SyncClockViewStateCopyWithImpl<SyncClockViewState>(this as SyncClockViewState, _$identity);
@override
bool operator ==(Object other) {
return identical(this, other) || (other.runtimeType == runtimeType&&other is SyncClockViewState&&(identical(other.deviceId, deviceId) || other.deviceId == deviceId)&&(identical(other.isLoading, isLoading) || other.isLoading == isLoading)&&(identical(other.errorMessage, errorMessage) || other.errorMessage == errorMessage));
}
@override
int get hashCode => Object.hash(runtimeType,deviceId,isLoading,errorMessage);
@override
String toString() {
return 'SyncClockViewState(deviceId: $deviceId, isLoading: $isLoading, errorMessage: $errorMessage)';
}
}
/// @nodoc
abstract mixin class $SyncClockViewStateCopyWith<$Res> {
factory $SyncClockViewStateCopyWith(SyncClockViewState value, $Res Function(SyncClockViewState) _then) = _$SyncClockViewStateCopyWithImpl;
@useResult
$Res call({
String deviceId, bool isLoading, String errorMessage
});
}
/// @nodoc
class _$SyncClockViewStateCopyWithImpl<$Res>
implements $SyncClockViewStateCopyWith<$Res> {
_$SyncClockViewStateCopyWithImpl(this._self, this._then);
final SyncClockViewState _self;
final $Res Function(SyncClockViewState) _then;
/// Create a copy of SyncClockViewState
/// with the given fields replaced by the non-null parameter values.
@pragma('vm:prefer-inline') @override $Res call({Object? deviceId = null,Object? isLoading = null,Object? errorMessage = null,}) {
return _then(_self.copyWith(
deviceId: null == deviceId ? _self.deviceId : deviceId // ignore: cast_nullable_to_non_nullable
as String,isLoading: null == isLoading ? _self.isLoading : isLoading // ignore: cast_nullable_to_non_nullable
as bool,errorMessage: null == errorMessage ? _self.errorMessage : errorMessage // ignore: cast_nullable_to_non_nullable
as String,
));
}
}
/// Adds pattern-matching-related methods to [SyncClockViewState].
extension SyncClockViewStatePatterns on SyncClockViewState {
/// A variant of `map` that fallback to returning `orElse`.
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case final Subclass value:
/// return ...;
/// case _:
/// return orElse();
/// }
/// ```
@optionalTypeArgs TResult maybeMap<TResult extends Object?>(TResult Function( _SyncClockViewState value)? $default,{required TResult orElse(),}){
final _that = this;
switch (_that) {
case _SyncClockViewState() when $default != null:
return $default(_that);case _:
return orElse();
}
}
/// A `switch`-like method, using callbacks.
///
/// Callbacks receives the raw object, upcasted.
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case final Subclass value:
/// return ...;
/// case final Subclass2 value:
/// return ...;
/// }
/// ```
@optionalTypeArgs TResult map<TResult extends Object?>(TResult Function( _SyncClockViewState value) $default,){
final _that = this;
switch (_that) {
case _SyncClockViewState():
return $default(_that);case _:
throw StateError('Unexpected subclass');
}
}
/// A variant of `map` that fallback to returning `null`.
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case final Subclass value:
/// return ...;
/// case _:
/// return null;
/// }
/// ```
@optionalTypeArgs TResult? mapOrNull<TResult extends Object?>(TResult? Function( _SyncClockViewState value)? $default,){
final _that = this;
switch (_that) {
case _SyncClockViewState() when $default != null:
return $default(_that);case _:
return null;
}
}
/// A variant of `when` that fallback to an `orElse` callback.
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case Subclass(:final field):
/// return ...;
/// case _:
/// return orElse();
/// }
/// ```
@optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( String deviceId, bool isLoading, String errorMessage)? $default,{required TResult orElse(),}) {final _that = this;
switch (_that) {
case _SyncClockViewState() when $default != null:
return $default(_that.deviceId,_that.isLoading,_that.errorMessage);case _:
return orElse();
}
}
/// A `switch`-like method, using callbacks.
///
/// As opposed to `map`, this offers destructuring.
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case Subclass(:final field):
/// return ...;
/// case Subclass2(:final field2):
/// return ...;
/// }
/// ```
@optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( String deviceId, bool isLoading, String errorMessage) $default,) {final _that = this;
switch (_that) {
case _SyncClockViewState():
return $default(_that.deviceId,_that.isLoading,_that.errorMessage);case _:
throw StateError('Unexpected subclass');
}
}
/// A variant of `when` that fallback to returning `null`
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case Subclass(:final field):
/// return ...;
/// case _:
/// return null;
/// }
/// ```
@optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( String deviceId, bool isLoading, String errorMessage)? $default,) {final _that = this;
switch (_that) {
case _SyncClockViewState() when $default != null:
return $default(_that.deviceId,_that.isLoading,_that.errorMessage);case _:
return null;
}
}
}
/// @nodoc
class _SyncClockViewState implements SyncClockViewState {
const _SyncClockViewState({this.deviceId = '', this.isLoading = true, this.errorMessage = ''});
@override@JsonKey() final String deviceId;
@override@JsonKey() final bool isLoading;
@override@JsonKey() final String errorMessage;
/// Create a copy of SyncClockViewState
/// with the given fields replaced by the non-null parameter values.
@override @JsonKey(includeFromJson: false, includeToJson: false)
@pragma('vm:prefer-inline')
_$SyncClockViewStateCopyWith<_SyncClockViewState> get copyWith => __$SyncClockViewStateCopyWithImpl<_SyncClockViewState>(this, _$identity);
@override
bool operator ==(Object other) {
return identical(this, other) || (other.runtimeType == runtimeType&&other is _SyncClockViewState&&(identical(other.deviceId, deviceId) || other.deviceId == deviceId)&&(identical(other.isLoading, isLoading) || other.isLoading == isLoading)&&(identical(other.errorMessage, errorMessage) || other.errorMessage == errorMessage));
}
@override
int get hashCode => Object.hash(runtimeType,deviceId,isLoading,errorMessage);
@override
String toString() {
return 'SyncClockViewState(deviceId: $deviceId, isLoading: $isLoading, errorMessage: $errorMessage)';
}
}
/// @nodoc
abstract mixin class _$SyncClockViewStateCopyWith<$Res> implements $SyncClockViewStateCopyWith<$Res> {
factory _$SyncClockViewStateCopyWith(_SyncClockViewState value, $Res Function(_SyncClockViewState) _then) = __$SyncClockViewStateCopyWithImpl;
@override @useResult
$Res call({
String deviceId, bool isLoading, String errorMessage
});
}
/// @nodoc
class __$SyncClockViewStateCopyWithImpl<$Res>
implements _$SyncClockViewStateCopyWith<$Res> {
__$SyncClockViewStateCopyWithImpl(this._self, this._then);
final _SyncClockViewState _self;
final $Res Function(_SyncClockViewState) _then;
/// Create a copy of SyncClockViewState
/// with the given fields replaced by the non-null parameter values.
@override @pragma('vm:prefer-inline') $Res call({Object? deviceId = null,Object? isLoading = null,Object? errorMessage = null,}) {
return _then(_SyncClockViewState(
deviceId: null == deviceId ? _self.deviceId : deviceId // ignore: cast_nullable_to_non_nullable
as String,isLoading: null == isLoading ? _self.isLoading : isLoading // ignore: cast_nullable_to_non_nullable
as bool,errorMessage: null == errorMessage ? _self.errorMessage : errorMessage // ignore: cast_nullable_to_non_nullable
as String,
));
}
}
// dart format on

View File

@@ -1,12 +1,12 @@
import 'package:design_system/design_system.dart';
import 'package:legacy_theme/legacy_theme.dart';
import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:legacy_theme/legacy_theme.dart';
import 'package:legacy_ui/legacy_ui.dart';
import 'package:navigation/navigation.dart';
import 'package:settings/src/features/sync_clock/presentation/providers/sync_clock_controller.dart';
import 'package:sf_localizations/sf_localizations.dart';
import '../../sound/presentation/state/sound_view_model.dart';
import 'package:sf_shared/sf_shared.dart';
class SyncClockScreen extends ConsumerWidget {
final NavigationContract navigationContract;
@@ -15,142 +15,40 @@ class SyncClockScreen extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
ref.listen(syncClockControllerProvider, (prev, next) async {
next.showErrorOn(context);
if (prev != null &&
prev.isLoading &&
!next.isLoading &&
!next.hasError) {
await showSuccessDialog(context, I18n.deviceUpdatedSuccess);
}
});
final isLoading = ref.watch(
syncClockControllerProvider.select((s) => s.isLoading),
);
return LegacyPageLayout(
title: context.translate(I18n.syncClock),
body: Padding(
padding: EdgeInsets.symmetric(horizontal: 18, vertical: 12),
padding: const EdgeInsets.symmetric(horizontal: 18, vertical: 12),
child: Column(
children: [
Text(context.translate(I18n.syncClockMessage)),
SizedBox(height: 36),
const SizedBox(height: 36),
],
),
),
footer: const _SaveSection(),
);
}
}
// ignore: unused_element
class _OptionsSection extends ConsumerWidget {
const _OptionsSection();
@override
Widget build(BuildContext context, WidgetRef ref) {
final vm = ref.read(soundViewModelProvider.notifier);
final soundOption = ref.watch(
soundViewModelProvider.select((s) => s.soundOption),
);
return SingleChildScrollView(
child: Column(
children: [
_SectionButton(
title: context.translate(I18n.soundAndVibration),
icon: Icons.volume_up_outlined,
active: soundOption == 'SOUND_AND_VIBRATION',
onPressed: () {
vm.setSoundOption('SOUND_AND_VIBRATION');
},
),
SizedBox(height: 12),
_SectionButton(
title: context.translate(I18n.soundOnly),
icon: Icons.volume_up_outlined,
active: soundOption == 'SOUND',
onPressed: () {
vm.setSoundOption('SOUND');
},
),
SizedBox(height: 12),
_SectionButton(
title: context.translate(I18n.vibrationOnly),
icon: Icons.vibration_outlined,
active: soundOption == 'VIBRATION',
onPressed: () {
vm.setSoundOption('VIBRATION');
},
),
SizedBox(height: 12),
_SectionButton(
title: context.translate(I18n.silent),
icon: Icons.volume_mute_outlined,
active: soundOption == 'SILENT',
onPressed: () {
vm.setSoundOption('SILENT');
},
),
],
),
);
}
}
class _SectionButton extends ConsumerWidget {
final String title;
final bool active;
final IconData icon;
final VoidCallback onPressed;
const _SectionButton({
required this.title,
required this.active,
required this.icon,
required this.onPressed,
});
@override
Widget build(BuildContext context, WidgetRef ref) {
return SectionButton(
onPressed: onPressed,
icon: Icon(
icon,
color: context.sfColors.legacyPrimary,
size: 36,
),
iconPadding: 8,
body: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
title,
style: TextStyle(
fontWeight: FontWeight.w500,
fontSize: 14,
color: active
? Theme.of(context).colorScheme.onSurface
: Theme.of(context).colorScheme.outline,
),
),
Switch(
value: active,
onChanged: (_) {
onPressed();
},
),
],
),
);
}
}
class _SaveSection extends ConsumerWidget {
const _SaveSection();
@override
Widget build(BuildContext context, WidgetRef ref) {
final vm = ref.read(soundViewModelProvider.notifier);
return Padding(
padding: EdgeInsets.symmetric(horizontal: 24, vertical: 12),
child: PrimaryButton(
onPressed: vm.submit,
text: context.translate(I18n.save),
color: context.sfColors.legacyPrimary,
footer: Padding(
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 12),
child: PrimaryButton(
onPressed: isLoading
? null
: ref.read(syncClockControllerProvider.notifier).sync,
text: context.translate(I18n.save),
color: context.sfColors.legacyPrimary,
),
),
);
}

View File

@@ -0,0 +1,35 @@
import 'dart:async';
import 'package:legacy_device_state/legacy_device_state.dart';
import 'package:riverpod_annotation/riverpod_annotation.dart';
import 'package:sf_shared/sf_shared.dart';
import 'package:sf_tracking/sf_tracking.dart';
part 'timezone_controller.g.dart';
@riverpod
class TimezoneController extends _$TimezoneController {
@override
FutureOr<void> build() {}
Future<void> save({
required DeviceEntity device,
required int newTimezone,
}) async {
if (newTimezone == device.settings.timezone) return;
state = const AsyncLoading();
state = await AsyncValue.guard(() async {
final updated = device.settings.copyWith(timezone: newTimezone);
await ref
.read(deviceSettingsUpdateProvider)
.updateDeviceSettings(device: device, updatedSettings: updated);
ref.syncDeviceSettings(device, updated);
unawaited(
ref
.read(sfTrackingProvider)
.legacySettingsTimezoneChanged(newTimezone.toString()),
);
});
}
}

View File

@@ -0,0 +1,56 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'timezone_controller.dart';
// **************************************************************************
// RiverpodGenerator
// **************************************************************************
// GENERATED CODE - DO NOT MODIFY BY HAND
// ignore_for_file: type=lint, type=warning
@ProviderFor(TimezoneController)
const timezoneControllerProvider = TimezoneControllerProvider._();
final class TimezoneControllerProvider
extends $AsyncNotifierProvider<TimezoneController, void> {
const TimezoneControllerProvider._()
: super(
from: null,
argument: null,
retry: null,
name: r'timezoneControllerProvider',
isAutoDispose: true,
dependencies: null,
$allTransitiveDependencies: null,
);
@override
String debugGetCreateSourceHash() => _$timezoneControllerHash();
@$internal
@override
TimezoneController create() => TimezoneController();
}
String _$timezoneControllerHash() =>
r'1f3122e3ef6ac18760e593a6747b82b8468caec4';
abstract class _$TimezoneController extends $AsyncNotifier<void> {
FutureOr<void> build();
@$mustCallSuper
@override
void runBuild() {
build();
final ref = this.ref as $Ref<AsyncValue<void>, void>;
final element =
ref.element
as $ClassProviderElement<
AnyNotifier<AsyncValue<void>, void>,
AsyncValue<void>,
Object?,
Object?
>;
element.handleValue(ref, null);
}
}

View File

@@ -0,0 +1,11 @@
import 'package:riverpod_annotation/riverpod_annotation.dart';
part 'timezone_selection_provider.g.dart';
@riverpod
class TimezoneSelection extends _$TimezoneSelection {
@override
int? build() => null;
void select(int value) => state = value;
}

View File

@@ -0,0 +1,63 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'timezone_selection_provider.dart';
// **************************************************************************
// RiverpodGenerator
// **************************************************************************
// GENERATED CODE - DO NOT MODIFY BY HAND
// ignore_for_file: type=lint, type=warning
@ProviderFor(TimezoneSelection)
const timezoneSelectionProvider = TimezoneSelectionProvider._();
final class TimezoneSelectionProvider
extends $NotifierProvider<TimezoneSelection, int?> {
const TimezoneSelectionProvider._()
: super(
from: null,
argument: null,
retry: null,
name: r'timezoneSelectionProvider',
isAutoDispose: true,
dependencies: null,
$allTransitiveDependencies: null,
);
@override
String debugGetCreateSourceHash() => _$timezoneSelectionHash();
@$internal
@override
TimezoneSelection create() => TimezoneSelection();
/// {@macro riverpod.override_with_value}
Override overrideWithValue(int? value) {
return $ProviderOverride(
origin: this,
providerOverride: $SyncValueProvider<int?>(value),
);
}
}
String _$timezoneSelectionHash() => r'f108a0d1664e0c7a8423b1ec735745ac1781795b';
abstract class _$TimezoneSelection extends $Notifier<int?> {
int? build();
@$mustCallSuper
@override
void runBuild() {
final created = build();
final ref = this.ref as $Ref<int?, int?>;
final element =
ref.element
as $ClassProviderElement<
AnyNotifier<int?, int?>,
int?,
Object?,
Object?
>;
element.handleValue(ref, created);
}
}

View File

@@ -1,80 +0,0 @@
import 'dart:async';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:legacy_device_state/legacy_device_state.dart';
import 'package:sf_shared/sf_shared.dart';
import 'package:sf_tracking/sf_tracking.dart';
import 'timezone_view_state.dart';
final timezoneViewModelProvider =
NotifierProvider.autoDispose<TimezoneViewModel, TimezoneViewState>(
TimezoneViewModel.new,
);
class TimezoneViewModel extends Notifier<TimezoneViewState> {
late final DeviceSettingsUpdateDatasource _datasource;
late final SfTrackingRepository _tracking;
@override
TimezoneViewState build() {
_datasource = ref.read(deviceSettingsUpdateProvider);
_tracking = ref.read(sfTrackingProvider);
Future.microtask(_load);
return const TimezoneViewState();
}
void _load() {
final device = ref.read(selectedDeviceProvider).value;
if (device == null) return;
state = state.copyWith(
timezone: device.settings.timezone,
isLoading: false,
);
}
void selectTimezone(int value) {
state = state.copyWith(timezone: value);
}
Future<void> save() async {
final device = ref.read(selectedDeviceProvider).value;
if (device == null) return;
if (state.timezone == device.settings.timezone) {
state = state.copyWith(saveSuccess: true);
return;
}
state = state.copyWith(
isSaving: true,
errorEvent: null,
saveSuccess: false,
);
try {
final updatedSettings = device.settings.copyWith(
timezone: state.timezone,
);
await _datasource.updateDeviceSettings(
device: device,
updatedSettings: updatedSettings,
);
if (!ref.mounted) return;
ref.syncDeviceSettings(device, updatedSettings);
unawaited(
_tracking.legacySettingsTimezoneChanged(state.timezone.toString()),
);
state = state.copyWith(isSaving: false, saveSuccess: true);
} catch (e) {
if (!ref.mounted) return;
state = state.copyWith(
isSaving: false,
errorEvent: TimezoneErrorEvent.update,
);
}
}
}

View File

@@ -1,16 +0,0 @@
import 'package:freezed_annotation/freezed_annotation.dart';
part 'timezone_view_state.freezed.dart';
enum TimezoneErrorEvent { update }
@freezed
abstract class TimezoneViewState with _$TimezoneViewState {
const factory TimezoneViewState({
@Default(true) bool isLoading,
@Default(false) bool isSaving,
@Default(0) int timezone,
TimezoneErrorEvent? errorEvent,
@Default(false) bool saveSuccess,
}) = _TimezoneViewState;
}

View File

@@ -1,283 +0,0 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
// coverage:ignore-file
// ignore_for_file: type=lint
// ignore_for_file: unused_element, deprecated_member_use, deprecated_member_use_from_same_package, use_function_type_syntax_for_parameters, unnecessary_const, avoid_init_to_null, invalid_override_different_default_values_named, prefer_expression_function_bodies, annotate_overrides, invalid_annotation_target, unnecessary_question_mark
part of 'timezone_view_state.dart';
// **************************************************************************
// FreezedGenerator
// **************************************************************************
// dart format off
T _$identity<T>(T value) => value;
/// @nodoc
mixin _$TimezoneViewState {
bool get isLoading; bool get isSaving; int get timezone; TimezoneErrorEvent? get errorEvent; bool get saveSuccess;
/// Create a copy of TimezoneViewState
/// with the given fields replaced by the non-null parameter values.
@JsonKey(includeFromJson: false, includeToJson: false)
@pragma('vm:prefer-inline')
$TimezoneViewStateCopyWith<TimezoneViewState> get copyWith => _$TimezoneViewStateCopyWithImpl<TimezoneViewState>(this as TimezoneViewState, _$identity);
@override
bool operator ==(Object other) {
return identical(this, other) || (other.runtimeType == runtimeType&&other is TimezoneViewState&&(identical(other.isLoading, isLoading) || other.isLoading == isLoading)&&(identical(other.isSaving, isSaving) || other.isSaving == isSaving)&&(identical(other.timezone, timezone) || other.timezone == timezone)&&(identical(other.errorEvent, errorEvent) || other.errorEvent == errorEvent)&&(identical(other.saveSuccess, saveSuccess) || other.saveSuccess == saveSuccess));
}
@override
int get hashCode => Object.hash(runtimeType,isLoading,isSaving,timezone,errorEvent,saveSuccess);
@override
String toString() {
return 'TimezoneViewState(isLoading: $isLoading, isSaving: $isSaving, timezone: $timezone, errorEvent: $errorEvent, saveSuccess: $saveSuccess)';
}
}
/// @nodoc
abstract mixin class $TimezoneViewStateCopyWith<$Res> {
factory $TimezoneViewStateCopyWith(TimezoneViewState value, $Res Function(TimezoneViewState) _then) = _$TimezoneViewStateCopyWithImpl;
@useResult
$Res call({
bool isLoading, bool isSaving, int timezone, TimezoneErrorEvent? errorEvent, bool saveSuccess
});
}
/// @nodoc
class _$TimezoneViewStateCopyWithImpl<$Res>
implements $TimezoneViewStateCopyWith<$Res> {
_$TimezoneViewStateCopyWithImpl(this._self, this._then);
final TimezoneViewState _self;
final $Res Function(TimezoneViewState) _then;
/// Create a copy of TimezoneViewState
/// with the given fields replaced by the non-null parameter values.
@pragma('vm:prefer-inline') @override $Res call({Object? isLoading = null,Object? isSaving = null,Object? timezone = null,Object? errorEvent = freezed,Object? saveSuccess = null,}) {
return _then(_self.copyWith(
isLoading: null == isLoading ? _self.isLoading : isLoading // ignore: cast_nullable_to_non_nullable
as bool,isSaving: null == isSaving ? _self.isSaving : isSaving // ignore: cast_nullable_to_non_nullable
as bool,timezone: null == timezone ? _self.timezone : timezone // ignore: cast_nullable_to_non_nullable
as int,errorEvent: freezed == errorEvent ? _self.errorEvent : errorEvent // ignore: cast_nullable_to_non_nullable
as TimezoneErrorEvent?,saveSuccess: null == saveSuccess ? _self.saveSuccess : saveSuccess // ignore: cast_nullable_to_non_nullable
as bool,
));
}
}
/// Adds pattern-matching-related methods to [TimezoneViewState].
extension TimezoneViewStatePatterns on TimezoneViewState {
/// A variant of `map` that fallback to returning `orElse`.
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case final Subclass value:
/// return ...;
/// case _:
/// return orElse();
/// }
/// ```
@optionalTypeArgs TResult maybeMap<TResult extends Object?>(TResult Function( _TimezoneViewState value)? $default,{required TResult orElse(),}){
final _that = this;
switch (_that) {
case _TimezoneViewState() when $default != null:
return $default(_that);case _:
return orElse();
}
}
/// A `switch`-like method, using callbacks.
///
/// Callbacks receives the raw object, upcasted.
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case final Subclass value:
/// return ...;
/// case final Subclass2 value:
/// return ...;
/// }
/// ```
@optionalTypeArgs TResult map<TResult extends Object?>(TResult Function( _TimezoneViewState value) $default,){
final _that = this;
switch (_that) {
case _TimezoneViewState():
return $default(_that);case _:
throw StateError('Unexpected subclass');
}
}
/// A variant of `map` that fallback to returning `null`.
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case final Subclass value:
/// return ...;
/// case _:
/// return null;
/// }
/// ```
@optionalTypeArgs TResult? mapOrNull<TResult extends Object?>(TResult? Function( _TimezoneViewState value)? $default,){
final _that = this;
switch (_that) {
case _TimezoneViewState() when $default != null:
return $default(_that);case _:
return null;
}
}
/// A variant of `when` that fallback to an `orElse` callback.
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case Subclass(:final field):
/// return ...;
/// case _:
/// return orElse();
/// }
/// ```
@optionalTypeArgs TResult maybeWhen<TResult extends Object?>(TResult Function( bool isLoading, bool isSaving, int timezone, TimezoneErrorEvent? errorEvent, bool saveSuccess)? $default,{required TResult orElse(),}) {final _that = this;
switch (_that) {
case _TimezoneViewState() when $default != null:
return $default(_that.isLoading,_that.isSaving,_that.timezone,_that.errorEvent,_that.saveSuccess);case _:
return orElse();
}
}
/// A `switch`-like method, using callbacks.
///
/// As opposed to `map`, this offers destructuring.
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case Subclass(:final field):
/// return ...;
/// case Subclass2(:final field2):
/// return ...;
/// }
/// ```
@optionalTypeArgs TResult when<TResult extends Object?>(TResult Function( bool isLoading, bool isSaving, int timezone, TimezoneErrorEvent? errorEvent, bool saveSuccess) $default,) {final _that = this;
switch (_that) {
case _TimezoneViewState():
return $default(_that.isLoading,_that.isSaving,_that.timezone,_that.errorEvent,_that.saveSuccess);case _:
throw StateError('Unexpected subclass');
}
}
/// A variant of `when` that fallback to returning `null`
///
/// It is equivalent to doing:
/// ```dart
/// switch (sealedClass) {
/// case Subclass(:final field):
/// return ...;
/// case _:
/// return null;
/// }
/// ```
@optionalTypeArgs TResult? whenOrNull<TResult extends Object?>(TResult? Function( bool isLoading, bool isSaving, int timezone, TimezoneErrorEvent? errorEvent, bool saveSuccess)? $default,) {final _that = this;
switch (_that) {
case _TimezoneViewState() when $default != null:
return $default(_that.isLoading,_that.isSaving,_that.timezone,_that.errorEvent,_that.saveSuccess);case _:
return null;
}
}
}
/// @nodoc
class _TimezoneViewState implements TimezoneViewState {
const _TimezoneViewState({this.isLoading = true, this.isSaving = false, this.timezone = 0, this.errorEvent, this.saveSuccess = false});
@override@JsonKey() final bool isLoading;
@override@JsonKey() final bool isSaving;
@override@JsonKey() final int timezone;
@override final TimezoneErrorEvent? errorEvent;
@override@JsonKey() final bool saveSuccess;
/// Create a copy of TimezoneViewState
/// with the given fields replaced by the non-null parameter values.
@override @JsonKey(includeFromJson: false, includeToJson: false)
@pragma('vm:prefer-inline')
_$TimezoneViewStateCopyWith<_TimezoneViewState> get copyWith => __$TimezoneViewStateCopyWithImpl<_TimezoneViewState>(this, _$identity);
@override
bool operator ==(Object other) {
return identical(this, other) || (other.runtimeType == runtimeType&&other is _TimezoneViewState&&(identical(other.isLoading, isLoading) || other.isLoading == isLoading)&&(identical(other.isSaving, isSaving) || other.isSaving == isSaving)&&(identical(other.timezone, timezone) || other.timezone == timezone)&&(identical(other.errorEvent, errorEvent) || other.errorEvent == errorEvent)&&(identical(other.saveSuccess, saveSuccess) || other.saveSuccess == saveSuccess));
}
@override
int get hashCode => Object.hash(runtimeType,isLoading,isSaving,timezone,errorEvent,saveSuccess);
@override
String toString() {
return 'TimezoneViewState(isLoading: $isLoading, isSaving: $isSaving, timezone: $timezone, errorEvent: $errorEvent, saveSuccess: $saveSuccess)';
}
}
/// @nodoc
abstract mixin class _$TimezoneViewStateCopyWith<$Res> implements $TimezoneViewStateCopyWith<$Res> {
factory _$TimezoneViewStateCopyWith(_TimezoneViewState value, $Res Function(_TimezoneViewState) _then) = __$TimezoneViewStateCopyWithImpl;
@override @useResult
$Res call({
bool isLoading, bool isSaving, int timezone, TimezoneErrorEvent? errorEvent, bool saveSuccess
});
}
/// @nodoc
class __$TimezoneViewStateCopyWithImpl<$Res>
implements _$TimezoneViewStateCopyWith<$Res> {
__$TimezoneViewStateCopyWithImpl(this._self, this._then);
final _TimezoneViewState _self;
final $Res Function(_TimezoneViewState) _then;
/// Create a copy of TimezoneViewState
/// with the given fields replaced by the non-null parameter values.
@override @pragma('vm:prefer-inline') $Res call({Object? isLoading = null,Object? isSaving = null,Object? timezone = null,Object? errorEvent = freezed,Object? saveSuccess = null,}) {
return _then(_TimezoneViewState(
isLoading: null == isLoading ? _self.isLoading : isLoading // ignore: cast_nullable_to_non_nullable
as bool,isSaving: null == isSaving ? _self.isSaving : isSaving // ignore: cast_nullable_to_non_nullable
as bool,timezone: null == timezone ? _self.timezone : timezone // ignore: cast_nullable_to_non_nullable
as int,errorEvent: freezed == errorEvent ? _self.errorEvent : errorEvent // ignore: cast_nullable_to_non_nullable
as TimezoneErrorEvent?,saveSuccess: null == saveSuccess ? _self.saveSuccess : saveSuccess // ignore: cast_nullable_to_non_nullable
as bool,
));
}
}
// dart format on

View File

@@ -1,106 +1,98 @@
import 'package:design_system/design_system.dart';
import 'package:legacy_theme/legacy_theme.dart';
import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:legacy_device_state/legacy_device_state.dart';
import 'package:legacy_theme/legacy_theme.dart';
import 'package:legacy_ui/legacy_ui.dart';
import 'package:settings/src/features/timezone/presentation/providers/timezone_controller.dart';
import 'package:settings/src/features/timezone/presentation/providers/timezone_selection_provider.dart';
import 'package:settings/src/features/timezone/timezone_data.dart';
import 'package:sf_localizations/sf_localizations.dart';
import '../timezone_data.dart';
import 'state/timezone_view_model.dart';
import 'package:sf_shared/sf_shared.dart';
class TimezoneScreen extends ConsumerWidget {
const TimezoneScreen({super.key});
@override
Widget build(BuildContext context, WidgetRef ref) {
ref.listen(timezoneControllerProvider, (prev, next) async {
next.showErrorOn(context);
if (prev != null &&
prev.isLoading &&
!next.isLoading &&
!next.hasError) {
await showSuccessDialog(context, I18n.deviceUpdatedSuccess);
}
});
final primaryColor = context.sfColors.legacyPrimary;
final state = ref.watch(timezoneViewModelProvider);
final vm = ref.read(timezoneViewModelProvider.notifier);
ref.listen(timezoneViewModelProvider.select((s) => s.errorEvent), (
previous,
next,
) {
if (next != null) {
showTopSnackbar(
context,
message: context.translate(I18n.errorTimezone),
type: MessageType.error,
);
}
});
ref.listen(timezoneViewModelProvider.select((s) => s.saveSuccess), (
previous,
next,
) {
if (next && !(previous ?? false)) {
showTopSnackbar(
context,
message: context.translate(I18n.timezoneUpdated),
type: MessageType.success,
);
}
});
final device = ref.watch(selectedDeviceProvider).value;
final currentTimezone = device?.settings.timezone ?? 0;
final selectedTimezone =
ref.watch(timezoneSelectionProvider) ?? currentTimezone;
final isSaving = ref.watch(
timezoneControllerProvider.select((s) => s.isLoading),
);
final selected = timezoneEntries
.where((t) => t.$1 == state.timezone)
.where((t) => t.$1 == selectedTimezone)
.firstOrNull;
final others = timezoneEntries
.where((t) => t.$1 != state.timezone)
.where((t) => t.$1 != selectedTimezone)
.toList();
return LegacyPageLayout(
title: context.translate(I18n.timezone),
body: state.isLoading
? const Center(child: CircularProgressIndicator())
: ListView(
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12),
children: [
if (selected != null) ...[
_SelectedTimezoneCard(
offset: selected.$1,
city: selected.$2,
continent: selected.$3,
label: formatUtcOffset(selected.$1),
primaryColor: primaryColor,
),
const SizedBox(height: 20),
Padding(
padding: const EdgeInsets.only(left: 4, bottom: 8),
child: Text(
context.translate(I18n.timezoneOther),
style: TextStyle(
fontSize: 13,
fontWeight: FontWeight.w600,
color: Theme.of(context).colorScheme.onSurfaceVariant,
),
),
),
],
...others.map(
(tz) => _TimezoneItem(
offset: tz.$1,
city: tz.$2,
continent: tz.$3,
label: formatUtcOffset(tz.$1),
primaryColor: primaryColor,
onTap: () => vm.selectTimezone(tz.$1),
),
),
],
body: ListView(
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12),
children: [
if (selected != null) ...[
_SelectedTimezoneCard(
city: selected.$2,
continent: selected.$3,
label: formatUtcOffset(selected.$1),
primaryColor: primaryColor,
),
const SizedBox(height: 20),
Padding(
padding: const EdgeInsets.only(left: 4, bottom: 8),
child: Text(
context.translate(I18n.timezoneOther),
style: TextStyle(
fontSize: 13,
fontWeight: FontWeight.w600,
color: Theme.of(context).colorScheme.onSurfaceVariant,
),
),
),
],
...others.map(
(tz) => _TimezoneItem(
city: tz.$2,
continent: tz.$3,
label: formatUtcOffset(tz.$1),
onTap: () => ref
.read(timezoneSelectionProvider.notifier)
.select(tz.$1),
),
),
],
),
footer: Padding(
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 10),
child: PrimaryButton(
onPressed: state.isSaving
onPressed: isSaving
? null
: () async {
if (device == null) return;
if (!await guardDeviceCommand(context, ref)) return;
vm.save();
if (!context.mounted) return;
ref.read(timezoneControllerProvider.notifier).save(
device: device,
newTimezone: selectedTimezone,
);
},
text: state.isSaving ? '...' : context.translate(I18n.save),
text: isSaving ? '...' : context.translate(I18n.save),
color: primaryColor,
),
),
@@ -109,14 +101,12 @@ class TimezoneScreen extends ConsumerWidget {
}
class _SelectedTimezoneCard extends StatelessWidget {
final int offset;
final String city;
final String continent;
final String label;
final Color primaryColor;
const _SelectedTimezoneCard({
required this.offset,
required this.city,
required this.continent,
required this.label,
@@ -177,19 +167,15 @@ class _SelectedTimezoneCard extends StatelessWidget {
}
class _TimezoneItem extends StatelessWidget {
final int offset;
final String city;
final String continent;
final String label;
final Color primaryColor;
final VoidCallback onTap;
const _TimezoneItem({
required this.offset,
required this.city,
required this.continent,
required this.label,
required this.primaryColor,
required this.onTap,
});
@@ -230,7 +216,10 @@ class _TimezoneItem extends StatelessWidget {
const SizedBox(height: 2),
Text(
continent,
style: TextStyle(fontSize: 12, color: Theme.of(context).colorScheme.onSurfaceVariant),
style: TextStyle(
fontSize: 12,
color: Theme.of(context).colorScheme.onSurfaceVariant,
),
),
],
),

View File

@@ -0,0 +1,96 @@
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:legacy_device_state/legacy_device_state.dart';
import 'package:mocktail/mocktail.dart';
import 'package:settings/src/features/sound/presentation/providers/sound_controller.dart';
import 'package:sf_infrastructure/sf_infrastructure.dart';
import 'package:sf_shared/sf_shared.dart';
import 'package:sf_shared/testing.dart';
import 'package:sf_tracking/sf_tracking.dart';
class MockDeviceSettingsUpdateDatasource extends Mock
implements DeviceSettingsUpdateDatasource {}
const _device = DeviceEntity(
id: 'device-1',
identificator: 'imei-1',
carrierName: 'Watch',
settings: DeviceSettingsEntity(soundMode: 'RINGING'),
);
void main() {
setUpAll(() {
registerFallbackValue(_device);
registerFallbackValue(const DeviceSettingsEntity());
});
ProviderContainer buildContainer(DeviceSettingsUpdateDatasource ds) {
return makeContainer(
overrides: [
deviceSettingsUpdateProvider.overrideWithValue(ds),
sfTrackingProvider.overrideWithValue(
SfTrackingRepository(clients: const []),
),
],
);
}
group('SoundController.save', () {
test('transitions to AsyncData when datasource succeeds', () async {
final ds = MockDeviceSettingsUpdateDatasource();
when(
() => ds.updateDeviceSettings(
device: any(named: 'device'),
updatedSettings: any(named: 'updatedSettings'),
),
).thenAnswer((_) async {});
final container = buildContainer(ds);
addTearDown(container.dispose);
await container
.read(soundControllerProvider.notifier)
.save(device: _device, newMode: 'VIBRATION');
expect(container.read(soundControllerProvider), isA<AsyncData<void>>());
});
test('no-ops when new mode equals current', () async {
final ds = MockDeviceSettingsUpdateDatasource();
final container = buildContainer(ds);
addTearDown(container.dispose);
await container
.read(soundControllerProvider.notifier)
.save(device: _device, newMode: 'RINGING');
verifyNever(
() => ds.updateDeviceSettings(
device: any(named: 'device'),
updatedSettings: any(named: 'updatedSettings'),
),
);
});
test('exposes AsyncError when datasource fails', () async {
final ds = MockDeviceSettingsUpdateDatasource();
when(
() => ds.updateDeviceSettings(
device: any(named: 'device'),
updatedSettings: any(named: 'updatedSettings'),
),
).thenThrow(const ApiException(message: 'boom', isNetworkError: true));
final container = buildContainer(ds);
addTearDown(container.dispose);
await container
.read(soundControllerProvider.notifier)
.save(device: _device, newMode: 'VIBRATION');
final state = container.read(soundControllerProvider);
expect(state, isA<AsyncError<void>>());
expect(state.error, isA<ApiException>());
});
});
}

View File

@@ -0,0 +1,96 @@
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:legacy_device_state/legacy_device_state.dart';
import 'package:mocktail/mocktail.dart';
import 'package:settings/src/features/timezone/presentation/providers/timezone_controller.dart';
import 'package:sf_infrastructure/sf_infrastructure.dart';
import 'package:sf_shared/sf_shared.dart';
import 'package:sf_shared/testing.dart';
import 'package:sf_tracking/sf_tracking.dart';
class MockDeviceSettingsUpdateDatasource extends Mock
implements DeviceSettingsUpdateDatasource {}
const _device = DeviceEntity(
id: 'device-1',
identificator: 'imei-1',
carrierName: 'Watch',
settings: DeviceSettingsEntity(timezone: 1),
);
void main() {
setUpAll(() {
registerFallbackValue(_device);
registerFallbackValue(const DeviceSettingsEntity());
});
ProviderContainer buildContainer(DeviceSettingsUpdateDatasource ds) {
return makeContainer(
overrides: [
deviceSettingsUpdateProvider.overrideWithValue(ds),
sfTrackingProvider.overrideWithValue(
SfTrackingRepository(clients: const []),
),
],
);
}
group('TimezoneController.save', () {
test('transitions to AsyncData when datasource succeeds', () async {
final ds = MockDeviceSettingsUpdateDatasource();
when(
() => ds.updateDeviceSettings(
device: any(named: 'device'),
updatedSettings: any(named: 'updatedSettings'),
),
).thenAnswer((_) async {});
final container = buildContainer(ds);
addTearDown(container.dispose);
await container
.read(timezoneControllerProvider.notifier)
.save(device: _device, newTimezone: 2);
expect(container.read(timezoneControllerProvider), isA<AsyncData<void>>());
});
test('no-ops when new timezone equals current', () async {
final ds = MockDeviceSettingsUpdateDatasource();
final container = buildContainer(ds);
addTearDown(container.dispose);
await container
.read(timezoneControllerProvider.notifier)
.save(device: _device, newTimezone: 1);
verifyNever(
() => ds.updateDeviceSettings(
device: any(named: 'device'),
updatedSettings: any(named: 'updatedSettings'),
),
);
});
test('exposes AsyncError when datasource fails', () async {
final ds = MockDeviceSettingsUpdateDatasource();
when(
() => ds.updateDeviceSettings(
device: any(named: 'device'),
updatedSettings: any(named: 'updatedSettings'),
),
).thenThrow(const ApiException(message: 'boom', isNetworkError: true));
final container = buildContainer(ds);
addTearDown(container.dispose);
await container
.read(timezoneControllerProvider.notifier)
.save(device: _device, newTimezone: 2);
final state = container.read(timezoneControllerProvider);
expect(state, isA<AsyncError<void>>());
expect(state.error, isA<ApiException>());
});
});
}