| | | 1 | | using System.ComponentModel; |
| | | 2 | | using System.Reflection; |
| | | 3 | | |
| | | 4 | | namespace UIBlazor.Services.Settings; |
| | | 5 | | |
| | | 6 | | public abstract class BaseSettingsProvider<TOptions> : IBaseSettingsProvider where TOptions : BaseOptions, new() |
| | | 7 | | { |
| | | 8 | | protected readonly ILocalStorageService Storage; |
| | | 9 | | protected readonly string StorageKey; |
| | | 10 | | protected readonly Debouncer Debouncer; |
| | | 11 | | private bool _isInitializing; |
| | | 12 | | private readonly ILogger _logger; |
| | | 13 | | |
| | 126 | 14 | | public TOptions Current { get; } = new(); |
| | | 15 | | |
| | 23 | 16 | | protected BaseSettingsProvider( |
| | 23 | 17 | | ILocalStorageService storage, |
| | 23 | 18 | | ILogger logger, |
| | 23 | 19 | | string storageKey, |
| | 23 | 20 | | TimeSpan? debounceDelay = null) |
| | | 21 | | { |
| | 23 | 22 | | Storage = storage; |
| | 23 | 23 | | _logger = logger; |
| | 23 | 24 | | StorageKey = storageKey; |
| | 23 | 25 | | Debouncer = new Debouncer(debounceDelay ?? TimeSpan.FromMilliseconds(750), SaveAsync); |
| | 23 | 26 | | } |
| | | 27 | | |
| | | 28 | | private void OnPropertyChanged(object? sender, PropertyChangedEventArgs e) |
| | | 29 | | { |
| | 3 | 30 | | if (!_isInitializing) |
| | | 31 | | { |
| | 1 | 32 | | OnAnyPropertyChanged(e.PropertyName); |
| | 1 | 33 | | Debouncer.Trigger(); |
| | | 34 | | } |
| | 3 | 35 | | } |
| | | 36 | | |
| | | 37 | | /// <summary> |
| | | 38 | | /// Свойство изменилось и будет сохранено |
| | | 39 | | /// </summary> |
| | | 40 | | /// <param name="propertyName">Имя измененного свойства</param> |
| | | 41 | | protected virtual void OnAnyPropertyChanged(string? propertyName) |
| | | 42 | | { |
| | 0 | 43 | | } |
| | | 44 | | |
| | | 45 | | /// <summary> |
| | | 46 | | /// Событие после сохранения настроек |
| | | 47 | | /// </summary> |
| | | 48 | | public event Action? OnSaved; |
| | | 49 | | |
| | | 50 | | /// <summary> |
| | | 51 | | /// Немедленное сохранение Current объекта.<br/> |
| | | 52 | | /// Автоматически вызывается при изменении любого свойства, которое использует <see cref="BaseOptions.SetIfChanged"/ |
| | | 53 | | /// </summary> |
| | | 54 | | public virtual async Task SaveAsync() |
| | | 55 | | { |
| | 9 | 56 | | await Storage.SetItemAsync(StorageKey, Current); |
| | 9 | 57 | | OnSaved?.Invoke(); |
| | 9 | 58 | | } |
| | | 59 | | |
| | | 60 | | /// <summary> |
| | | 61 | | /// Загрузка настроек |
| | | 62 | | /// </summary> |
| | | 63 | | public async Task InitializeAsync() |
| | | 64 | | { |
| | 13 | 65 | | _isInitializing = true; |
| | | 66 | | try |
| | | 67 | | { |
| | 13 | 68 | | var saved = await Storage.GetItemAsync<TOptions>(StorageKey); |
| | 12 | 69 | | if (saved != null) |
| | | 70 | | { |
| | 5 | 71 | | CopyProperties(saved, Current); |
| | | 72 | | } |
| | | 73 | | |
| | 12 | 74 | | Current.PropertyChanged += OnPropertyChanged; |
| | 12 | 75 | | await AfterInitAsync(); |
| | 12 | 76 | | } |
| | 1 | 77 | | catch (Exception ex) |
| | | 78 | | { |
| | 1 | 79 | | _logger.LogError($"Failed to initialize settings for {StorageKey}: {ex.Message}"); |
| | 1 | 80 | | } |
| | | 81 | | finally |
| | | 82 | | { |
| | 13 | 83 | | _isInitializing = false; |
| | | 84 | | } |
| | 13 | 85 | | } |
| | | 86 | | |
| | | 87 | | /// <summary> |
| | | 88 | | /// Вызывается сразу после загрузки настроек |
| | | 89 | | /// Сюда надо добавлять первичные манипуляции с настройками |
| | | 90 | | /// </summary> |
| | 0 | 91 | | protected virtual Task AfterInitAsync() => Task.CompletedTask; |
| | | 92 | | |
| | | 93 | | protected virtual void CopyProperties(TOptions from, TOptions to) |
| | | 94 | | { |
| | 5 | 95 | | var properties = typeof(TOptions).GetProperties(BindingFlags.Public | BindingFlags.Instance) |
| | 15 | 96 | | .Where(p => p is { CanRead: true, CanWrite: true }); |
| | | 97 | | |
| | 30 | 98 | | foreach (var prop in properties) |
| | | 99 | | { |
| | 10 | 100 | | var value = prop.GetValue(from); |
| | 10 | 101 | | prop.SetValue(to, value); |
| | | 102 | | } |
| | 5 | 103 | | } |
| | | 104 | | |
| | | 105 | | public abstract Task ResetAsync(); |
| | | 106 | | |
| | | 107 | | public virtual void Dispose() |
| | | 108 | | { |
| | 0 | 109 | | Current.PropertyChanged -= OnPropertyChanged; |
| | 0 | 110 | | Debouncer.Dispose(); |
| | 0 | 111 | | } |
| | | 112 | | } |