DetailModel.cs 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. using System;
  2. using System.IO;
  3. using System.Linq;
  4. using System.Linq.Expressions;
  5. using System.Threading;
  6. using InABox.Clients;
  7. using InABox.Core;
  8. using System.Diagnostics.CodeAnalysis;
  9. namespace comal.timesheets
  10. {
  11. public abstract class DetailModel<TParent, TItem, TEntity> : Model<TParent,TItem,TEntity>, IDetailModel<TParent, TItem, TEntity>
  12. where TParent : DetailModel<TParent, TItem, TEntity>, IModel
  13. where TEntity : Entity, IRemotable, IPersistent, new()
  14. where TItem : DetailShell<TParent, TEntity>, new()
  15. {
  16. protected DetailModel(IModelHost host, Func<Filter<TEntity>> filter, bool transient = false) : base(host, filter, transient)
  17. {
  18. }
  19. protected DetailModel(IModelHost host, Func<Filter<TEntity>> filter, [NotNull] String filename) : base(host, filter, filename)
  20. {
  21. }
  22. protected override void Initialize()
  23. {
  24. Filter = null;
  25. Item = null;
  26. }
  27. private TItem _item;
  28. public TItem Item
  29. {
  30. get => _item;
  31. set => SetProperty(ref _item, value);
  32. }
  33. protected virtual Expression<Func<TEntity, object>> ImageColumn { get; }
  34. private void DoBeforeLoad(MultiQuery query)
  35. {
  36. if (ImageColumn != null)
  37. query.Add<Document>(
  38. new Filter<Document>(x => x.ID).InQuery<TEntity>(Filter(),ImageColumn),
  39. new Columns<Document>(x=>x.ID).Add(x=>x.Data)
  40. );
  41. BeforeLoad(query);
  42. }
  43. public override void BeforeLoad(MultiQuery query)
  44. {
  45. }
  46. private void DoAfterLoad(MultiQuery query, Action loaded = null)
  47. {
  48. if (ImageColumn != null)
  49. {
  50. Images.Clear();
  51. query.Get<Document>().IntoDictionary<Document, Guid, byte[]>(Images, x => x.ID,
  52. r => r.Get<Document, byte[]>(x => x.Data));
  53. }
  54. AfterLoad(query);
  55. Loaded = true;
  56. loaded?.Invoke();
  57. }
  58. public override void AfterLoad(MultiQuery query)
  59. {
  60. }
  61. private TItem CreateItemFromEntity(TEntity entity)
  62. {
  63. CoreTable table = new CoreTable();
  64. table.LoadColumns(Columns);
  65. CoreRow row = table.NewRow();
  66. if (entity != null)
  67. table.LoadRow(row, entity);
  68. var item = new TItem() { Row = row, Parent = this as TParent };
  69. item.PropertyChanged += DoPropertyChanged;
  70. return item;
  71. }
  72. private TItem CreateItemFromRow(CoreRow row)
  73. {
  74. var item = (row != null)
  75. ? new TItem() { Row = row, Parent = this as TParent }
  76. : CreateItemFromEntity(null);
  77. item.PropertyChanged += DoPropertyChanged;
  78. return item;
  79. }
  80. public void Load(TEntity entity, Action loaded = null)
  81. {
  82. Filter = () => new Filter<TEntity>(x => x.ID).IsEqualTo(entity.ID);
  83. MultiQuery query = new MultiQuery();
  84. DoBeforeLoad(query);
  85. Item = CreateItemFromEntity(entity);
  86. if (loaded != null)
  87. query.Query((q) => DoAfterLoad(q, loaded));
  88. else
  89. {
  90. query.Query();
  91. DoAfterLoad(query);
  92. }
  93. }
  94. public override void Load(Action loaded = null)
  95. {
  96. MultiQuery query = new MultiQuery();
  97. query.Add(
  98. Filter(),
  99. Columns
  100. );
  101. DoBeforeLoad(query);
  102. if (Host.IsConnected())
  103. {
  104. if (loaded != null)
  105. query.Query((q) =>
  106. {
  107. if (Type == ModelType.Persistent)
  108. SaveToStorage(query);
  109. Item = CreateItemFromRow(q.Get<TEntity>().Rows.FirstOrDefault());
  110. DoAfterLoad(q, loaded);
  111. });
  112. else
  113. {
  114. query.Query();
  115. if (Type == ModelType.Persistent)
  116. SaveToStorage(query);
  117. Item = CreateItemFromRow(query.Get<TEntity>().Rows.FirstOrDefault());
  118. DoAfterLoad(query);
  119. }
  120. }
  121. else
  122. {
  123. if (Type == ModelType.Transient)
  124. {
  125. InitializeTables(query);
  126. }
  127. else if (Type == ModelType.Normal)
  128. {
  129. // Only load
  130. if (_item == null)
  131. InitializeTables(query);
  132. }
  133. else if (Type == ModelType.Persistent)
  134. {
  135. // Treat it as normal, unless its the first time through
  136. // in which case try to load it from storage, if the
  137. // data has been previously cached
  138. if (_item == null)
  139. LoadFromStorage(query);
  140. }
  141. Item = CreateItemFromRow(query.Get<TEntity>().Rows.FirstOrDefault());
  142. DoAfterLoad(query, loaded);
  143. }
  144. }
  145. protected virtual void BeforeSave(TItem item)
  146. {
  147. }
  148. protected virtual void AfterSave(TItem item)
  149. {
  150. }
  151. public void Save(String auditmessage)
  152. {
  153. BeforeSave(Item);
  154. new Client<TEntity>().Save(Item.Entity,auditmessage);
  155. AfterSave(Item);
  156. }
  157. public override void Refresh(bool force, Action loaded = null)
  158. {
  159. if (!Loaded || force)
  160. Load(loaded);
  161. else
  162. loaded?.Invoke();
  163. }
  164. }
  165. }