DynamicOneToManyGrid.cs 13 KB

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