EmbeddedDynamicEditorForm.xaml.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544
  1. using InABox.Core;
  2. using InABox.Wpf;
  3. using System;
  4. using System.Collections.Generic;
  5. using System.ComponentModel;
  6. using System.Diagnostics.CodeAnalysis;
  7. using System.Linq;
  8. using System.Windows;
  9. using System.Windows.Controls;
  10. using System.Windows.Media;
  11. using System.Windows.Media.Imaging;
  12. namespace InABox.DynamicGrid
  13. {
  14. /// <summary>
  15. /// Interaction logic for EmbeddedDynamicEditorForm.xaml
  16. /// </summary>
  17. public partial class EmbeddedDynamicEditorForm : UserControl, IDynamicEditorForm
  18. {
  19. public event OnBeforeLoad? OnBeforeLoad;
  20. public void BeforeLoad() => OnBeforeLoad?.Invoke(this);
  21. public event OnAfterLoad? OnAfterLoad;
  22. public void AfterLoad() => OnAfterLoad?.Invoke(this);
  23. public event IDynamicEditorForm.OnReloadEventHandler? OnReload;
  24. public event EventHandler OnChanged;
  25. private bool bChanged = false;
  26. public void DoChanged()
  27. {
  28. bChanged = true;
  29. //OKButton.IsEnabled = true;
  30. //CancelButton.IsEnabled = true;
  31. OnChanged?.Invoke(this, EventArgs.Empty);
  32. UpdateButtonEnabled();
  33. }
  34. public delegate void OKEvent();
  35. public delegate void CancelEvent();
  36. public DynamicEditorPages Pages { get; private set; } = [];
  37. private BaseObject[] _items;
  38. public BaseObject[] Items
  39. {
  40. get => _items;
  41. set
  42. {
  43. _items = value;
  44. DynamicEditorFormModel.Slug = Items?.FirstOrDefault()?.GetType().EntityName().Split('.').Last() ?? "";
  45. Editor.Load(Pages);
  46. bChanged = _items.Any(x => x is not Entity e || e.ID == Guid.Empty);
  47. UpdateButtonEnabled();
  48. foreach (var page in Pages)
  49. page.OnChanged += (sender, args) => DoChanged();
  50. }
  51. }
  52. public bool ReadOnly
  53. {
  54. get => Editor.ReadOnly;
  55. set
  56. {
  57. Editor.ReadOnly = value;
  58. UpdateButtonEnabled();
  59. }
  60. }
  61. private bool _disableIfUnchanged = false;
  62. public bool DisableOKIfUnchanged
  63. {
  64. get => _disableIfUnchanged;
  65. set
  66. {
  67. _disableIfUnchanged = value;
  68. UpdateButtonEnabled();
  69. }
  70. }
  71. private bool _hideButtons = false;
  72. public bool HideButtons
  73. {
  74. get => _hideButtons;
  75. set
  76. {
  77. _hideButtons = value;
  78. OKButton.Visibility = value ? Visibility.Collapsed : Visibility.Visible;
  79. CancelButton.Visibility = value ? Visibility.Collapsed : Visibility.Visible;
  80. }
  81. }
  82. private bool _highlightButtons = false;
  83. public bool HighlightButtons
  84. {
  85. get => _highlightButtons;
  86. set
  87. {
  88. _highlightButtons = value;
  89. UpdateButtonHighlight(OKButton, Colors.DarkGreen, Colors.LimeGreen, Colors.White);
  90. UpdateButtonHighlight(CancelButton, Colors.Firebrick, Colors.Red, Colors.White);
  91. }
  92. }
  93. public double ContentWidth => Editor.TotalWidth;
  94. public double ContentHeight => Editor.TotalHeight;
  95. private void UpdateButtonEnabled()
  96. {
  97. OKButton.IsEnabled = !ReadOnly && (!DisableOKIfUnchanged || bChanged);
  98. CancelButton.IsEnabled = !ReadOnly;
  99. }
  100. private void UpdateButtonHighlight(Button button, Color border, Color background, Color foreground)
  101. {
  102. button.BorderBrush = _highlightButtons
  103. ? new SolidColorBrush(border)
  104. : new SolidColorBrush(Colors.Gray);
  105. button.Background = _highlightButtons
  106. ? new SolidColorBrush(background)
  107. : new SolidColorBrush(Colors.Gainsboro);
  108. button.Foreground = _highlightButtons
  109. ? new SolidColorBrush(foreground)
  110. : new SolidColorBrush(Colors.Black);
  111. button.FontWeight = _highlightButtons
  112. ? FontWeights.Bold
  113. : FontWeights.Normal;
  114. }
  115. public static readonly DependencyProperty ButtonsVisibleProperty =
  116. DependencyProperty.Register(
  117. nameof(ButtonsVisible),
  118. typeof(bool),
  119. typeof(EmbeddedDynamicEditorForm),
  120. new UIPropertyMetadata(true)
  121. );
  122. public bool ButtonsVisible
  123. {
  124. get => (bool)GetValue(ButtonsVisibleProperty);
  125. set
  126. {
  127. SetValue(ButtonsVisibleProperty, value);
  128. UpdateButtonsRowVisibility();
  129. }
  130. }
  131. private void UpdateButtonsRowVisibility()
  132. {
  133. ButtonRow.Height = ButtonsVisible
  134. ? new GridLength(1, GridUnitType.Auto)
  135. : new GridLength(0, GridUnitType.Pixel);
  136. }
  137. public static readonly DependencyProperty TabsVisibleProperty =
  138. DependencyProperty.Register(
  139. nameof(TabsVisible),
  140. typeof(bool),
  141. typeof(EmbeddedDynamicEditorForm),
  142. new UIPropertyMetadata(true)
  143. );
  144. public bool TabsVisible
  145. {
  146. get => (bool)GetValue(TabsVisibleProperty);
  147. set
  148. {
  149. SetValue(TabsVisibleProperty, value);
  150. UpdateTabsVisibility();
  151. }
  152. }
  153. private void UpdateTabsVisibility()
  154. {
  155. Editor.TabStripVisible = TabsVisible;
  156. }
  157. #region Events
  158. public event OnValidateData? OnValidateData;
  159. public OnCustomiseColumns? OnCustomiseColumns { get; set; }
  160. public OnDefineLookupFilter? OnDefineFilter { get; set; }
  161. public OnDefineLookup? OnDefineLookups { get; set; }
  162. public DefineEditorEventHandler? OnDefineEditor { get; set; }
  163. public event OnFormCustomiseEditor? OnFormCustomiseEditor;
  164. public OnReconfigureEditors? OnReconfigureEditors { get; set; }
  165. public event EditorValueChangedHandler? OnEditorValueChanged;
  166. public event OnAfterEditorValueChanged? OnAfterEditorValueChanged;
  167. public event OnSelectPage? OnSelectPage;
  168. public DynamicGridSaveEvent? OnSaveItem { get; set; }
  169. public DynamicEditorGrid.EditorCreatedHandler? OnEditorCreated;
  170. public event OKEvent? OnOK;
  171. public event CancelEvent? OnCancel;
  172. #endregion
  173. public EmbeddedDynamicEditorForm()
  174. {
  175. InitializeComponent();
  176. ReadOnly = false;
  177. Editor.Form = this;
  178. }
  179. public override void OnApplyTemplate()
  180. {
  181. base.OnApplyTemplate();
  182. UpdateButtonsRowVisibility();
  183. UpdateTabsVisibility();
  184. }
  185. public EmbeddedDynamicEditorForm(Type type, DynamicEditorPages? pages = null, DynamicEditorButtons? buttons = null,
  186. Func<Type, CoreTable>? PageDataHandler = null, bool PreloadPages = false): this()
  187. {
  188. Setup(type, pages, buttons, PageDataHandler, PreloadPages);
  189. }
  190. private IFilter? Editor_OnDefineFilter(Type type, string column)
  191. {
  192. return OnDefineFilter?.Invoke(type, column);
  193. }
  194. private void ClearEvents()
  195. {
  196. OnEditorValueChanged = null;
  197. OnFormCustomiseEditor = null;
  198. OnAfterEditorValueChanged = null;
  199. OnSelectPage = null;
  200. OnValidateData = null;
  201. }
  202. public void Setup(Type type, DynamicEditorPages? pages = null, DynamicEditorButtons? buttons = null,
  203. Func<Type, CoreTable?>? PageDataHandler = null, bool PreloadPages = false)
  204. {
  205. ClearEvents();
  206. Editor.UnderlyingType = type;
  207. Editor.OnLoadPage = page => { page.Load(Items.First(), PageDataHandler); };
  208. Editor.PreloadPages = PreloadPages;
  209. Pages = pages ?? new DynamicEditorPages();
  210. Buttons.Children.Clear();
  211. if (buttons != null)
  212. foreach (var button in buttons)
  213. {
  214. var btn = new Button();
  215. UpdateButton(btn, button.Image, button.Name);
  216. btn.Tag = button;
  217. btn.Margin = new Thickness(0, 0, 5, 0);
  218. btn.Padding = new Thickness(5, 0, 5, 0);
  219. btn.Click += Btn_Click;
  220. Buttons.Children.Add(btn);
  221. button.Button = btn;
  222. button.Form = this;
  223. }
  224. }
  225. public void UnloadEditorPages(bool saved)
  226. {
  227. Editor.UnloadPages(saved);
  228. }
  229. protected void UpdateButton(Button button, BitmapImage? image, string text)
  230. {
  231. var stackPnl = new StackPanel();
  232. stackPnl.Orientation = Orientation.Horizontal;
  233. //stackPnl.Margin = new Thickness(2);
  234. if (image != null)
  235. {
  236. var img = new Image();
  237. img.Source = image;
  238. img.Margin = new Thickness(2);
  239. stackPnl.Children.Add(img);
  240. }
  241. if (!string.IsNullOrEmpty(text))
  242. {
  243. var lbl = new Label();
  244. lbl.Content = text;
  245. lbl.VerticalAlignment = VerticalAlignment.Stretch;
  246. lbl.VerticalContentAlignment = VerticalAlignment.Center;
  247. lbl.Margin = new Thickness(2, 0, 5, 0);
  248. stackPnl.Children.Add(lbl);
  249. }
  250. button.Content = stackPnl;
  251. }
  252. private Dictionary<string, object?> EditorValueChanged(IDynamicEditorForm sender, string name, object value)
  253. {
  254. if (OnEditorValueChanged != null)
  255. return OnEditorValueChanged(sender, name, value);
  256. return DynamicGridUtils.UpdateEditorValue(_items, name, value);
  257. }
  258. private void Editor_OnEditorCreated(object sender, double height, double width)
  259. {
  260. OnEditorCreated?.Invoke(sender, height, width);
  261. Editor.VerticalAlignment = VerticalAlignment.Stretch;
  262. Editor.HorizontalAlignment = HorizontalAlignment.Stretch;
  263. }
  264. private void Editor_OnCustomiseColumns(object sender, DynamicGridColumns columns)
  265. {
  266. columns.Clear();
  267. if (_items != null && _items.Any())
  268. columns.ExtractColumns(_items.First().GetType());
  269. OnCustomiseColumns?.Invoke(this, columns);
  270. }
  271. private void Btn_Click(object sender, RoutedEventArgs e)
  272. {
  273. var button = (Button)sender;
  274. var deb = (DynamicEditorButton)button.Tag;
  275. deb.Click();
  276. }
  277. private void OKButton_Click(object sender, RoutedEventArgs e)
  278. {
  279. var errors = OnValidateData?.Invoke(this, Items);
  280. if (errors != null && errors.Any())
  281. {
  282. MessageWindow.ShowMessage(
  283. string.Format("The following errors have been found with your data!\nPlease correct them and try again.\n\n- {0}",
  284. string.Join("\n- ", errors)),
  285. "Validation Error");
  286. return;
  287. }
  288. OnOK?.Invoke();
  289. //OKButton.IsEnabled = false;
  290. //CancelButton.IsEnabled = false;
  291. // Don't Commit the changes here, because we want to refer back to thos changes when we save the item
  292. // to trigger specific processes in the database
  293. //Close();
  294. }
  295. private void CancelButton_Click(object sender, RoutedEventArgs e)
  296. {
  297. // However, if we cancel the edits, then we can safely revert the items back to their original (loaded) state
  298. foreach (var item in _items)
  299. item.CancelChanges();
  300. foreach(var page in Editor.Pages)
  301. {
  302. page.Cancel();
  303. }
  304. OnCancel?.Invoke();
  305. //OKButton.IsEnabled = false;
  306. //CancelButton.IsEnabled = false;
  307. //Close();
  308. }
  309. public void SaveItem(CancelEventArgs e)
  310. {
  311. OnSaveItem?.Invoke(this, e);
  312. }
  313. public bool TryFindEditor(string columnname, [NotNullWhen(true)] out IDynamicEditorControl? editor)
  314. {
  315. return Editor.TryFindEditor(columnname, out editor);
  316. }
  317. public object? GetEditorValue(string columnName) => this.FindEditor(columnName).GetValue(columnName);
  318. public void SetEditorValue(string columnName, object? value) => this.FindEditor(columnName).SetValue(columnName, value);
  319. public void SetLayoutType<T>() where T : DynamicEditorGridLayout => Editor.SetLayoutType<T>();
  320. public void SetLayoutType(Type t) => Editor.SetLayoutType(t);
  321. public void SetLayout(DynamicEditorGridLayout layout) => Editor.SetLayout(layout);
  322. private void Editor_OnSelectPage(DynamicEditorGrid sender, BaseObject[] items)
  323. {
  324. OnSelectPage?.Invoke(sender, items);
  325. }
  326. private void Editor_OnUnloadPage(IDynamicEditorPage page, bool saved)
  327. {
  328. if (!saved)
  329. page.BeforeSave(Items.First());
  330. else
  331. page.AfterSave(Items.First());
  332. }
  333. private Dictionary<string, object?>? Editor_OnAfterEditorValueChanged(DynamicEditorGrid sender, AfterEditorValueChangedArgs args)
  334. {
  335. if(args.ChangedValues.Count > 0)
  336. {
  337. DoChanged();
  338. }
  339. return OnAfterEditorValueChanged?.Invoke(sender, args);
  340. }
  341. private void Editor_OnReconfigureEditors(DynamicEditorGrid sender)
  342. {
  343. OnReconfigureEditors?.Invoke(sender);
  344. }
  345. private void Editor_OnGridCustomiseEditor(DynamicEditorGrid sender, DynamicGridColumn column, BaseEditor editor)
  346. {
  347. OnFormCustomiseEditor?.Invoke(this, Items, column, editor);
  348. }
  349. private decimal Editor_OnGetSequence(DynamicGridColumn column)
  350. {
  351. if (_items.Any())
  352. return CoreUtils.GetPropertySequence(_items.First().GetType(), column.ColumnName);
  353. return 0.0M;
  354. }
  355. private void Editor_OnDefineLookups(ILookupEditorControl editor)
  356. {
  357. OnDefineLookups?.Invoke(editor);
  358. }
  359. private BaseObject[] Editor_GetItems()
  360. {
  361. return _items;
  362. }
  363. private BaseEditor Editor_OnGetEditor(DynamicGridColumn column)
  364. {
  365. if (_items != null && _items.Any())
  366. {
  367. var property = DatabaseSchema.Property(Editor.UnderlyingType, column.ColumnName);
  368. if (property == null) return new NullEditor();
  369. if (property.Editor is NullEditor)
  370. return property.Editor.CloneEditor();
  371. BaseEditor editor;
  372. if (property is CustomProperty)
  373. {
  374. editor = property.Editor.CloneEditor();
  375. }
  376. else
  377. {
  378. editor = OnDefineEditor?.Invoke(_items[0], column) ?? column.Editor.CloneEditor();
  379. var propEditor = property.Editor;
  380. editor.Page = propEditor.Page;
  381. editor.Caption = propEditor.Caption;
  382. }
  383. //if (ReadOnly && editor.Editable.IsEditable())
  384. // editor.Editable = Editable.Disabled;
  385. return editor;
  386. }
  387. return new NullEditor();
  388. }
  389. private object? Editor_OnGetPropertyValue(object sender, string column)
  390. {
  391. if (!_items.Any())
  392. return null;
  393. object? result;
  394. try
  395. {
  396. result = CoreUtils.GetPropertyValue(_items.First(), column);
  397. }
  398. catch
  399. {
  400. result = _items.First().UserProperties.ContainsKey(column) ? _items.First().UserProperties[column] : null;
  401. }
  402. if (result == null)
  403. return null;
  404. foreach (var _item in _items)
  405. {
  406. object? curvalue;
  407. try
  408. {
  409. curvalue = CoreUtils.GetPropertyValue(_item, column);
  410. }
  411. catch
  412. {
  413. curvalue = _item.UserProperties.ContainsKey(column) ? _item.UserProperties[column] : null;
  414. }
  415. if (curvalue == null)
  416. return null;
  417. if (!curvalue.Equals(result))
  418. return null;
  419. }
  420. return result;
  421. }
  422. private void Editor_OnSetPropertyValue(object sender, string column, object value)
  423. {
  424. foreach (var _item in _items)
  425. if (_item.UserProperties.ContainsKey(column))
  426. _item.UserProperties[column] = value;
  427. else
  428. CoreUtils.SetPropertyValue(_item, column, value);
  429. }
  430. //public void EditLayout() => Editor.EditLayout();
  431. //public void ResetLayout() => Editor.ResetLayout();
  432. public void AddButton(Button button)
  433. {
  434. Buttons.Children.Add(button);
  435. }
  436. private Dictionary<string, object?> Editor_OnEditorValueChanged(DynamicEditorGrid sender, string name, object value)
  437. {
  438. return EditorValueChanged(this, name, value);
  439. }
  440. private void Editor_OnReload(DynamicEditorGrid sender)
  441. {
  442. OnReload?.Invoke(this);
  443. }
  444. }
  445. }