DynamicOneToManyGrid.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  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. {
  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>, IDynamicEditorPage, IDynamicOneToManyGrid<TOne, TMany>
  24. where TOne : Entity, new() where TMany : Entity, IPersistent, IRemotable, new()
  25. {
  26. private TMany[] MasterList = Array.Empty<TMany>();
  27. private readonly PropertyInfo property;
  28. public PageType PageType => PageType.Other;
  29. public DynamicOneToManyGrid()
  30. {
  31. Ready = false;
  32. Items = new List<TMany>();
  33. Criteria = new Filters<TMany>();
  34. property = CoreUtils.GetOneToManyProperty(typeof(TMany), typeof(TOne));
  35. /*var t = typeof(TMany).BaseType;
  36. if (t != null && t.IsConstructedGenericType && t.GetGenericTypeDefinition() == typeof(EntityForm<,>))
  37. {
  38. ActionColumns.Add(new DynamicImageColumn(QAEditImage, QAEditClick));
  39. if (DynamicGridUtils.PreviewReport != null)
  40. ActionColumns.Add(new DynamicImageColumn(QAPrintImage, QAPrintClick));
  41. HiddenColumns.Add(x => (x as IDigitalFormInstance).FormData);
  42. HiddenColumns.Add(x => (x as IDigitalFormInstance).FormCompleted);
  43. HiddenColumns.Add(x => (x as IDigitalFormInstance).FormCompletedBy.ID);
  44. HiddenColumns.Add(x => (x as IDigitalFormInstance).Form.ID);
  45. }*/
  46. }
  47. protected override void Init()
  48. {
  49. }
  50. protected override void DoReconfigure(FluentList<DynamicGridOption> options)
  51. {
  52. options.BeginUpdate();
  53. options.Add(DynamicGridOption.RecordCount)
  54. .Add(DynamicGridOption.SelectColumns);
  55. if (Security.CanEdit<TMany>() && !ReadOnly)
  56. options.Add(DynamicGridOption.AddRows).Add(DynamicGridOption.EditRows);
  57. if (Security.CanDelete<TMany>() && !ReadOnly)
  58. options.Add(DynamicGridOption.DeleteRows);
  59. if (Security.CanImport<TMany>() && !ReadOnly)
  60. options.Add(DynamicGridOption.ImportData);
  61. if (Security.CanExport<TMany>())
  62. options.Add(DynamicGridOption.ExportData);
  63. if (Security.CanMerge<TMany>())
  64. options.Add(DynamicGridOption.MultiSelect);
  65. options.EndUpdate();
  66. }
  67. private bool _readOnly;
  68. public bool ReadOnly
  69. {
  70. get => _readOnly;
  71. set
  72. {
  73. if(_readOnly != value)
  74. {
  75. _readOnly = value;
  76. Reconfigure();
  77. }
  78. }
  79. }
  80. private static bool IsAutoEntity => typeof(TMany).HasAttribute<AutoEntity>();
  81. protected Filters<TMany> Criteria { get; } = new Filters<TMany>();
  82. public TOne Item { get; protected set; }
  83. public bool Ready { get; set; }
  84. public DynamicEditorGrid EditorGrid { get; set; }
  85. public string Caption()
  86. {
  87. var caption = typeof(TMany).GetCustomAttribute(typeof(Caption));
  88. if (caption != null)
  89. return ((Caption)caption).Text;
  90. var result = new Inflector.Inflector(new CultureInfo("en")).Pluralize(typeof(TMany).Name);
  91. return result;
  92. }
  93. public virtual int Order()
  94. {
  95. return int.MinValue;
  96. }
  97. public virtual void Load(object item, Func<Type, CoreTable?>? PageDataHandler)
  98. {
  99. Reconfigure();
  100. Item = (TOne)item;
  101. var data = PageDataHandler?.Invoke(typeof(TMany));
  102. if (data == null)
  103. {
  104. if (Item.ID == Guid.Empty)
  105. {
  106. data = new CoreTable();
  107. data.LoadColumns(typeof(TMany));
  108. }
  109. else
  110. {
  111. var criteria = new Filters<TMany>();
  112. var exp = CoreUtils.GetPropertyExpression<TMany>(property.Name + ".ID");
  113. criteria.Add(new Filter<TMany>(exp).IsEqualTo(Item.ID).And(exp).IsNotEqualTo(Guid.Empty));
  114. criteria.AddRange(Criteria.Items);
  115. var sort = LookupFactory.DefineSort<TMany>();
  116. data = new Client<TMany>().Query(criteria.Combine(), null, sort);
  117. }
  118. }
  119. MasterList = data.Rows.Select(x => x.ToObject<TMany>()).ToArray();
  120. Items = MasterList.ToList();
  121. Refresh(true, true);
  122. Ready = true;
  123. }
  124. public virtual void BeforeSave(object item)
  125. {
  126. // Don't need to do anything here
  127. }
  128. protected virtual void OnDeleteItem(TMany item)
  129. {
  130. if (IsAutoEntity)
  131. {
  132. return;
  133. }
  134. new Client<TMany>().Delete(item, typeof(TMany).Name + " Deleted by User");
  135. }
  136. public void AfterSave(object item)
  137. {
  138. if (IsAutoEntity)
  139. {
  140. return;
  141. }
  142. // First remove any deleted files
  143. foreach (var map in MasterList)
  144. if (!Items.Contains(map))
  145. OnDeleteItem(map);
  146. foreach (var map in Items)
  147. {
  148. var prop = (property.GetValue(map) as IEntityLink)!;
  149. prop.ID = Item.ID;
  150. prop.Synchronise(Item);
  151. //if (map.IsChanged())
  152. // new Client<TMany>().Save(map, "Updated by User");
  153. }
  154. new Client<TMany>().Save(Items.Where(x => x.IsChanged()), "Updated by User");
  155. }
  156. protected override CoreTable LoadImportKeys(string[] fields)
  157. {
  158. var result = base.LoadImportKeys(fields);
  159. result.LoadRows(MasterList);
  160. return result;
  161. }
  162. protected override bool CustomiseImportItem(TMany item)
  163. {
  164. var result = base.CustomiseImportItem(item);
  165. if (result)
  166. {
  167. var prop = (property.GetValue(item) as IEntityLink)!;
  168. prop.ID = Item.ID;
  169. prop.Synchronise(Item);
  170. }
  171. return result;
  172. }
  173. public Size MinimumSize()
  174. {
  175. return new Size(400, 400);
  176. }
  177. public List<TMany> Items { get; private set; }
  178. public void LoadItems(TMany[] items)
  179. {
  180. Items.Clear();
  181. Items.AddRange(items);
  182. Refresh(false, true);
  183. }
  184. private static string GetTag()
  185. {
  186. return typeof(TOne).Name + "." + typeof(TMany).Name;
  187. }
  188. protected override DynamicGridColumns LoadColumns()
  189. {
  190. var tag = GetTag();
  191. var global = Task.Run(() => new GlobalConfiguration<DynamicGridColumns>(tag).Load());
  192. var user = Task.Run(() => new UserConfiguration<DynamicGridColumns>(tag).Load());
  193. Task.WaitAll(global, user);
  194. var columns = user.Result.Any() ? user.Result : global.Result;
  195. if (columns.Count == 0)
  196. columns.AddRange(base.LoadColumns().Where(x => !x.ColumnName.StartsWith(property.Name)));
  197. return columns;
  198. }
  199. protected override void SaveColumns(DynamicGridColumns columns)
  200. {
  201. var tag = GetTag();
  202. new UserConfiguration<DynamicGridColumns>(tag).Save(columns);
  203. }
  204. protected override void LoadColumnsMenu(ContextMenu menu)
  205. {
  206. base.LoadColumnsMenu(menu);
  207. //menu.Items.Add(new Separator());
  208. var ResetColumns = new MenuItem { Header = "Reset Columns to Default" };
  209. ResetColumns.Click += ResetColumnsClick;
  210. menu.Items.Add(ResetColumns);
  211. if (Security.IsAllowed<CanSetDefaultColumns>())
  212. {
  213. menu.Items.Add(new Separator());
  214. var UpdateDefaultColumns = new MenuItem { Header = "Mark Columns as Default" };
  215. UpdateDefaultColumns.Click += UpdateDefaultColumnsClick;
  216. menu.Items.Add(UpdateDefaultColumns);
  217. }
  218. }
  219. private void ResetColumnsClick(object sender, RoutedEventArgs e)
  220. {
  221. VisibleColumns.Clear();
  222. SaveColumns(VisibleColumns);
  223. Refresh(true, true);
  224. }
  225. private void UpdateDefaultColumnsClick(object sender, RoutedEventArgs e)
  226. {
  227. var tag = GetTag();
  228. new GlobalConfiguration<DynamicGridColumns>(tag).Save(VisibleColumns);
  229. new UserConfiguration<DynamicGridColumns>(tag).Delete();
  230. Refresh(true, true);
  231. }
  232. protected override DynamicGridSettings LoadSettings()
  233. {
  234. var tag = GetTag();
  235. var user = Task.Run(() => new UserConfiguration<DynamicGridSettings>(tag).Load());
  236. user.Wait();
  237. //var global = Task.Run(() => new GlobalConfiguration<DynamicGridSettings>(tag).Load());
  238. //global.Wait();
  239. //Task.WaitAll(user, global);
  240. //var columns = user.Result.Any() ? user.Result : global.Result;
  241. return user.Result;
  242. }
  243. protected override void SaveSettings(DynamicGridSettings settings)
  244. {
  245. var tag = GetTag();
  246. new UserConfiguration<DynamicGridSettings>(tag).Save(settings);
  247. }
  248. protected override TMany CreateItem()
  249. {
  250. var result = new TMany();
  251. var prop = (property.GetValue(result) as IEntityLink)!;
  252. prop.ID = Item.ID;
  253. prop.Synchronise(Item);
  254. return result;
  255. }
  256. protected override TMany LoadItem(CoreRow row)
  257. {
  258. return Items[_recordmap[row].Index];
  259. }
  260. protected override TMany[] LoadItems(CoreRow[] rows)
  261. {
  262. var result = new List<TMany>();
  263. foreach (var row in rows)
  264. result.Add(LoadItem(row));
  265. return result.ToArray();
  266. }
  267. public override void SaveItem(TMany item)
  268. {
  269. if (!Items.Contains(item))
  270. Items.Add(item);
  271. if (item is ISequenceable) Items = Items.AsQueryable().OrderBy(x => (x as ISequenceable)!.Sequence).ToList();
  272. //var sort = LookupFactory.DefineSort<TMany>();
  273. //if (sort != null)
  274. // Items = Items.AsQueryable().SortBy(sort.Expression).ToList();
  275. }
  276. protected override void DeleteItems(params CoreRow[] rows)
  277. {
  278. var items = rows.Select(LoadItem).ToList();
  279. foreach(var item in items)
  280. {
  281. Items.Remove(item);
  282. }
  283. }
  284. protected override void Reload(Filters<TMany> criteria, Columns<TMany> columns, ref SortOrder<TMany>? sort,
  285. Action<CoreTable?, Exception?> action)
  286. {
  287. //var exp = BaseObject.DefaultSortOrder<TMany>().Expression;
  288. //Items = Items.AsQueryable().SortBy(exp).To
  289. var results = new CoreTable();
  290. results.LoadColumns(typeof(TMany));
  291. if (sort != null)
  292. {
  293. var exp = IQueryableExtensions.ToLambda<TMany>(sort.Expression);
  294. var sorted = sort.Direction == SortDirection.Ascending
  295. ? Items.AsQueryable().OrderBy(exp)
  296. : Items.AsQueryable().OrderByDescending(exp);
  297. foreach (var then in sort.Thens)
  298. {
  299. var thexp = IQueryableExtensions.ToLambda<TMany>(then.Expression);
  300. sorted = sort.Direction == SortDirection.Ascending ? sorted.ThenBy(exp) : sorted.ThenByDescending(exp);
  301. }
  302. Items = sorted.ToList();
  303. }
  304. results.LoadRows(Items);
  305. //if (sort != null)
  306. // results.LoadRows(Items.AsQueryable().SortBy(sort.Expression));
  307. //else
  308. // results.LoadRows(Items.OrderBy(x=>x.Sort));
  309. action.Invoke(results, null);
  310. }
  311. protected override BaseEditor? GetEditor(object item, DynamicGridColumn column)
  312. {
  313. var type = CoreUtils.GetProperty(typeof(TMany), column.ColumnName).DeclaringType;
  314. if (type.GetInterfaces().Contains(typeof(IEntityLink)) && type.ContainsInheritedGenericType(typeof(TOne)))
  315. return new NullEditor();
  316. return base.GetEditor(item, column);
  317. }
  318. //protected override void EditorValueChanged(object item, string name, object value, List<String> changes)
  319. //{
  320. // Entity entity = (Entity)item;
  321. // Dictionary<String, String> previous = new Dictionary<string, string>();
  322. // if (entity.OriginalValues != null)
  323. // {
  324. // foreach (var key in entity.OriginalValues.Keys)
  325. // previous[key] = entity.OriginalValues.ToString();
  326. // }
  327. // base.EditorValueChanged(item, name, value, changes);
  328. // if (entity.OriginalValues != null)
  329. // {
  330. // foreach (var key in entity.OriginalValues.Keys)
  331. // {
  332. // if (key != name)
  333. // {
  334. // String oldval = entity.OriginalValues[key] != null ? entity.OriginalValues[key].ToString() : "";
  335. // if ((!previous.ContainsKey(key)) || (!previous[key].Equals(oldval)))
  336. // {
  337. // if (!changes.Contains(key))
  338. // changes.Add(key);
  339. // }
  340. // }
  341. // }
  342. // }
  343. //}
  344. /*private BitmapImage QAPrintImage(CoreRow arg)
  345. {
  346. return Wpf.Resources.print.AsBitmapImage();
  347. }
  348. private bool QAPrintClick(CoreRow arg)
  349. {
  350. var formid = arg.Get<TMany, Guid>(x => (x as IDigitalFormInstance).Form.ID);
  351. var model = new DigitalFormReportDataModel<TMany>(new Filter<TMany>("Parent.ID").IsEqualTo(Item.ID), formid);
  352. var section = formid.ToString();
  353. // TODO: This is a hack
  354. DynamicGridUtils.PrintMenu?.Invoke(null, section, model, true);
  355. return false;
  356. }*/
  357. /*private BitmapImage QAEditImage(CoreRow arg)
  358. {
  359. if (arg == null)
  360. return Wpf.Resources.pencil.AsBitmapImage();
  361. var completed = arg.Get<TMany, DateTime>(x => (x as IBaseDigitalFormInstance).FormCompleted);
  362. return completed.IsEmpty() ? Wpf.Resources.pencil.AsBitmapImage() : Wpf.Resources.view.AsBitmapImage();
  363. }
  364. private bool QAEditClick(CoreRow arg)
  365. {
  366. var item = LoadItem(arg);
  367. var result = DynamicFormEditWindow.EditDigitalForm(item as IDigitalFormInstance);
  368. if (result)
  369. SaveItem(item);
  370. return result;
  371. }*/
  372. public override DynamicEditorPages LoadEditorPages(TMany item)
  373. {
  374. return item.ID != Guid.Empty ? base.LoadEditorPages(item) : new DynamicEditorPages();
  375. }
  376. protected override bool BeforePaste(IEnumerable<TMany> items, ClipAction action)
  377. {
  378. if(action == ClipAction.Copy)
  379. {
  380. foreach(var item in items)
  381. {
  382. item.ID = Guid.Empty;
  383. }
  384. }
  385. return base.BeforePaste(items, action);
  386. }
  387. }
  388. }