DynamicOneToManyGrid.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections.Immutable;
  4. using System.Globalization;
  5. using System.Linq;
  6. using System.Reflection;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. using System.Windows;
  10. using System.Windows.Controls;
  11. using System.Windows.Media.Imaging;
  12. using InABox.Clients;
  13. using InABox.Configuration;
  14. using InABox.Core;
  15. using InABox.Core.Reports;
  16. using InABox.WPF;
  17. namespace InABox.DynamicGrid;
  18. public interface IDynamicOneToManyGrid<TOne, TMany> : IDynamicEditorPage
  19. {
  20. List<TMany> Items { get; }
  21. void LoadItems(TMany[] items);
  22. }
  23. public class DynamicOneToManyGrid<TOne, TMany> : DynamicGrid<TMany>,
  24. IDynamicEditorPage,
  25. IDynamicOneToManyGrid<TOne, TMany>,
  26. IDynamicMemoryEntityGrid<TMany>
  27. where TOne : Entity, new() where TMany : Entity, IPersistent, IRemotable, new()
  28. {
  29. private TMany[] MasterList = Array.Empty<TMany>();
  30. private readonly PropertyInfo property;
  31. public HashSet<string>? LoadedColumns { get; set; }
  32. IEnumerable<TMany> IDynamicMemoryEntityGrid<TMany>.Items => Items;
  33. protected DynamicGridCustomColumnsComponent<TMany> ColumnsComponent;
  34. public virtual bool Visible => Security.CanView<TMany>();
  35. public DynamicOneToManyGrid()
  36. {
  37. Ready = false;
  38. Items = new List<TMany>();
  39. Criteria = new Filters<TMany>();
  40. property = CoreUtils.GetOneToManyProperty(typeof(TMany), typeof(TOne));
  41. AddHiddenColumn(property.Name + "." + nameof(IEntityLink.ID));
  42. foreach (var col in LookupFactory.RequiredColumns<TMany>())
  43. HiddenColumns.Add(col);
  44. ColumnsComponent = new DynamicGridCustomColumnsComponent<TMany>(this, GetTag());
  45. }
  46. protected override void Init()
  47. {
  48. }
  49. protected override void DoReconfigure(DynamicGridOptions options)
  50. {
  51. options.RecordCount = true;
  52. options.SelectColumns = true;
  53. if (Security.CanEdit<TMany>() && !ReadOnly)
  54. {
  55. options.AddRows = true;
  56. options.EditRows = true;
  57. }
  58. if (Security.CanDelete<TMany>() && !ReadOnly)
  59. options.DeleteRows = true;
  60. if (Security.CanImport<TMany>() && !ReadOnly)
  61. options.ImportData = true;
  62. if (Security.CanExport<TMany>())
  63. options.ExportData = true;
  64. if (Security.CanMerge<TMany>())
  65. options.MultiSelect = true;
  66. }
  67. private static bool IsAutoEntity => typeof(TMany).HasAttribute<AutoEntity>();
  68. protected Filters<TMany> Criteria { get; } = new Filters<TMany>();
  69. public TOne Item { get; protected set; }
  70. public List<TMany> Items { get; private set; }
  71. public void LoadItems(TMany[] items)
  72. {
  73. Items.Clear();
  74. Items.AddRange(items);
  75. Refresh(false, true);
  76. }
  77. private static string GetTag()
  78. {
  79. return typeof(TOne).Name + "." + typeof(TMany).Name;
  80. }
  81. #region IDynamicEditorPage
  82. public DynamicEditorGrid EditorGrid { get; set; }
  83. public PageType PageType => PageType.Other;
  84. public bool Ready { get; set; }
  85. private bool _readOnly;
  86. public bool ReadOnly
  87. {
  88. get => _readOnly;
  89. set
  90. {
  91. if (_readOnly != value)
  92. {
  93. _readOnly = value;
  94. Reconfigure();
  95. }
  96. }
  97. }
  98. public virtual void Load(object item, Func<Type, CoreTable?>? PageDataHandler)
  99. {
  100. Reconfigure();
  101. Item = (TOne)item;
  102. Refresh(true, false);
  103. var data = PageDataHandler?.Invoke(typeof(TMany));
  104. if (data == null)
  105. {
  106. if (Item.ID == Guid.Empty)
  107. {
  108. data = new CoreTable();
  109. data.LoadColumns(typeof(TMany));
  110. }
  111. else
  112. {
  113. var criteria = new Filters<TMany>();
  114. var exp = CoreUtils.GetPropertyExpression<TMany>(property.Name + ".ID");
  115. criteria.Add(new Filter<TMany>(exp).IsEqualTo(Item.ID).And(exp).IsNotEqualTo(Guid.Empty));
  116. criteria.AddRange(Criteria.Items);
  117. var sort = LookupFactory.DefineSort<TMany>();
  118. var columns = DynamicGridUtils.LoadEditorColumns(DataColumns());
  119. data = Client.Query(criteria.Combine(), columns, sort);
  120. LoadedColumns = columns.ColumnNames().ToHashSet();
  121. }
  122. }
  123. MasterList = data.ToArray<TMany>();
  124. Items = MasterList.ToList();
  125. Refresh(false, true);
  126. Ready = true;
  127. }
  128. public void Cancel()
  129. {
  130. foreach(var item in MasterList)
  131. {
  132. item.CancelChanges();
  133. }
  134. Items = MasterList.ToList();
  135. Refresh(false, true);
  136. }
  137. public virtual void BeforeSave(object item)
  138. {
  139. // Don't need to do anything here
  140. }
  141. public virtual void AfterSave(object item)
  142. {
  143. if (IsAutoEntity)
  144. {
  145. return;
  146. }
  147. // First remove any deleted files
  148. foreach (var map in MasterList)
  149. if (!Items.Contains(map))
  150. OnDeleteItem(map);
  151. foreach (var map in Items)
  152. {
  153. var prop = (property.GetValue(map) as IEntityLink)!;
  154. prop.ID = Item.ID;
  155. prop.Synchronise(Item);
  156. }
  157. new Client<TMany>().Save(Items.Where(x => x.IsChanged()), "Updated by User");
  158. }
  159. public Size MinimumSize()
  160. {
  161. return new Size(400, 400);
  162. }
  163. public string Caption()
  164. {
  165. var caption = typeof(TMany).GetCustomAttribute(typeof(Caption));
  166. if (caption != null)
  167. return ((Caption)caption).Text;
  168. var result = new Inflector.Inflector(new CultureInfo("en")).Pluralize(typeof(TMany).Name);
  169. return result;
  170. }
  171. public virtual int Order { get; set; } = int.MinValue;
  172. #endregion
  173. #region DynamicGrid
  174. protected virtual void OnDeleteItem(TMany item)
  175. {
  176. if (IsAutoEntity)
  177. {
  178. return;
  179. }
  180. Client.Delete(item, typeof(TMany).Name + " Deleted by User");
  181. }
  182. protected override CoreTable LoadImportKeys(string[] fields)
  183. {
  184. var result = base.LoadImportKeys(fields);
  185. result.LoadRows(MasterList);
  186. return result;
  187. }
  188. protected override bool CustomiseImportItem(TMany item)
  189. {
  190. var result = base.CustomiseImportItem(item);
  191. if (result)
  192. {
  193. var prop = (property.GetValue(item) as IEntityLink)!;
  194. prop.ID = Item.ID;
  195. prop.Synchronise(Item);
  196. }
  197. return result;
  198. }
  199. public override DynamicGridColumns GenerateColumns()
  200. {
  201. var cols = new DynamicGridColumns();
  202. cols.AddRange(base.GenerateColumns().Where(x => !x.ColumnName.StartsWith(property.Name + ".")));
  203. return cols;
  204. }
  205. protected override DynamicGridColumns LoadColumns()
  206. {
  207. return ColumnsComponent.LoadColumns();
  208. }
  209. protected override void SaveColumns(DynamicGridColumns columns)
  210. {
  211. ColumnsComponent.SaveColumns(columns);
  212. }
  213. protected override void LoadColumnsMenu(ContextMenu menu)
  214. {
  215. base.LoadColumnsMenu(menu);
  216. ColumnsComponent.LoadColumnsMenu(menu);
  217. }
  218. protected override DynamicGridSettings LoadSettings()
  219. {
  220. var tag = GetTag();
  221. var user = Task.Run(() => new UserConfiguration<DynamicGridSettings>(tag).Load());
  222. user.Wait();
  223. return user.Result;
  224. }
  225. protected override void SaveSettings(DynamicGridSettings settings)
  226. {
  227. var tag = GetTag();
  228. new UserConfiguration<DynamicGridSettings>(tag).Save(settings);
  229. }
  230. public override TMany CreateItem()
  231. {
  232. var result = new TMany();
  233. var prop = (property.GetValue(result) as IEntityLink)!;
  234. prop.ID = Item.ID;
  235. prop.Synchronise(Item);
  236. return result;
  237. }
  238. public override TMany LoadItem(CoreRow row)
  239. {
  240. return Items[_recordmap[row].Index];
  241. }
  242. public override void SaveItem(TMany item)
  243. {
  244. if (!Items.Contains(item))
  245. Items.Add(item);
  246. if (item is ISequenceable) Items = Items.AsQueryable().OrderBy(x => (x as ISequenceable)!.Sequence).ToList();
  247. }
  248. public override void DeleteItems(params CoreRow[] rows)
  249. {
  250. var items = rows.Select(LoadItem).ToList();
  251. foreach (var item in items)
  252. {
  253. Items.Remove(item);
  254. }
  255. }
  256. protected override void Reload(
  257. Filters<TMany> criteria, Columns<TMany> columns, ref SortOrder<TMany>? sort,
  258. CancellationToken token, Action<CoreTable?, Exception?> action)
  259. {
  260. var results = new CoreTable();
  261. results.LoadColumns(typeof(TMany));
  262. this.EnsureColumns(columns);
  263. if (sort != null)
  264. {
  265. var exp = IQueryableExtensions.ToLambda<TMany>(sort.Expression);
  266. var sorted = sort.Direction == SortDirection.Ascending
  267. ? Items.AsQueryable().OrderBy(exp)
  268. : Items.AsQueryable().OrderByDescending(exp);
  269. foreach (var then in sort.Thens)
  270. {
  271. var thexp = IQueryableExtensions.ToLambda<TMany>(then.Expression);
  272. sorted = sort.Direction == SortDirection.Ascending ? sorted.ThenBy(exp) : sorted.ThenByDescending(exp);
  273. }
  274. Items = sorted.ToList();
  275. }
  276. results.LoadRows(Items);
  277. action.Invoke(results, null);
  278. }
  279. protected override BaseEditor? GetEditor(object item, DynamicGridColumn column)
  280. {
  281. var type = CoreUtils.GetProperty(typeof(TMany), column.ColumnName).DeclaringType;
  282. if (type.GetInterfaces().Contains(typeof(IEntityLink)) && type.ContainsInheritedGenericType(typeof(TOne)))
  283. return new NullEditor();
  284. return base.GetEditor(item, column);
  285. }
  286. public override void LoadEditorButtons(TMany item, DynamicEditorButtons buttons)
  287. {
  288. base.LoadEditorButtons(item, buttons);
  289. if (ClientFactory.IsSupported<AuditTrail>())
  290. buttons.Add("Audit Trail", Wpf.Resources.view.AsBitmapImage(), item, AuditTrailClick);
  291. }
  292. private void AuditTrailClick(object sender, object? item)
  293. {
  294. if (item is not TMany entity) return;
  295. var window = new AuditWindow(entity.ID);
  296. window.ShowDialog();
  297. }
  298. public override DynamicEditorPages LoadEditorPages(TMany item)
  299. {
  300. return item.ID != Guid.Empty ? base.LoadEditorPages(item) : new DynamicEditorPages();
  301. }
  302. protected override bool BeforeCopy(IList<TMany> items)
  303. {
  304. if (!base.BeforeCopy(items)) return false;
  305. for(int i = 0; i < items.Count; ++i)
  306. {
  307. var newItem = items[i].Clone();
  308. newItem.ID = Guid.Empty;
  309. items[i] = newItem;
  310. }
  311. return true;
  312. }
  313. #endregion
  314. }