DynamicEditorGrid.xaml.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Diagnostics.CodeAnalysis;
  4. using System.Linq;
  5. using System.Windows;
  6. using System.Windows.Controls;
  7. using System.Windows.Media;
  8. using InABox.Clients;
  9. using InABox.Core;
  10. using InABox.Wpf;
  11. using InABox.WPF;
  12. using NPOI.HSSF.Record.Aggregates;
  13. namespace InABox.DynamicGrid;
  14. public delegate void OnUpdateOtherEditorHandler(string columnname, object value);
  15. public delegate Dictionary<string, object?> EditorValueChangedHandler(IDynamicEditorForm sender, string name, object value);
  16. /// <summary>
  17. /// Interaction logic for DynamicEditorGrid.xaml
  18. /// </summary>
  19. public partial class DynamicEditorGrid : UserControl, IDynamicEditorHost
  20. {
  21. public delegate void EditorCreatedHandler(object sender, double height, double width);
  22. public delegate object? GetPropertyValueHandler(object sender, string name);
  23. public delegate void SetPropertyValueHandler(object sender, string name, object value);
  24. public delegate BaseObject[] GetItemsEvent();
  25. // Column Definitions as defined by calling model
  26. private DynamicGridColumns _columns = new();
  27. private Type? LayoutType;
  28. private DynamicEditorGridLayout? Layout;
  29. private bool _tabStripVisible = true;
  30. public bool TabStripVisible
  31. {
  32. get { return _tabStripVisible; }
  33. set
  34. {
  35. _tabStripVisible = value;
  36. if (Layout != null)
  37. Layout.TabStripVisible = value;
  38. }
  39. }
  40. public double TotalWidth => Layout?.TotalWidth ?? 0;
  41. public double TotalHeight => Layout?.TotalHeight ?? 0;
  42. public DynamicEditorGrid()
  43. {
  44. InitializeComponent();
  45. Loaded += DynamicEditorGrid_Loaded;
  46. }
  47. private DynamicEditorPages _pages = new();
  48. public IEnumerable<IDynamicEditorPage> Pages => _pages;
  49. private void AddPage(IDynamicEditorPage page)
  50. {
  51. page.ReadOnly = ReadOnly;
  52. _pages.Add(page);
  53. }
  54. private void SetPages(DynamicEditorPages pages)
  55. {
  56. _pages = pages;
  57. foreach (var page in _pages)
  58. {
  59. page.ReadOnly = ReadOnly;
  60. }
  61. }
  62. public bool PreloadPages { get; set; }
  63. public Type UnderlyingType { get; set; }
  64. public OnLoadPage? OnLoadPage { get; set; }
  65. public event OnSelectPage? OnSelectPage;
  66. public event OnUnloadPage? OnUnloadPage;
  67. private bool _readOnly;
  68. public bool ReadOnly
  69. {
  70. get => _readOnly;
  71. set
  72. {
  73. _readOnly = value;
  74. foreach(var page in Pages)
  75. {
  76. page.ReadOnly = value;
  77. }
  78. }
  79. }
  80. public IEnumerable<IDynamicEditorControl> Editors
  81. {
  82. get
  83. {
  84. foreach (var page in Pages)
  85. {
  86. if (page is DynamicEditPage editPage)
  87. {
  88. foreach(var editor in editPage.Editors)
  89. {
  90. yield return editor;
  91. }
  92. }
  93. }
  94. }
  95. }
  96. public bool TryFindEditor(string columnname, [NotNullWhen(true)] out IDynamicEditorControl? editor)
  97. {
  98. foreach (var page in Pages)
  99. {
  100. if (page is DynamicEditPage editPage)
  101. {
  102. if (editPage.TryFindEditor(columnname, out editor))
  103. return true;
  104. }
  105. }
  106. editor = null;
  107. return false;
  108. }
  109. public IDynamicEditorControl? FindEditor(string columnname)
  110. {
  111. TryFindEditor(columnname, out var editor);
  112. return editor;
  113. }
  114. public virtual void ReconfigureEditors()
  115. {
  116. OnReconfigureEditors?.Invoke(this);
  117. }
  118. public object? GetPropertyValue(string columnname)
  119. {
  120. return OnGetPropertyValue?.Invoke(this, columnname);
  121. }
  122. public event EditorCreatedHandler? OnEditorCreated;
  123. public event OnCustomiseColumns? OnCustomiseColumns;
  124. public event OnGetEditor? OnGetEditor;
  125. public event OnGridCustomiseEditor? OnGridCustomiseEditor;
  126. public event OnGetEditorSequence? OnGetSequence;
  127. public event GetPropertyValueHandler? OnGetPropertyValue;
  128. public event SetPropertyValueHandler? OnSetPropertyValue;
  129. public delegate Dictionary<string, object?> EditorGridValueChangedHandler(DynamicEditorGrid sender, string name, object? value);
  130. public event EditorGridValueChangedHandler? OnEditorValueChanged;
  131. public event OnAfterEditorValueChanged? OnAfterEditorValueChanged;
  132. public event OnReconfigureEditors? OnReconfigureEditors;
  133. public event OnDefineLookupFilter? OnDefineFilter;
  134. public event OnDefineLookup? OnDefineLookups;
  135. public event GetItemsEvent? GetItems;
  136. private void DynamicEditorGrid_Loaded(object sender, RoutedEventArgs e)
  137. {
  138. //Reload();
  139. }
  140. public void Reload()
  141. {
  142. LoadPages();
  143. ReconfigureEditors();
  144. }
  145. #region Host Implementation
  146. public void LoadLookups(ILookupEditorControl editor)
  147. {
  148. OnDefineLookups?.Invoke(editor);
  149. }
  150. BaseObject[] IDynamicEditorHost.GetItems() => GetItems?.Invoke() ?? Array.Empty<BaseObject>();
  151. public BaseEditor? GetEditor(DynamicGridColumn column) => OnGetEditor?.Invoke(column);
  152. public Type GetEditorType() => UnderlyingType;
  153. #endregion
  154. #region Edit Page
  155. public class DynamicEditPage : ContentControl, IDynamicEditorPage
  156. {
  157. private Grid Grid;
  158. public DynamicEditorGrid EditorGrid { get; set; } = null!; // Set by DynamicEditorGrid
  159. public bool Ready { get; set; }
  160. private List<BaseDynamicEditorControl> EditorList { get; set; }
  161. public IEnumerable<IDynamicEditorControl> Editors => EditorList;
  162. public PageType PageType => PageType.Editor;
  163. public int PageOrder { get; set; }
  164. public string Header { get; set; }
  165. private double GeneralHeight = 30;
  166. private bool _readOnly;
  167. public bool ReadOnly
  168. {
  169. get => _readOnly;
  170. set
  171. {
  172. if(_readOnly != value)
  173. {
  174. _readOnly = value;
  175. foreach(var editor in EditorList)
  176. {
  177. editor.IsEnabled = !value && editor.EditorDefinition.Editable.IsEditable();
  178. }
  179. }
  180. }
  181. }
  182. public DynamicEditPage(string header)
  183. {
  184. Header = header;
  185. EditorList = [];
  186. InitialiseContent();
  187. }
  188. public void AddEditor(string columnName, BaseEditor editor)
  189. {
  190. BaseDynamicEditorControl? element = DynamicEditorControlFactory.CreateControl(editor, EditorGrid);
  191. if (element != null)
  192. {
  193. element.IsEnabled = !ReadOnly && editor.Editable.IsEditable();
  194. if (!string.IsNullOrWhiteSpace(editor.ToolTip))
  195. {
  196. element.ToolTip = new ToolTip() { Content = editor.ToolTip };
  197. }
  198. var label = new Label();
  199. label.Content = CoreUtils.Neatify(editor.Caption); // 2
  200. label.Margin = new Thickness(0F, 0F, 0F, 0F);
  201. label.HorizontalAlignment = HorizontalAlignment.Stretch;
  202. label.VerticalAlignment = VerticalAlignment.Stretch;
  203. label.HorizontalContentAlignment = HorizontalAlignment.Left;
  204. label.VerticalContentAlignment = VerticalAlignment.Center;
  205. label.SetValue(Grid.RowProperty, Grid.RowDefinitions.Count);
  206. label.SetValue(Grid.ColumnProperty, 0);
  207. label.Visibility = string.IsNullOrWhiteSpace(editor.Caption) ? Visibility.Collapsed : Visibility.Visible;
  208. Grid.Children.Add(label);
  209. element.ColumnName = columnName;
  210. element.Color = editor is UniqueCodeEditor ? Color.FromArgb(0xFF, 0xF6, 0xC9, 0xE8) : Colors.LightYellow;
  211. EditorList.Add(element);
  212. element.Margin = new Thickness(5F, 2.5F, 5F, 2.5F);
  213. double iHeight = element.DesiredHeight();
  214. if (iHeight == int.MaxValue)
  215. {
  216. Grid.RowDefinitions.Add(new RowDefinition { Height = new GridLength(1, GridUnitType.Star) });
  217. GeneralHeight += element.MinHeight + 5.0F;
  218. }
  219. else
  220. {
  221. Grid.RowDefinitions.Add(new RowDefinition { Height = new GridLength(iHeight + 5.0F) });
  222. GeneralHeight += iHeight + 5.0F;
  223. }
  224. double iWidth = element.EditorDefinition?.Information != null ? int.MaxValue : element.DesiredWidth();
  225. if (iWidth == int.MaxValue)
  226. {
  227. element.HorizontalAlignment = HorizontalAlignment.Stretch;
  228. }
  229. else
  230. {
  231. element.HorizontalAlignment = HorizontalAlignment.Left;
  232. element.Width = iWidth;
  233. }
  234. element.SetValue(Grid.RowProperty, Grid.RowDefinitions.Count - 1);
  235. element.SetValue(Grid.ColumnProperty, 1);
  236. Grid.Children.Add(element);
  237. }
  238. }
  239. [MemberNotNull(nameof(Grid))]
  240. private void InitialiseContent()
  241. {
  242. Grid = new Grid
  243. {
  244. HorizontalAlignment = HorizontalAlignment.Stretch,
  245. VerticalAlignment = VerticalAlignment.Stretch,
  246. Margin = new Thickness(0, 2.5, 0, 2.5)
  247. };
  248. Grid.ColumnDefinitions.Add(new ColumnDefinition { Width = new GridLength(1, GridUnitType.Auto) });
  249. Grid.ColumnDefinitions.Add(new ColumnDefinition { Width = new GridLength(1, GridUnitType.Star) });
  250. var scroll = new ScrollViewer
  251. {
  252. HorizontalAlignment = HorizontalAlignment.Stretch,
  253. VerticalAlignment = VerticalAlignment.Stretch,
  254. VerticalScrollBarVisibility = ScrollBarVisibility.Auto,
  255. Padding = new Thickness(2),
  256. Content = Grid
  257. };
  258. var border = new Border
  259. {
  260. BorderBrush = new SolidColorBrush(Colors.Gray),
  261. Background = new SolidColorBrush(Colors.White),
  262. BorderThickness = new Thickness(0.75),
  263. Child = scroll
  264. };
  265. Content = border;
  266. }
  267. public void AfterSave(object item)
  268. {
  269. }
  270. public event EventHandler? OnChanged;
  271. public void BeforeSave(object item)
  272. {
  273. }
  274. public string Caption() => Header;
  275. public bool TryFindEditor(string columnname, [NotNullWhen(true)] out IDynamicEditorControl? editor)
  276. {
  277. editor = EditorList.FirstOrDefault(x => x.ColumnName.Equals(columnname));
  278. editor ??= EditorList.FirstOrDefault(x => columnname.StartsWith(x.ColumnName + '.'));
  279. return editor is not null;
  280. }
  281. public IEnumerable<BaseDynamicEditorControl> FindEditors(DynamicGridColumn column)
  282. {
  283. return EditorList.Where(x => string.Equals(x.ColumnName, column.ColumnName));
  284. }
  285. #region Configure Editors
  286. private void Lookup_OnUpdateOtherEditor(string columnname, object value)
  287. {
  288. var editor = EditorList.FirstOrDefault(x => x.ColumnName.Equals(columnname));
  289. if (editor != null)
  290. CoreUtils.SetPropertyValue(editor, "Value", value);
  291. }
  292. private void ConfigureEditors()
  293. {
  294. foreach (var editor in EditorList)
  295. {
  296. editor.Configure();
  297. editor.Loaded = true;
  298. }
  299. }
  300. #endregion
  301. private void EditorValueChanged(IDynamicEditorControl sender, Dictionary<string, object?> values)
  302. {
  303. //Logger.Send(LogType.Information, "", string.Format("DynamicEditorGrid.EditorValueChanged({0})", values.Keys.Count));
  304. var changededitors = new Dictionary<string, object?>();
  305. void ExtractChanged(Dictionary<string, object?>? columns)
  306. {
  307. if (columns != null)
  308. foreach (var (change, value) in columns)
  309. if (!changededitors.ContainsKey(change) && !change.Equals(sender.ColumnName))
  310. changededitors[change] = value;
  311. }
  312. var name = sender.ColumnName;
  313. var resetAll = false;
  314. if(values.Remove(name, out var value))
  315. {
  316. var changedcolumns = EditorGrid.OnEditorValueChanged?.Invoke(EditorGrid, name, value);
  317. resetAll = changedcolumns?.ContainsKey(name) ?? false;
  318. ExtractChanged(changedcolumns);
  319. }
  320. else
  321. {
  322. }
  323. foreach (var (key, otherValue) in values)
  324. {
  325. var changes = new Dictionary<string, object?>();
  326. if (resetAll)
  327. {
  328. var prop = DatabaseSchema.Property(EditorGrid.UnderlyingType, key);
  329. if(prop is not null)
  330. {
  331. var def = CoreUtils.GetDefault(prop.PropertyType);
  332. var resetChanges = EditorGrid.OnEditorValueChanged?.Invoke(EditorGrid, key, def);
  333. if(resetChanges is not null)
  334. {
  335. changes = resetChanges;
  336. }
  337. }
  338. else
  339. {
  340. }
  341. }
  342. var changedOtherColumns = EditorGrid.OnEditorValueChanged?.Invoke(EditorGrid, key, otherValue);
  343. if(changedOtherColumns is not null)
  344. {
  345. foreach (var (k, v) in changedOtherColumns)
  346. {
  347. changes[k] = v;
  348. }
  349. }
  350. ExtractChanged(changes);
  351. }
  352. var afterchanged = EditorGrid.OnAfterEditorValueChanged?.Invoke(EditorGrid, new AfterEditorValueChangedArgs(sender.ColumnName, changededitors));
  353. ExtractChanged(afterchanged);
  354. if (changededitors.Count != 0)
  355. LoadEditorValues(changededitors);
  356. EditorGrid.ReconfigureEditors();
  357. }
  358. private void LoadEditorValues(Dictionary<string, object?>? changededitors = null)
  359. {
  360. var columnnames = changededitors != null ? changededitors.Keys.ToArray() : EditorList.Select(x => x.ColumnName).ToArray();
  361. foreach (var columnname in columnnames)
  362. {
  363. if (!TryFindEditor(columnname, out var editor))
  364. continue;
  365. var bLoaded = editor.Loaded;
  366. editor.Loaded = false;
  367. if (changededitors != null && changededitors.ContainsKey(columnname))
  368. {
  369. editor.SetValue(columnname, changededitors[columnname]);
  370. }
  371. else
  372. {
  373. var curvalue = EditorGrid.GetPropertyValue(columnname);
  374. try
  375. {
  376. editor.SetValue(columnname, curvalue);
  377. }
  378. catch (Exception e)
  379. {
  380. MessageWindow.ShowError($"Unable to set editor value for {columnname} -> {curvalue}", e);
  381. }
  382. editor.Changed = false;
  383. }
  384. editor.Loaded = bLoaded;
  385. editor.OnEditorValueChanged += EditorValueChanged;
  386. }
  387. }
  388. public void Load(object item, Func<Type, CoreTable?>? PageDataHandler)
  389. {
  390. ConfigureEditors();
  391. LoadEditorValues();
  392. foreach (var editor in EditorList)
  393. {
  394. foreach(var (column, editorValue) in editor.GetValues())
  395. {
  396. var entityValue = EditorGrid.GetPropertyValue(column);
  397. if (!Equals(editorValue, entityValue))
  398. {
  399. bool bLoaded = editor.Loaded;
  400. editor.Loaded = false;
  401. editor.SetValue(column, entityValue);
  402. editor.Loaded = bLoaded;
  403. }
  404. }
  405. }
  406. EditorList.FirstOrDefault()?.SetFocus();
  407. Ready = true;
  408. }
  409. public void DoChanged()
  410. {
  411. OnChanged?.Invoke(this, EventArgs.Empty);
  412. }
  413. public Size MinimumSize() => new Size(800, GeneralHeight);
  414. public int Order() => PageOrder;
  415. }
  416. #endregion
  417. #region Loading + Editing Layout
  418. private decimal GetSequence(DynamicGridColumn column)
  419. {
  420. if (OnGetSequence != null)
  421. return OnGetSequence.Invoke(column);
  422. return 999;
  423. }
  424. private DynamicEditPage GetEditPage(string name)
  425. {
  426. var page = Pages.Where(x => x is DynamicEditPage page && page.Header == name).FirstOrDefault() as DynamicEditPage;
  427. if(page is null)
  428. {
  429. page = new DynamicEditPage(name)
  430. {
  431. // Setting this here because it's needed now to be able to create the layout.
  432. EditorGrid = this
  433. };
  434. if (name == "General")
  435. {
  436. page.PageOrder = -1;
  437. }
  438. else
  439. {
  440. page.PageOrder = 0;
  441. }
  442. AddPage(page);
  443. }
  444. return page;
  445. }
  446. public void SetLayoutType<T>() where T : DynamicEditorGridLayout
  447. {
  448. LayoutType = typeof(T);
  449. }
  450. public void SetLayoutType(Type t)
  451. {
  452. if (!t.IsSubclassOf(typeof(DynamicEditorGridLayout)))
  453. throw new Exception($"{t.Name} is not a {nameof(DynamicEditorGridLayout)}!");
  454. LayoutType = t;
  455. }
  456. private void InitialiseLayout()
  457. {
  458. Layout = (Activator.CreateInstance(LayoutType ?? typeof(DefaultDynamicEditorGridLayout)) as DynamicEditorGridLayout)!;
  459. Layout.OnSelectPage += Layout_SelectPage;
  460. Layout.TabStripVisible = _tabStripVisible;
  461. Content = Layout;
  462. }
  463. private void CreateLayout()
  464. {
  465. if(Layout is null)
  466. {
  467. InitialiseLayout();
  468. }
  469. foreach (var column in _columns.OrderBy(x => GetSequence(x)))
  470. {
  471. var iProp = DatabaseSchema.Property(UnderlyingType, column.ColumnName);
  472. var editor = OnGetEditor?.Invoke(column);
  473. if (editor != null && iProp?.ShouldShowEditor() != true)
  474. {
  475. editor.Visible = Visible.Hidden;
  476. editor.Editable = Editable.Hidden;
  477. }
  478. if(editor is not null)
  479. {
  480. foreach(var security in editor.Security)
  481. {
  482. if (!Security.IsAllowed(security.SecurityDescriptor))
  483. {
  484. editor.Editable = editor.Editable.Combine(security.Editable);
  485. if (editor.Editable == Editable.Hidden)
  486. {
  487. break;
  488. }
  489. }
  490. }
  491. }
  492. if(editor is not null)
  493. {
  494. OnGridCustomiseEditor?.Invoke(this, column, editor);
  495. }
  496. if (editor != null && editor.Editable.EditorVisible())
  497. {
  498. var page = string.IsNullOrWhiteSpace(editor.Page) ? iProp is StandardProperty ? "General" : "Custom Fields" : editor.Page;
  499. var editPage = GetEditPage(page);
  500. editPage.AddEditor(column.ColumnName, editor);
  501. }
  502. else if (iProp?.HasParentEditor() == true)
  503. {
  504. // Add the parent editor if it hasn't been added already.
  505. var parent = iProp.GetParentWithEditor();
  506. if(parent is not null)
  507. {
  508. var parentEditor = parent.Editor?.CloneEditor();
  509. if(parentEditor is not null)
  510. {
  511. OnGridCustomiseEditor?.Invoke(this, new DynamicGridColumn { ColumnName = parent.Name }, parentEditor);
  512. }
  513. if(parentEditor is not null && parentEditor.Editable.EditorVisible())
  514. {
  515. var page = string.IsNullOrWhiteSpace(parentEditor.Page)
  516. ? parent is StandardProperty
  517. ? "General"
  518. : "Custom Fields"
  519. : parentEditor.Page;
  520. var editPage = GetEditPage(page);
  521. if (!editPage.TryFindEditor(parent.Name, out var editorControl))
  522. {
  523. editPage.AddEditor(parent.Name, parentEditor);
  524. }
  525. }
  526. }
  527. }
  528. }
  529. OnEditorCreated?.Invoke(this, 0, 800);
  530. }
  531. #endregion
  532. #region Pages
  533. private void Layout_SelectPage(IDynamicEditorPage page)
  534. {
  535. if (!page.Ready)
  536. using (new WaitCursor())
  537. {
  538. OnLoadPage?.Invoke(page);
  539. }
  540. OnSelectPage?.Invoke(this, null);
  541. }
  542. public void UnloadPages(bool saved)
  543. {
  544. if(Pages is not null)
  545. foreach (var page in Pages)
  546. if (page.Ready)
  547. OnUnloadPage?.Invoke(page, saved);
  548. }
  549. private void LoadPages()
  550. {
  551. if (Pages != null && Layout is not null)
  552. using (new WaitCursor())
  553. {
  554. foreach (var page in Pages)
  555. {
  556. page.Ready = false;
  557. page.EditorGrid = this;
  558. }
  559. Layout.LoadPages(Pages);
  560. if (PreloadPages)
  561. {
  562. foreach(var page in Pages)
  563. {
  564. OnLoadPage?.Invoke(page);
  565. }
  566. }
  567. }
  568. }
  569. public void Load(DynamicEditorPages pages)
  570. {
  571. SetPages(pages);
  572. _columns = new DynamicGridColumns();
  573. OnCustomiseColumns?.Invoke(this, _columns);
  574. CreateLayout();
  575. Reload();
  576. }
  577. #endregion
  578. }