CoreRepository.cs 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764
  1. using System.Collections;
  2. using System.Collections.ObjectModel;
  3. using System.ComponentModel;
  4. using System.Diagnostics.CodeAnalysis;
  5. using System.Linq.Expressions;
  6. using System.Runtime.CompilerServices;
  7. using Avalonia.Controls.Selection;
  8. using Avalonia.Threading;
  9. using CommunityToolkit.Mvvm.ComponentModel;
  10. using InABox.Clients;
  11. using InABox.Configuration;
  12. using InABox.Core;
  13. using JetBrains.Annotations;
  14. using NotNullAttribute = JetBrains.Annotations.NotNullAttribute;
  15. namespace InABox.Avalonia
  16. {
  17. public class CoreRepositoryItemCreatedArgs<TShell> : EventArgs
  18. {
  19. public TShell Item { get; private set; }
  20. public CoreRepositoryItemCreatedArgs(TShell item)
  21. {
  22. Item = item;
  23. }
  24. }
  25. public delegate void CoreRepositoryItemCreatedEvent<TShell>(object sender, CoreRepositoryItemCreatedArgs<TShell> args);
  26. public abstract class CoreRepository
  27. {
  28. public static bool IsCached(string? filename) =>
  29. !String.IsNullOrWhiteSpace(filename)
  30. && File.Exists(CacheFileName(filename));
  31. public static string CacheFileName(string filename) =>
  32. Path.Combine(CacheFolder(), filename);
  33. public static string CacheFolder()
  34. {
  35. var result = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData);
  36. if (OperatingSystem.IsWindows())
  37. {
  38. var assembly = Path.GetFileNameWithoutExtension(System.Diagnostics.Process.GetCurrentProcess().MainModule.ModuleName);
  39. result = Path.Combine(result, assembly);
  40. }
  41. if (CacheID != Guid.Empty)
  42. result = Path.Combine(result,CacheID.ToString());
  43. if (!Directory.Exists(result))
  44. Directory.CreateDirectory(result);
  45. return result;
  46. }
  47. public static Guid CacheID { get; set; }
  48. }
  49. public partial class CoreRepositoryFilter : ObservableObject
  50. {
  51. [ObservableProperty]
  52. private string? _name;
  53. [ObservableProperty]
  54. private string? _filter;
  55. [ObservableProperty]
  56. private bool _selected;
  57. public override string ToString() => Name ?? "";
  58. }
  59. public abstract class CoreRepository<TParent, TItem, TEntity> : CoreRepository, ICoreRepository, IEnumerable<TItem>
  60. where TParent : CoreRepository<TParent, TItem, TEntity>
  61. where TEntity : Entity, IRemotable, IPersistent, new()
  62. where TItem : Shell<TParent,TEntity>, new()
  63. {
  64. readonly MultiQuery _query = new();
  65. public Func<Filter<TEntity>> Filter { get; set; }
  66. protected virtual Filter<TEntity>? BaseFilter() => null;
  67. public IModelHost Host { get; set; }
  68. private DateTime _lastUpdated = DateTime.MinValue;
  69. public DateTime LastUpdated
  70. {
  71. get => _lastUpdated;
  72. protected set
  73. {
  74. _lastUpdated = value;
  75. OnPropertyChanged();
  76. }
  77. }
  78. public Func<string>? FileName { get; }
  79. private string DataFileName() => FileName != null
  80. ? $"{FileName.Invoke()}.data"
  81. : string.Empty;
  82. private string FilterFileName() => !string.IsNullOrWhiteSpace(FilterTag) && FileName != null
  83. ? $"{FileName.Invoke()}.filter"
  84. : string.Empty;
  85. protected CoreRepository(IModelHost host, Func<Filter<TEntity>> filter, Func<string>? filename = null)
  86. {
  87. AllItems = new CoreObservableCollection<TItem>();
  88. AllItems.CollectionChanged += (sender, args) => ItemsChanged(AllItems);
  89. // EnableSynchronization(AllItems);
  90. Items = new CoreObservableCollection<TItem>();
  91. // EnableSynchronization(Items);
  92. Reset();
  93. Host = host;
  94. Filter = filter;
  95. FileName = filename;
  96. }
  97. protected virtual void ItemsChanged(IEnumerable<TItem> items)
  98. {
  99. }
  100. // private void EnableSynchronization(IEnumerable items)
  101. // {
  102. // // BindingBase.EnableCollectionSynchronization(items, null,
  103. // // (collection, context, method, access) =>
  104. // // {
  105. // // lock (collection)
  106. // // {
  107. // // method?.Invoke();
  108. // // }
  109. // // }
  110. // // );
  111. // }
  112. #region INotifyPropertyChanged
  113. public event PropertyChangedEventHandler? PropertyChanged;
  114. protected void DoPropertyChanged(object sender, PropertyChangedEventArgs args)
  115. {
  116. PropertyChanged?.Invoke(sender, args);
  117. }
  118. protected bool SetProperty<T>(ref T field, T value, [CallerMemberName] string propertyName = "")
  119. {
  120. if (EqualityComparer<T>.Default.Equals(field, value))
  121. return false;
  122. field = value;
  123. OnPropertyChanged(propertyName);
  124. return true;
  125. }
  126. protected void OnPropertyChanged([CallerMemberName] string propertyName = "")
  127. => DoPropertyChanged(this, new PropertyChangedEventArgs(propertyName));
  128. #endregion
  129. #region Image Lookups
  130. public Dictionary<Guid, byte[]> Images { get; private set; } = new Dictionary<Guid, byte[]>();
  131. public byte[]? GetImageSource(Guid id)
  132. {
  133. return Images.GetValueOrDefault(id);
  134. }
  135. public byte[]? GetImage(Guid id) => Images.GetValueOrDefault(id);
  136. public bool HasImages() => Images.Any();
  137. #endregion
  138. protected virtual string FilterTag => typeof(TEntity).EntityName().Split('.').Last();
  139. public CoreObservableCollection<CoreRepositoryFilter> AvailableFilters { get; } = new();
  140. IEnumerable ICoreRepository.AvailableFilters => AvailableFilters;
  141. protected Filter<TEntity>? SelectedFilter =>
  142. Serialization.Deserialize<Filter<TEntity>>(AvailableFilters.FirstOrDefault(x => x.Selected)?.Filter);
  143. public bool FiltersVisible => AvailableFilters.Any();
  144. public string? SelectedFilterName => AvailableFilters.FirstOrDefault(x => x.Selected)?.Name;
  145. public void SelectFilter(String? name)
  146. {
  147. var definition = AvailableFilters.FirstOrDefault(x => String.Equals(x.Name, name));
  148. foreach (var availableFilter in AvailableFilters)
  149. availableFilter.Selected = definition == availableFilter;
  150. OnPropertyChanged(nameof(AvailableFilters));
  151. }
  152. protected Filter<TEntity>? EffectiveFilter()
  153. {
  154. var filters = new Filters<TEntity>();
  155. filters.Add(BaseFilter());
  156. filters.Add(Filter?.Invoke());
  157. filters.Add(SelectedFilter);
  158. var result = filters.Combine();
  159. return result;
  160. }
  161. protected Columns<TOtherEntity> GetColumns<TOtherItem, TOtherEntity>()
  162. where TOtherItem : Shell<TParent, TOtherEntity>, new()
  163. where TOtherEntity : Entity, IRemotable, IPersistent, new()
  164. {
  165. return new TOtherItem().Columns.Columns;
  166. }
  167. protected virtual void Initialize()
  168. {
  169. Loaded = false;
  170. AllItems.Clear();
  171. Items.Clear();
  172. Images.Clear();
  173. }
  174. public bool Loaded { get; protected set; }
  175. private void DoRefresh(bool force)
  176. {
  177. // force = force || Host.Status == ConnectionStatus.Connected;
  178. var selectedIDs = SelectedItems.Select(x => x.ID).ToHashSet();
  179. Items.Clear();
  180. var dataFileName = DataFileName();
  181. if (!force && !Loaded && CoreRepository.IsCached(dataFileName))
  182. {
  183. DoBeforeLoad();
  184. if (LoadFromStorage())
  185. {
  186. DoAfterLoad();
  187. foreach(var item in Items)
  188. {
  189. item.IsSelected = selectedIDs.Contains(item.ID);
  190. }
  191. return;
  192. }
  193. }
  194. if ((force || !Loaded) && (Host.Status == ConnectionStatus.Connected))
  195. {
  196. DoLoad();
  197. SaveToStorage();
  198. foreach(var item in Items)
  199. {
  200. item.IsSelected = selectedIDs.Contains(item.ID);
  201. }
  202. return;
  203. }
  204. foreach(var item in Items)
  205. {
  206. item.IsSelected = selectedIDs.Contains(item.ID);
  207. }
  208. }
  209. private void AfterRefresh()
  210. {
  211. Loaded = true;
  212. Dispatcher.UIThread.Invoke(Search);
  213. NotifyChanged();
  214. }
  215. public virtual ICoreRepository Refresh(bool force)
  216. {
  217. DoRefresh(force);
  218. AfterRefresh();
  219. return this;
  220. }
  221. public void Refresh(bool force, Action loaded)
  222. {
  223. Task.Run(
  224. () =>
  225. {
  226. DoRefresh(force);
  227. Dispatcher.UIThread.Post(
  228. () =>
  229. {
  230. AfterRefresh();
  231. loaded?.Invoke();
  232. }
  233. );
  234. }
  235. );
  236. }
  237. public Task<ICoreRepository> RefreshAsync(bool force)
  238. {
  239. return Task.Run(() => Refresh(force));
  240. }
  241. public void Reset()
  242. {
  243. Initialize();
  244. }
  245. public event CoreRepositoryChangedEvent? Changed;
  246. protected void NotifyChanged() => Changed?.Invoke(this, new CoreRepositoryChangedEventArgs());
  247. public virtual SortOrder<TEntity>? Sort => LookupFactory.DefineSort<TEntity>();
  248. protected CoreObservableCollection<TItem> AllItems { get; private set; }
  249. private CoreTable _table = new CoreTable();
  250. public CoreObservableCollection<TItem> Items { get; private set; }
  251. public int ItemCount => Items.Count;
  252. IEnumerable ICoreRepository.Items => Items;
  253. #region Item Selection
  254. IEnumerable ICoreRepository.SelectedItems => SelectedItems;
  255. IEnumerable<TItem> SelectedItems => Items.Where(x => x.IsSelected);
  256. public bool IsSelected(TItem? item) => item is not null && item.IsSelected;
  257. public void SetSelectedItems(IEnumerable<TItem> items)
  258. {
  259. var selectedItems = items.Select(x => x.ID).ToHashSet();
  260. foreach(var item in Items)
  261. {
  262. item.IsSelected = selectedItems.Contains(item.ID);
  263. }
  264. Search();
  265. }
  266. public void SelectItem([CanBeNull] TItem? item)
  267. {
  268. if ((item != null) && !item.IsSelected)
  269. {
  270. item.IsSelected = true;
  271. Search();
  272. }
  273. }
  274. public void UnselectItem([CanBeNull] TItem? item)
  275. {
  276. if ((item != null) && item.IsSelected)
  277. {
  278. item.IsSelected = false;
  279. Search();
  280. }
  281. }
  282. public void ToggleSelection(TItem? item)
  283. {
  284. if (IsSelected(item))
  285. UnselectItem(item);
  286. else
  287. SelectItem(item);
  288. }
  289. public void SelectNone()
  290. {
  291. foreach(var item in AllItems)
  292. {
  293. item.IsSelected = false;
  294. }
  295. Search();
  296. }
  297. public void SelectAll()
  298. {
  299. foreach(var item in AllItems)
  300. {
  301. item.IsSelected = true;
  302. }
  303. Search();
  304. }
  305. void ICoreRepository.SelectItem(object item) => SelectItem(item as TItem);
  306. void ICoreRepository.UnselectItem(object item) => UnselectItem(item as TItem);
  307. void ICoreRepository.ToggleSelection(object item) => ToggleSelection(item as TItem);
  308. bool ICoreRepository.IsSelected(object item) => IsSelected(item as TItem);
  309. void ICoreRepository.SetSelectedItems(IEnumerable<object> items) => SetSelectedItems(items.OfType<TItem>());
  310. #endregion
  311. #region Searching
  312. public Func<TItem, bool>? SearchPredicate { get; set; }
  313. public Func<List<TItem>,List<TItem>>? SortPredicate { get; set; }
  314. public ICoreRepository Search(Func<TItem, bool> searchpredicate, Func<List<TItem>,List<TItem>> sortpredicate)
  315. {
  316. SortPredicate = sortpredicate;
  317. SearchPredicate = searchpredicate;
  318. Search();
  319. return this;
  320. }
  321. public ICoreRepository Search(Func<TItem, bool>? searchpredicate)
  322. {
  323. SearchPredicate = searchpredicate;
  324. Search();
  325. return this;
  326. }
  327. public ICoreRepository Search()
  328. {
  329. var items = AllItems == null
  330. ? new List<TItem>()
  331. : SearchPredicate != null
  332. ? new List<TItem>(AllItems.Where(SearchPredicate))
  333. : new List<TItem>(AllItems);
  334. if (SortPredicate != null)
  335. items = SortPredicate(items);
  336. Items.ReplaceRange(items);
  337. OnPropertyChanged(nameof(Items));
  338. OnPropertyChanged(nameof(ItemCount));
  339. return this;
  340. }
  341. ICoreRepository ICoreRepository.Search(Func<object,bool> method)
  342. => Search((o) => method(o as TItem));
  343. #endregion
  344. protected virtual Expression<Func<TEntity, object?>>? ImageColumn => null;
  345. #region Loading
  346. private void DoBeforeLoad()
  347. {
  348. _query.Clear();
  349. _query.Add(
  350. EffectiveFilter(),
  351. GetColumns<TItem,TEntity>(),
  352. Sort
  353. );
  354. if (ImageColumn is not null)
  355. {
  356. _query.Add(
  357. new Filter<Document>(x => x.ID).InQuery(EffectiveFilter(), ImageColumn),
  358. Columns.None<Document>().Add(x => x.ID)
  359. .Add(x => x.Data)
  360. );
  361. }
  362. }
  363. protected virtual void BeforeLoad(MultiQuery query)
  364. {
  365. }
  366. protected virtual void AfterLoad(MultiQuery query)
  367. {
  368. }
  369. protected void DoLoad()
  370. {
  371. try
  372. {
  373. var selected = AvailableFilters.FirstOrDefault(x => x.Selected)?.Name;
  374. if (!string.IsNullOrWhiteSpace(FilterTag))
  375. {
  376. var filters = new GlobalConfiguration<CoreFilterDefinitions>(FilterTag).Load()
  377. .Where(x => x.Visibility == CoreFilterDefinitionVisibility.DesktopAndMobile)
  378. .Select(x => new CoreRepositoryFilter() { Name = x.Name, Filter = x.Filter, Selected = string.Equals(x.Name,selected) })
  379. .ToList();
  380. if (filters.Any())
  381. filters.Insert(0, new CoreRepositoryFilter() { Name="All", Filter = "", Selected = !filters.Any(x => string.Equals(x.Name, selected)) });
  382. AvailableFilters.ReplaceRange(filters);
  383. Dispatcher.UIThread.Post(() => OnPropertyChanged(nameof(FiltersVisible)));
  384. }
  385. DoBeforeLoad();
  386. BeforeLoad(_query);
  387. Task.Run(() =>
  388. {
  389. _query.Query();
  390. DoAfterLoad();
  391. }).Wait();
  392. AfterLoad(_query);
  393. Search();
  394. LastUpdated = DateTime.Now;
  395. }
  396. catch (Exception e)
  397. {
  398. MobileLogging.Log(e,"CoreRepository");
  399. }
  400. }
  401. protected void DoAfterLoad()
  402. {
  403. _table = _query.Get<TEntity>();
  404. AllItems.ReplaceRange(_query.Get<TEntity>().Rows.Select(CreateItem<TItem>));
  405. if (ImageColumn != null)
  406. {
  407. Images.Clear();
  408. _query.Get<Document>().IntoDictionary<Document, Guid, byte[]>(Images, x => x.ID,
  409. r => r.Get<Document, byte[]>(x => x.Data));
  410. }
  411. }
  412. #endregion
  413. #region Persistent Storage
  414. protected void InitializeTables()
  415. {
  416. var defs = _query.Definitions();
  417. foreach (var def in defs)
  418. {
  419. var table = InitializeTable(def.Value);
  420. _query.Set(def.Key, table);
  421. }
  422. }
  423. protected CoreTable InitializeTable(IQueryDef def)
  424. {
  425. var table = new CoreTable();
  426. if (def.Columns != null)
  427. table.LoadColumns(def.Columns);
  428. else
  429. table.LoadColumns(def.Type);
  430. return table;
  431. }
  432. protected class QueryStorage : ISerializeBinary
  433. {
  434. private readonly Dictionary<String, CoreTable> _data = new Dictionary<string, CoreTable>();
  435. public CoreTable Get([NotNull] String key) => _data[key];
  436. public void Set([NotNull] String key, CoreTable table) => _data[key] = table;
  437. public bool Contains([NotNull] String key) => _data.ContainsKey(key);
  438. public bool TryGet(string key, [NotNullWhen(true)] out CoreTable? table)
  439. {
  440. return _data.TryGetValue(key, out table);
  441. }
  442. public void SerializeBinary(CoreBinaryWriter writer)
  443. {
  444. writer.Write(_data.Count);
  445. foreach (var key in _data.Keys)
  446. {
  447. writer.Write(key);
  448. _data[key].SerializeBinary(writer);
  449. }
  450. }
  451. public void DeserializeBinary(CoreBinaryReader reader)
  452. {
  453. int count = reader.ReadInt32();
  454. for (int i = 0; i < count; i++)
  455. {
  456. String key = reader.ReadString();
  457. CoreTable table = new CoreTable();
  458. table.DeserializeBinary(reader);
  459. _data[key] = table;
  460. }
  461. }
  462. }
  463. protected bool LoadFromStorage()
  464. {
  465. var filterFileName = FilterFileName();
  466. if (!filterFileName.IsNullOrWhiteSpace())
  467. {
  468. if(CacheManager.TryLoadJSON<ObservableCollection<CoreRepositoryFilter>>(filterFileName, out var filters, out var _))
  469. {
  470. AvailableFilters.ReplaceRange(filters);
  471. Dispatcher.UIThread.Post(() => OnPropertyChanged(nameof(FiltersVisible)));
  472. }
  473. }
  474. var dataFileName = DataFileName();
  475. if (dataFileName.IsNullOrWhiteSpace())
  476. {
  477. InitializeTables();
  478. return true;
  479. }
  480. if(CacheManager.TryLoadBinary<QueryStorage>(dataFileName, out var storage, out var lastUpdated))
  481. {
  482. LastUpdated = lastUpdated;
  483. var defs = _query.Definitions();
  484. foreach (var key in defs.Keys)
  485. {
  486. var keyStr = key.ToString() ?? "";
  487. if(!storage.TryGet(keyStr, out var table))
  488. {
  489. table = InitializeTable(defs[key]);
  490. }
  491. var queryDef = _query.Definitions()[key];
  492. if (CheckColumns(table, queryDef.Type, queryDef.Columns))
  493. _query.Set(key, table);
  494. else
  495. return false;
  496. }
  497. }
  498. else
  499. {
  500. InitializeTables();
  501. }
  502. return true;
  503. }
  504. private bool CheckColumns(CoreTable table, Type T, IColumns? required)
  505. {
  506. required = CoreUtils.GetColumns(T, required);
  507. foreach (var column in required.ColumnNames())
  508. {
  509. if (!table.Columns.Any(x => String.Equals(x.ColumnName, column)))
  510. return false;
  511. }
  512. return true;
  513. }
  514. protected void SaveToStorage()
  515. {
  516. var filterFileName = FilterFileName();
  517. if (!string.IsNullOrWhiteSpace(filterFileName))
  518. {
  519. CacheManager.SaveJSON(filterFileName, AvailableFilters, DateTime.MaxValue);
  520. }
  521. var dataFileName = DataFileName();
  522. if (dataFileName.IsNullOrWhiteSpace())
  523. return;
  524. QueryStorage storage = new QueryStorage();
  525. var results = _query.Results();
  526. foreach (var key in results.Keys)
  527. storage.Set(key.ToString() ?? "", results[key]);
  528. CacheManager.SaveBinary(dataFileName, storage, DateTime.MaxValue);
  529. }
  530. #endregion
  531. #region CRUD Operations
  532. public event CoreRepositoryItemCreatedEvent<TItem>? ItemAdded;
  533. private T CreateItem<T>(CoreRow row)
  534. where T : Shell<TParent,TEntity>, new()
  535. {
  536. var result = new T() { Row = row, Parent = (TParent)this };
  537. result.PropertyChanged += (_, args) => DoPropertyChanged(result, args);
  538. return result;
  539. }
  540. public virtual TItem CreateItem()
  541. {
  542. CoreRow row = _table.NewRow();
  543. var entity = new TEntity();
  544. _table.FillRow(row,entity);
  545. var result = CreateItem<TItem>(row);
  546. ItemAdded?.Invoke(this, new CoreRepositoryItemCreatedArgs<TItem>(result));
  547. return result;
  548. }
  549. public virtual void CommitItem(TItem item)
  550. {
  551. _table.Rows.Add(item.Row);
  552. AllItems.Add(item);
  553. Search();
  554. NotifyChanged();
  555. }
  556. public virtual TItem AddItem()
  557. {
  558. var result = CreateItem();
  559. CommitItem(result);
  560. return result;
  561. }
  562. public virtual void DeleteItem(TItem item)
  563. {
  564. _table.Rows.Remove(item.Row);
  565. AllItems.Remove(item);
  566. Search();
  567. NotifyChanged();
  568. }
  569. public virtual void Save(string auditMessage)
  570. {
  571. new Client<TEntity>().Save(Items.Select(x=>x.Entity).Where(x=>x.IsChanged()),auditMessage);
  572. }
  573. public virtual Task SaveAsync(string auditMessage)
  574. {
  575. return Task.Run(() => Save(auditMessage));
  576. }
  577. object ICoreRepository.CreateItem() => this.CreateItem();
  578. void ICoreRepository.CommitItem(object item)
  579. {
  580. if (item is TItem titem)
  581. CommitItem(titem);
  582. }
  583. object ICoreRepository.AddItem() => this.AddItem();
  584. void ICoreRepository.DeleteItem(object item)
  585. {
  586. if (item is TItem titem)
  587. DeleteItem(titem);
  588. }
  589. #endregion
  590. #region IEnumerable Interface
  591. IEnumerator<TItem> IEnumerable<TItem>.GetEnumerator()
  592. {
  593. return Items.GetEnumerator();
  594. }
  595. public IEnumerator GetEnumerator()
  596. {
  597. return Items.GetEnumerator();
  598. }
  599. #endregion
  600. }
  601. }