DynamicEditorGrid.xaml.cs 25 KB

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