DynamicEditorGrid.xaml.cs 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610
  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 RoslynPad.Editor;
  12. namespace InABox.DynamicGrid
  13. {
  14. public delegate void OnUpdateOtherEditorHandler(string columnname, object value);
  15. public delegate Dictionary<string, object?> EditorValueChangedHandler(object 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 Document? FindDocumentEvent(string FileName);
  23. public delegate Document? GetDocumentEvent(Guid id);
  24. public delegate object? GetPropertyValueHandler(object sender, string name);
  25. public delegate void SaveDocumentEvent(Document document);
  26. public delegate void SetPropertyValueHandler(object sender, string name, object value);
  27. public delegate object?[] GetItemsEvent();
  28. // Column Definitions as defined by calling model
  29. private DynamicGridColumns _columns = new();
  30. private Type? LayoutType;
  31. private DynamicEditorGridLayout? Layout;
  32. public bool TabStripVisible
  33. {
  34. get { return Layout != null ? Layout.TabStripVisible : false; }
  35. set
  36. {
  37. if (Layout != null)
  38. Layout.TabStripVisible = value;
  39. }
  40. }
  41. public DynamicEditorGrid()
  42. {
  43. InitializeComponent();
  44. Loaded += DynamicEditorGrid_Loaded;
  45. }
  46. public DynamicEditorPages Pages { get; private set; } = new();
  47. public bool PreloadPages { get; set; }
  48. public Type UnderlyingType { get; set; }
  49. public OnLoadPage? OnLoadPage { get; set; }
  50. public OnSelectPage? OnSelectPage { get; set; }
  51. public OnUnloadPage? OnUnloadPage { get; set; }
  52. public DynamicGridColumns Columns => _columns;
  53. public bool TryFindEditor(string columnname, [NotNullWhen(true)] out IDynamicEditorControl? editor)
  54. {
  55. foreach (var page in Pages)
  56. {
  57. if (page is DynamicEditPage editPage)
  58. {
  59. if (editPage.TryFindEditor(columnname, out editor))
  60. return true;
  61. }
  62. }
  63. editor = null;
  64. return false;
  65. }
  66. public IDynamicEditorControl? FindEditor(string columnname)
  67. {
  68. TryFindEditor(columnname, out var editor);
  69. return editor;
  70. }
  71. public virtual void ReconfigureEditors()
  72. {
  73. OnReconfigureEditors?.Invoke(this);
  74. }
  75. public object? GetPropertyValue(string columnname)
  76. {
  77. return OnGetPropertyValue?.Invoke(this, columnname);
  78. }
  79. public event EditorCreatedHandler? OnEditorCreated;
  80. public event OnCustomiseColumns? OnCustomiseColumns;
  81. public event OnGetEditor? OnGetEditor;
  82. public event OnGridCustomiseEditor? OnGridCustomiseEditor;
  83. public event OnGetEditorSequence? OnGetSequence;
  84. public event GetPropertyValueHandler? OnGetPropertyValue;
  85. public event SetPropertyValueHandler? OnSetPropertyValue;
  86. public event EditorValueChangedHandler? OnEditorValueChanged;
  87. public event OnAfterEditorValueChanged? OnAfterEditorValueChanged;
  88. public event OnReconfigureEditors? OnReconfigureEditors;
  89. public event OnDefineFilter? OnDefineFilter;
  90. public event OnDefineLookup? OnDefineLookups;
  91. public event GetDocumentEvent? OnGetDocument;
  92. public event FindDocumentEvent? OnFindDocument;
  93. public event SaveDocumentEvent? OnSaveDocument;
  94. public event GetItemsEvent? GetItems;
  95. private void DynamicEditorGrid_Loaded(object sender, RoutedEventArgs e)
  96. {
  97. //Reload();
  98. }
  99. public void Reload()
  100. {
  101. LoadPages();
  102. ReconfigureEditors();
  103. }
  104. #region Host Implementation
  105. IEnumerable<DynamicGridColumn> IDynamicEditorHost.Columns => Columns;
  106. public void LoadColumns(string column, Dictionary<string, string> columns)
  107. {
  108. columns.Clear();
  109. var comps = column.Split('.').ToList();
  110. comps.RemoveAt(comps.Count - 1);
  111. var prefix = string.Format("{0}.", string.Join(".", comps));
  112. var cols = Columns.Where(x => !x.ColumnName.Equals(column) && x.ColumnName.StartsWith(prefix));
  113. foreach (var col in cols)
  114. columns[col.ColumnName.Replace(prefix, "")] = col.ColumnName;
  115. }
  116. public IFilter? DefineFilter(Type type) => OnDefineFilter?.Invoke(type);
  117. public void LoadLookups(ILookupEditorControl editor)
  118. {
  119. OnDefineLookups?.Invoke(editor);
  120. }
  121. public Document? FindDocument(string filename) => OnFindDocument?.Invoke(filename);
  122. public Document? GetDocument(Guid id) => OnGetDocument?.Invoke(id);
  123. public void SaveDocument(Document document) => OnSaveDocument?.Invoke(document);
  124. object?[] IDynamicEditorHost.GetItems() => GetItems?.Invoke() ?? Array.Empty<object?>();
  125. public BaseEditor? GetEditor(DynamicGridColumn column) => OnGetEditor?.Invoke(column);
  126. #endregion
  127. #region Edit Page
  128. public class DynamicEditPage : ContentControl, IDynamicEditorPage
  129. {
  130. private Grid Grid;
  131. public DynamicEditorGrid EditorGrid { get; set; } = null!; // Set by DynamicEditorGrid
  132. public bool Ready { get; set; }
  133. private List<BaseDynamicEditorControl> Editors { get; set; }
  134. public PageType PageType => PageType.Editor;
  135. public int PageOrder { get; set; }
  136. public string Header { get; set; }
  137. private double GeneralHeight = 30;
  138. public DynamicEditPage(string header)
  139. {
  140. Header = header;
  141. Editors = new List<BaseDynamicEditorControl>();
  142. InitialiseContent();
  143. }
  144. public void AddEditor(string columnName, BaseEditor editor)
  145. {
  146. BaseDynamicEditorControl? element = DynamicEditorControlFactory.CreateControl(editor, EditorGrid);
  147. if (element != null)
  148. {
  149. element.EditorDefinition = editor;
  150. element.IsEnabled = editor.Editable == Editable.Enabled;
  151. if (!string.IsNullOrWhiteSpace(editor.ToolTip))
  152. {
  153. element.ToolTip = new ToolTip() { Content = editor.ToolTip };
  154. }
  155. var label = new Label();
  156. label.Content = CoreUtils.Neatify(editor.Caption); // 2
  157. label.Margin = new Thickness(0F, 0F, 0F, 0F);
  158. label.HorizontalAlignment = HorizontalAlignment.Stretch;
  159. label.VerticalAlignment = VerticalAlignment.Stretch;
  160. label.HorizontalContentAlignment = HorizontalAlignment.Left;
  161. label.VerticalContentAlignment = VerticalAlignment.Center;
  162. label.SetValue(Grid.RowProperty, Grid.RowDefinitions.Count);
  163. label.SetValue(Grid.ColumnProperty, 0);
  164. label.Visibility = string.IsNullOrWhiteSpace(editor.Caption) ? Visibility.Collapsed : Visibility.Visible;
  165. Grid.Children.Add(label);
  166. element.ColumnName = columnName;
  167. element.Color = editor is UniqueCodeEditor ? Color.FromArgb(0xFF, 0xF6, 0xC9, 0xE8) : Colors.LightYellow;
  168. Editors.Add(element);
  169. element.Margin = new Thickness(5F, 2.5F, 5F, 2.5F);
  170. double iHeight = element.DesiredHeight();
  171. if (iHeight == int.MaxValue)
  172. {
  173. Grid.RowDefinitions.Add(new RowDefinition { Height = new GridLength(1, GridUnitType.Star) });
  174. GeneralHeight += element.MinHeight + 5.0F;
  175. }
  176. else
  177. {
  178. Grid.RowDefinitions.Add(new RowDefinition { Height = new GridLength(iHeight + 5.0F) });
  179. GeneralHeight += iHeight + 5.0F;
  180. }
  181. double iWidth = element.DesiredWidth();
  182. if (iWidth == int.MaxValue)
  183. {
  184. element.HorizontalAlignment = HorizontalAlignment.Stretch;
  185. }
  186. else
  187. {
  188. element.HorizontalAlignment = HorizontalAlignment.Left;
  189. element.Width = iWidth;
  190. }
  191. element.SetValue(Grid.RowProperty, Grid.RowDefinitions.Count - 1);
  192. element.SetValue(Grid.ColumnProperty, 1);
  193. Grid.Children.Add(element);
  194. }
  195. }
  196. [MemberNotNull(nameof(Grid))]
  197. private void InitialiseContent()
  198. {
  199. Grid = new Grid
  200. {
  201. HorizontalAlignment = HorizontalAlignment.Stretch,
  202. VerticalAlignment = VerticalAlignment.Stretch,
  203. Margin = new Thickness(0, 2.5, 0, 2.5)
  204. };
  205. Grid.ColumnDefinitions.Add(new ColumnDefinition { Width = new GridLength(1, GridUnitType.Auto) });
  206. Grid.ColumnDefinitions.Add(new ColumnDefinition { Width = new GridLength(1, GridUnitType.Star) });
  207. var scroll = new ScrollViewer
  208. {
  209. HorizontalAlignment = HorizontalAlignment.Stretch,
  210. VerticalAlignment = VerticalAlignment.Stretch,
  211. VerticalScrollBarVisibility = ScrollBarVisibility.Auto,
  212. Padding = new Thickness(2),
  213. Content = Grid
  214. };
  215. var border = new Border
  216. {
  217. BorderBrush = new SolidColorBrush(Colors.Gray),
  218. Background = new SolidColorBrush(Colors.White),
  219. BorderThickness = new Thickness(0.75),
  220. Child = scroll
  221. };
  222. Content = border;
  223. }
  224. public void AfterSave(object item)
  225. {
  226. }
  227. public void BeforeSave(object item)
  228. {
  229. }
  230. public string Caption() => Header;
  231. public bool TryFindEditor(string columnname, [NotNullWhen(true)] out IDynamicEditorControl? editor)
  232. {
  233. editor = Editors.FirstOrDefault(x => x.ColumnName.Equals(columnname));
  234. editor ??= Editors.FirstOrDefault(x => columnname.StartsWith(x.ColumnName + '.'));
  235. return editor is not null;
  236. }
  237. public IEnumerable<BaseDynamicEditorControl> FindEditors(DynamicGridColumn column)
  238. {
  239. return Editors.Where(x => string.Equals(x.ColumnName, column.ColumnName));
  240. }
  241. #region Configure Editors
  242. private void Lookup_OnUpdateOtherEditor(string columnname, object value)
  243. {
  244. var editor = Editors.FirstOrDefault(x => x.ColumnName.Equals(columnname));
  245. if (editor != null)
  246. CoreUtils.SetPropertyValue(editor, "Value", value);
  247. }
  248. private void ConfigureEditors()
  249. {
  250. foreach (var Editor in Editors)
  251. {
  252. var editor = Editor.EditorDefinition;
  253. var column = Editor.ColumnName;
  254. Editor.Configure();
  255. if (!Editors.Any(x => x.ColumnName.Equals(Editor.ColumnName)))
  256. Editors.Add(Editor);
  257. Editor.Loaded = true;
  258. }
  259. }
  260. #endregion
  261. private void EditorValueChanged(IDynamicEditorControl sender, Dictionary<string, object> values)
  262. {
  263. //Logger.Send(LogType.Information, "", string.Format("DynamicEditorGrid.EditorValueChanged({0})", values.Keys.Count));
  264. var changededitors = new Dictionary<string, object?>();
  265. void ExtractChanged(Dictionary<String, object?>? columns)
  266. {
  267. if (columns != null)
  268. foreach (var (change, value) in columns)
  269. if (!changededitors.ContainsKey(change) && !change.Equals(sender.ColumnName))
  270. changededitors[change] = value;
  271. }
  272. foreach (var key in values.Keys)
  273. {
  274. var changedcolumns = EditorGrid.OnEditorValueChanged?.Invoke(EditorGrid, key, values[key]);
  275. ExtractChanged(changedcolumns);
  276. }
  277. var afterchanged = EditorGrid.OnAfterEditorValueChanged?.Invoke(EditorGrid, sender.ColumnName);
  278. ExtractChanged(afterchanged);
  279. if (changededitors.Any())
  280. LoadEditorValues(changededitors);
  281. EditorGrid.ReconfigureEditors();
  282. }
  283. private void LoadEditorValues(Dictionary<string, object?>? changededitors = null)
  284. {
  285. var columnnames = changededitors != null ? changededitors.Keys.ToArray() : Editors.Select(x => x.ColumnName).ToArray();
  286. foreach (var columnname in columnnames)
  287. {
  288. if (!TryFindEditor(columnname, out var editor))
  289. continue;
  290. var bLoaded = editor.Loaded;
  291. editor.Loaded = false;
  292. if (changededitors != null && changededitors.ContainsKey(columnname))
  293. {
  294. editor.SetValue(columnname, changededitors[columnname]);
  295. }
  296. else
  297. {
  298. var curvalue = EditorGrid.GetPropertyValue(columnname);
  299. try
  300. {
  301. editor.SetValue(columnname, curvalue);
  302. }
  303. catch (Exception e)
  304. {
  305. MessageBox.Show($"Unable to set editor value for {columnname} -> {curvalue}: {CoreUtils.FormatException(e)}");
  306. }
  307. editor.Changed = false;
  308. }
  309. editor.Loaded = bLoaded;
  310. editor.OnEditorValueChanged += EditorValueChanged;
  311. }
  312. }
  313. public void Load(object item, Func<Type, CoreTable>? PageDataHandler)
  314. {
  315. ConfigureEditors();
  316. LoadEditorValues();
  317. foreach (var editor in Editors)
  318. {
  319. foreach(var (column, editorValue) in editor.GetValues())
  320. {
  321. var entityValue = EditorGrid.GetPropertyValue(column);
  322. if (!Equals(editorValue, entityValue))
  323. {
  324. bool bLoaded = editor.Loaded;
  325. editor.Loaded = false;
  326. editor.SetValue(column, entityValue);
  327. editor.Loaded = bLoaded;
  328. }
  329. }
  330. }
  331. Editors.FirstOrDefault()?.SetFocus();
  332. Ready = true;
  333. }
  334. public Size MinimumSize() => new Size(800, GeneralHeight);
  335. public int Order() => PageOrder;
  336. }
  337. #endregion
  338. #region Loading + Editing Layout
  339. private decimal GetSequence(DynamicGridColumn column)
  340. {
  341. if (OnGetSequence != null)
  342. return OnGetSequence.Invoke(column);
  343. return 999;
  344. }
  345. private DynamicEditPage GetEditPage(string name)
  346. {
  347. var page = Pages.Where(x => x is DynamicEditPage page && page.Header == name).FirstOrDefault() as DynamicEditPage;
  348. if(page is null)
  349. {
  350. page = new DynamicEditPage(name)
  351. {
  352. // Setting this here because it's needed now to be able to create the layout.
  353. EditorGrid = this
  354. };
  355. if (name == "General")
  356. {
  357. page.PageOrder = -1;
  358. }
  359. else
  360. {
  361. page.PageOrder = 0;
  362. }
  363. Pages.Add(page);
  364. }
  365. return page;
  366. }
  367. public void SetLayoutType<T>() where T : DynamicEditorGridLayout
  368. {
  369. LayoutType = typeof(T);
  370. }
  371. private void InitialiseLayout()
  372. {
  373. Layout = (Activator.CreateInstance(LayoutType ?? typeof(DefaultDynamicEditorGridLayout)) as DynamicEditorGridLayout)!;
  374. Layout.OnSelectPage += Layout_SelectPage;
  375. Content = Layout;
  376. }
  377. private void CreateLayout()
  378. {
  379. if(Layout is null)
  380. {
  381. InitialiseLayout();
  382. }
  383. foreach (var column in _columns.OrderBy(x => GetSequence(x)))
  384. {
  385. var iProp = DatabaseSchema.Property(UnderlyingType, column.ColumnName);
  386. var editor = OnGetEditor?.Invoke(column);
  387. if (editor != null && iProp?.ShouldShowEditor() != true)
  388. {
  389. editor.Visible = Visible.Hidden;
  390. editor.Editable = Editable.Hidden;
  391. }
  392. if(editor is not null)
  393. {
  394. OnGridCustomiseEditor?.Invoke(this, column, editor);
  395. }
  396. if (editor != null && editor.Editable != Editable.Hidden)
  397. {
  398. var page = string.IsNullOrWhiteSpace(editor.Page) ? iProp is StandardProperty ? "General" : "Custom Fields" : editor.Page;
  399. var editPage = GetEditPage(page);
  400. editPage.AddEditor(column.ColumnName, editor);
  401. }
  402. else if (iProp?.HasParentEditor() == true)
  403. {
  404. var parent = iProp.GetParentWithEditor();
  405. if(parent is not null)
  406. {
  407. var parentEditor = parent.Editor;
  408. if(parentEditor is not null)
  409. {
  410. OnGridCustomiseEditor?.Invoke(this, new DynamicGridColumn { ColumnName = parent.Name }, parentEditor);
  411. }
  412. if(parentEditor is not null && parentEditor.Editable != Editable.Hidden)
  413. {
  414. var page = string.IsNullOrWhiteSpace(parentEditor.Page)
  415. ? parent is StandardProperty
  416. ? "General"
  417. : "Custom Fields"
  418. : parentEditor.Page;
  419. var editPage = GetEditPage(page);
  420. if (!editPage.TryFindEditor(parent.Name, out var editorControl))
  421. {
  422. editPage.AddEditor(parent.Name, parentEditor);
  423. }
  424. }
  425. }
  426. }
  427. }
  428. OnEditorCreated?.Invoke(this, 0, 800);
  429. }
  430. #endregion
  431. #region Pages
  432. private void Layout_SelectPage(IDynamicEditorPage page)
  433. {
  434. if (!page.Ready)
  435. using (new WaitCursor())
  436. {
  437. OnLoadPage?.Invoke(page);
  438. }
  439. OnSelectPage?.Invoke(this, null);
  440. }
  441. public void UnloadPages(bool saved)
  442. {
  443. if(Pages is not null)
  444. foreach (var page in Pages)
  445. if (page.Ready)
  446. OnUnloadPage?.Invoke(page, saved);
  447. }
  448. private void LoadPages()
  449. {
  450. if (Pages != null && Layout is not null)
  451. using (new WaitCursor())
  452. {
  453. foreach (var page in Pages)
  454. {
  455. page.Ready = false;
  456. page.EditorGrid = this;
  457. }
  458. Layout.LoadPages(Pages);
  459. if (PreloadPages)
  460. {
  461. foreach(var page in Pages)
  462. {
  463. OnLoadPage?.Invoke(page);
  464. }
  465. }
  466. }
  467. }
  468. public void Load(DynamicEditorPages pages)
  469. {
  470. Pages = pages;
  471. _columns = new DynamicGridColumns();
  472. OnCustomiseColumns?.Invoke(this, _columns);
  473. CreateLayout();
  474. Reload();
  475. }
  476. #endregion
  477. }
  478. }