DynamicOneToManyGrid.cs 11 KB

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