1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396 |
- using System;
- using System.Collections.Generic;
- using System.ComponentModel;
- using System.Data;
- using System.Diagnostics;
- using System.Globalization;
- using System.Linq;
- using System.Linq.Expressions;
- using System.Threading.Tasks;
- using System.Windows;
- using System.Windows.Controls;
- using System.Windows.Data;
- using System.Windows.Input;
- using System.Windows.Media;
- using System.Windows.Media.Animation;
- using System.Windows.Media.Imaging;
- using InABox.Clients;
- using InABox.Core;
- using InABox.Wpf;
- using InABox.WPF;
- using Syncfusion.Data;
- using Syncfusion.UI.Xaml.Grid;
- using Syncfusion.UI.Xaml.Grid.Helpers;
- using static InABox.DynamicGrid.IDynamicGrid;
- using Color = System.Drawing.Color;
- using Columns = InABox.Core.Columns;
- using Image = System.Windows.Controls.Image;
- using RowColumnIndex = Syncfusion.UI.Xaml.ScrollAxis.RowColumnIndex;
- using SolidColorBrush = System.Windows.Media.SolidColorBrush;
- using String = System.String;
- using VerticalAlignment = System.Windows.VerticalAlignment;
- using VirtualizingCellsControl = Syncfusion.UI.Xaml.Grid.VirtualizingCellsControl;
- using System.Threading;
- using System.Diagnostics.CodeAnalysis;
- namespace InABox.DynamicGrid;
- public abstract class DynamicGrid<T> : BaseDynamicGrid, IDynamicGridUIComponentParent<T>, IDynamicGrid<T>,
- IImportDynamicGrid, IExportDynamicGrid, IDuplicateDynamicGrid, IHelpDynamicGrid
- where T : BaseObject, new()
- {
- #region Events
- public ValidateEvent<T>? OnValidate { get; set; }
- public event OnDefineFilter? OnDefineFilter;
- public event OnCreateItem? OnCreateItem;
- public event OnAfterCreateItem? OnAfterCreateItem;
- public event EditorValueChangedHandler? OnEditorValueChanged;
- public event OnCustomiseEditor<T>? OnCustomiseEditor;
- public event EntitySaveEvent? OnBeforeSave;
- public event EntitySaveEvent? OnAfterSave;
- public delegate void EditorLoaded(IDynamicEditorForm editor, T[] items);
- public event EditorLoaded? OnEditorLoaded;
- public event OnLoadEditorButtons<T>? OnLoadEditorButtons;
- #endregion
- protected virtual string HelpSlug()
- {
- return typeof(T).Name.Split('.').Last().SplitCamelCase().Replace(" ", "_");
- }
- string IHelpDynamicGrid.HelpSlug() => HelpSlug();
- protected override bool CanDuplicate => typeof(T).IsAssignableTo(typeof(IDuplicatable));
- public DynamicGrid() : base()
- {
- }
- protected sealed override void PreInit()
- {
- MasterColumns = new DynamicGridColumns();
- MasterColumns.ExtractColumns(typeof(T));
- HiddenColumns = new HiddenColumnsList();
- if (typeof(T).IsAssignableTo(typeof(ISequenceable)))
- {
- HiddenColumns.Add(x => (x as ISequenceable)!.Sequence);
- }
- }
- protected override void Init()
- {
- }
- #region IDynamicGridUIComponentParent
- protected override IDynamicGridUIComponent CreateUIComponent()
- {
- return new DynamicGridGridUIComponent<T>
- {
- Parent = this
- };
- }
- T IDynamicGrid<T>.LoadItem(CoreRow row) => LoadItem(row);
- void IDynamicGridUIComponentParent<T>.EntityChanged(T obj, CoreRow row, string changedColumn, Dictionary<string, object?> changes)
- => EntityChanged(obj, row, changedColumn, changes);
- void IDynamicGridUIComponentParent<T>.UpdateData(T obj, CoreRow row, string changedColumn, Dictionary<CoreColumn, object?> updates)
- {
- var result = new Dictionary<string, object?>();
- foreach (var (col, value) in updates)
- {
- UpdateRow(row, col.ColumnName, value, refresh: false);
- DynamicGridUtils.UpdateEditorValue(new BaseObject[] { obj }, col.ColumnName, value, result);
- }
- EntityChanged(obj, row, changedColumn, result);
- }
- #endregion
- protected override DynamicGridRowStyleSelector GetRowStyleSelector()
- {
- return new DynamicGridRowStyleSelector<T, DynamicGridRowStyle>();
- }
- public DynamicGridColumns MasterColumns { get; protected set; }
- public class HiddenColumnsList
- {
- private List<string> Columns { get; set; } = new();
- public IEnumerable<string> ColumnNames => Columns;
- public void Add(Expression<Func<T, object?>> column) => Add(CoreUtils.GetFullPropertyName(column, "."));
- public void Add(IColumn column) => Add(column.Property);
- public void Add(string column)
- {
- if (!Contains(column))
- Columns.Add(column);
- }
- public bool Contains(string column) => Columns.Contains(column);
- }
- public void AddHiddenColumn(string column) => HiddenColumns.Add(column);
- public HiddenColumnsList HiddenColumns { get; private set; }
- private static bool IsSequenced => typeof(T).GetInterfaces().Any(x => x.Equals(typeof(ISequenceable)));
- #region Options
- protected override void DoReconfigure(DynamicGridOptions options)
- {
- options.ReorderRows = IsSequenced;
- }
- #endregion
- private void EntityChanged(T obj, CoreRow row, string changedColumn, Dictionary<string, object?> changes)
- {
- OnAfterEditorValueChanged(null, [obj], new AfterEditorValueChangedArgs(changedColumn, changes), changes);
- SaveItem(obj);
- foreach (var (key, value) in changes)
- {
- row[key] = value;
- }
- GetUIComponent().UpdateRow(row);
- }
- #region Column Handling
- /// <summary>
- /// Create a set of <see cref="DynamicGridColumns"/> according to the natural default sizes and formats and captions, based on the contents of <paramref name="columns"/>.
- /// </summary>
- /// <param name="columns"></param>
- /// <returns></returns>
- protected DynamicGridColumns ExtractColumns(Columns<T> columns)
- {
- var cols = new DynamicGridColumns();
- foreach (var col in columns)
- {
- var mc = MasterColumns.FirstOrDefault(x => x.ColumnName.Equals(col.Property));
- if (mc != null && mc.Editor is not NullEditor && mc.Editor.Visible != Visible.Hidden)
- cols.Add(mc);
- }
- return cols;
- }
- DynamicGridColumns IDynamicGrid.ExtractColumns(IColumns columns)
- {
- var cols = new DynamicGridColumns();
- foreach (var col in columns)
- {
- var mc = MasterColumns.FirstOrDefault(x => x.ColumnName.Equals(col.Property));
- if (mc != null && mc.Editor is not NullEditor && mc.Editor.Visible != Visible.Hidden)
- cols.Add(mc);
- }
- return cols;
- }
- public override DynamicGridColumns GenerateColumns()
- {
- var cols = IsDirectEditMode()
- ? new Columns<T>(ColumnTypeFlags.IncludeVisible | ColumnTypeFlags.IncludeForeignKeys)
- : new Columns<T>(ColumnTypeFlags.IncludeVisible | ColumnTypeFlags.IncludeLinked);
-
- var columns = ExtractColumns(cols);
- OnGenerateColumns?.Invoke(this, new GenerateColumnsEventArgs(columns));
- return columns;
- }
-
- public event GenerateColumnsEvent? OnGenerateColumns;
- public event SaveColumnsEvent? OnSaveColumns;
- public event GetAvailableColumnsEvent? GetAvailableColumns;
- protected override void SaveColumns(DynamicGridColumns columns)
- {
- OnSaveColumns?.Invoke(this, new(columns));
- }
- #endregion
- #region Refresh / Reload
- protected abstract void Reload(
- Filters<T> criteria, Columns<T> columns, ref SortOrder<T>? sort,
- CancellationToken token, Action<CoreTable?, Exception?> action);
- public Filter<T>? DefineFilter()
- {
- if (OnDefineFilter is null)
- return null;
- var result = OnDefineFilter.Invoke(typeof(T)) as Filter<T>;
- return result;
- }
- public IEnumerable<TType> ExtractValues<TType>(Expression<Func<T, TType>> column, Selection selection)
- {
- var result = selection == Selection.None
- ? Enumerable.Empty<TType>()
- : selection == Selection.Selected
- ? SelectedRows.Select(r => r.Get(column))
- : Data.ExtractValues(column);
- return result;
- }
- protected override void ReloadData(CancellationToken token, Action<CoreTable?, Exception?> action)
- {
- _lookupcache.Clear();
- var criteria = new Filters<T>();
- var filter = DefineFilter();
- if (filter != null)
- criteria.Add(filter);
- var columns = DataColumns();
- var sort = LookupFactory.DefineSort<T>();
- if (sort == null && IsSequenced)
- sort = new SortOrder<T>("Sequence");
- Reload(criteria, columns, ref sort, token, action);
- }
- public Columns<T> DataColumns()
- {
- var columns = Columns.None<T>();
- foreach (var column in VisibleColumns)
- columns.Add(column.ColumnName);
- foreach (var column in HiddenColumns.ColumnNames)
- columns.Add(new Column<T>(column));
- if (!Options.ReadOnly)
- {
- foreach (var column in LookupFactory.RequiredColumns<T>())
- {
- columns.Add(column);
- }
- }
- return columns;
- }
- public void UpdateRow(CoreRow row, T obj, bool invalidateRow = true)
- {
- ObjectToRow(obj, row);
- ObjectToRow(obj, _recordmap[row]);
- if (invalidateRow)
- {
- InvalidateRow(row);
- }
- }
- public void UpdateRows(CoreRow[] rows, T[] objs, bool invalidateRows = true)
- {
- for(var i = 0; i < objs.Length; ++i)
- {
- UpdateRow(rows[i], objs[i], invalidateRow: invalidateRows);
- }
- }
- public void AddRow(T data)
- {
- if (MasterData is null) return;
- MasterData.LoadRow(data);
- Refresh(false, false);
- }
- #endregion
- #region Item Manipulation
- #region Load/Save/Delete
- public virtual T[] LoadItems(IList<CoreRow> rows)
- {
- return rows.ToArray(LoadItem);
- }
- public abstract T LoadItem(CoreRow row);
- public abstract void SaveItem(T item);
- public virtual void SaveItems(IEnumerable<T> items)
- {
- foreach (var item in items)
- SaveItem(item);
- }
- protected virtual bool CanDeleteItems(params CoreRow[] rows)
- {
- return true;
- }
- public abstract void DeleteItems(params CoreRow[] rows);
- protected override bool CanDeleteRows(params CoreRow[] rows) => CanDeleteItems(rows);
- public override void DeleteRows(params CoreRow[] rows) => DeleteItems(rows);
- #endregion
- #region Edit
- protected override void NewRow()
- {
- CreateItems(null);
- }
- protected void CreateItems(Func<IEnumerable<T>>? create)
- {
- var newRows = new List<CoreRow>();
- var items = create?.Invoke() ?? CoreUtils.One(CreateItem());
- foreach (var item in items)
- {
- if (!AfterCreate(item))
- return;
- SaveItem(item);
- var datarow = Data.NewRow();
- ObjectToRow(item, datarow);
- Data.Rows.Add(datarow);
- newRows.Add(datarow);
- var masterrow = MasterData.NewRow();
- ObjectToRow(item, masterrow);
- MasterData.Rows.Add(masterrow);
- _recordmap[datarow] = masterrow;
- }
- InvalidateGrid();
- SelectedRows = newRows.ToArray();
- DoChanged();
- }
- public virtual DynamicEditorPages LoadEditorPages(T item)
- {
- DynamicEditorPages pages = new DynamicEditorPages();
- DynamicGridUtils.LoadOneToManyPages(typeof(T), pages);
- DynamicGridUtils.LoadEnclosedListPages(typeof(T), pages);
- DynamicGridUtils.LoadManyToManyPages(typeof(T), pages);
- DynamicGridUtils.LoadCustomEditorPages(typeof(T), pages);
- pages.RemoveAll(x => !x.Visible);
- foreach (var page in pages)
- page.Ready = false;
- return pages;
- }
- public virtual void LoadEditorButtons(T item, DynamicEditorButtons buttons)
- {
- buttons.Clear();
- buttons.Add(
- "",
- Wpf.Resources.help.AsBitmapImage(),
- item,
- (f, i) =>
- {
- Process.Start(new ProcessStartInfo("https://prsdigital.com.au/wiki/index.php/" + typeof(T).Name.SplitCamelCase().Replace(" ", "_"))
- { UseShellExecute = true });
- }
- );
- OnLoadEditorButtons?.Invoke(item, buttons);
- }
- protected virtual void BeforeLoad(IDynamicEditorForm form, T[] items)
- {
- form.BeforeLoad();
- }
- void IDynamicGrid.InitialiseEditorForm(IDynamicEditorForm editor, object[] items, Func<Type, CoreTable>? pageDataHandler, bool preloadPages)
- {
- InitialiseEditorForm(editor, items.Cast<T>().ToArray(), pageDataHandler, preloadPages);
- }
- public virtual bool EditItems(object[] items, Func<Type, CoreTable?>? PageDataHandler = null, bool PreloadPages = false)
- {
- var values = items.Cast<T>().ToArray();
- return EditItems(values, PageDataHandler, PreloadPages);
- }
- public virtual void InitialiseEditorForm(IDynamicEditorForm editor, T[] items, Func<Type, CoreTable?>? pageDataHandler = null, bool preloadPages = false)
- {
- var pages = items.Length == 1 ? LoadEditorPages(items.First()) : new DynamicEditorPages();
- var buttons = new DynamicEditorButtons();
- if (items.Length == 1)
- LoadEditorButtons(items.First(), buttons);
- editor.Setup(items.Any() ? items.First().GetType() : typeof(T), pages, buttons, pageDataHandler, preloadPages);
- editor.OnCustomiseColumns = (sender, columns) =>
- {
- columns.Clear();
- columns.AddRange(MasterColumns);
- };
- editor.OnDefineEditor = (o, c) =>
- {
- var result = GetEditor(o, c);
- if (result != null)
- result = result.CloneEditor();
- return result;
- };
- editor.OnFormCustomiseEditor += DoCustomiseEditor;
- editor.OnDefineFilter = (type, column) => { return DefineLookupFilter(type, column, items); };
- //editor.OnDefineFilter += (o, e) => { return DefineFilter(items, e); };
- editor.OnDefineLookups = editor => DefineLookups(editor, items);
- editor.OnEditorValueChanged += (s, n, v) => EditorValueChanged(editor, items, n, v);
- editor.OnAfterEditorValueChanged += (g, args) => AfterEditorValueChanged(g, items, args);
- editor.OnReconfigureEditors = g => DoReconfigureEditors(g, items);
- editor.OnValidateData += (o, i) => ValidateData(items);
- editor.OnSelectPage += SelectPage;
- editor.OnSaveItem = (o, e) =>
- {
- try
- {
- using var Wait = new WaitCursor();
- DoBeforeSave(editor, items);
- if (items.Length == 1)
- editor.UnloadEditorPages(false);
- foreach (var item in items)
- SaveItem(item);
- if (items.Length == 1)
- editor.UnloadEditorPages(true);
- DoAfterSave(editor, items);
- }
- catch (Exception err)
- {
- MessageBox.Show(err.Message);
- e.Cancel = true;
- }
- };
- BeforeLoad(editor, items);
- editor.Items = items;
- AfterLoad(editor, items);
- }
- private void DoCustomiseEditor(IDynamicEditorForm sender, object[] items, DynamicGridColumn column, BaseEditor editor)
- {
- CustomiseEditor(sender, (T[])items, column, editor);
- OnCustomiseEditor?.Invoke(sender, (T[])items, column, editor);
- }
- protected virtual void CustomiseEditor(IDynamicEditorForm form, T[] items, DynamicGridColumn column, BaseEditor editor)
- {
- }
- protected virtual void DoAfterSave(IDynamicEditorForm editor, T[] items)
- {
- OnAfterSave?.Invoke(editor, items);
- }
- protected virtual void DoBeforeSave(IDynamicEditorForm editor, T[] items)
- {
- OnBeforeSave?.Invoke(editor, items);
- }
- /// <summary>
- /// Edit the <paramref name="items"/> with a non-modal editor window, attaching them to the <paramref name="host"/> provided.
- /// </summary>
- /// <param name="items">List of objects to edit.</param>
- /// <param name="callback">A callback for when the items are finished being edited.</param>
- public virtual void EditItemsNonModal(ISubPanelHost host, T[] items, Action<T[], bool> callback, Func<Type, CoreTable?>? PageDataHandler = null, bool PreloadPages = false)
- {
- if (!DynamicGridUtils.TryEdit(items, out var editLock))
- {
- if(editLock.Panel is Window window)
- {
- Task.Delay(100).ContinueWith(task =>
- {
- window.WindowState = WindowState.Normal;
- window.Activate();
- }, TaskScheduler.FromCurrentSynchronizationContext());
- }
- else
- {
- MessageWindow.ShowMessage("One or more items are already being edited in an open window. You cannot edit the same entity multiple times simultaneously.", "Simultaneous edit.");
- }
- return;
- }
- DynamicEditorForm editor;
- using (var cursor = new WaitCursor())
- {
- editor = new DynamicEditorForm();
- editor.SetValue(Panel.ZIndexProperty, 999);
- editor.Form.DisableOKIfUnchanged = true;
- InitialiseEditorForm(editor, items, PageDataHandler, PreloadPages);
- OnEditorLoaded?.Invoke(editor, items);
- }
- editLock.Panel = editor;
- host.AddSubPanel(editor);
- editor.SubPanelClosed += o =>
- {
- try
- {
- DynamicGridUtils.FinishEdit(items);
- callback(items, editor.Result == true);
- }
- catch(Exception e)
- {
- MessageWindow.ShowError("Error occurred while closing editor.", e);
- }
- };
- editor.Show();
- }
- /// <summary>
- /// Edit the <paramref name="items"/> with a modal editor window.
- /// </summary>
- /// <param name="items">List of objects to edit.</param>
- /// <returns><see langword="true"/> if the items were saved.</returns>
- public virtual bool EditItems(T[] items, Func<Type, CoreTable?>? PageDataHandler = null, bool PreloadPages = false)
- {
- DynamicEditorForm editor;
- using (var cursor = new WaitCursor())
- {
- editor = new DynamicEditorForm()
- {
- WindowStartupLocation = WindowStartupLocation.CenterScreen
- };
-
- editor.SetValue(Panel.ZIndexProperty, 999);
-
- InitialiseEditorForm(editor, items, PageDataHandler, PreloadPages);
- OnEditorLoaded?.Invoke(editor, items);
- }
- return editor.ShowDialog() == true;
- }
- private Dictionary<String, object?> AfterEditorValueChanged(DynamicEditorGrid grid, T[] items, AfterEditorValueChangedArgs args)
- {
- var changes = new Dictionary<string, object?>();
- OnAfterEditorValueChanged(grid, items, args, changes);
- return changes;
- }
- protected virtual void OnAfterEditorValueChanged(DynamicEditorGrid? grid, T[] items, AfterEditorValueChangedArgs args, Dictionary<String, object?> changes)
- {
- }
-
- protected virtual void DoReconfigureEditors(DynamicEditorGrid grid, T[] items)
- {
- /*if (items.First() is IDimensioned dimensioned)
- {
- UpdateEditor(grid, x => x.Dimensions.Quantity, dimensioned.Dimensions.GetUnit().HasQuantity);
- UpdateEditor(grid, x => x.Dimensions.Length, dimensioned.Dimensions.GetUnit().HasLength);
- UpdateEditor(grid, x => x.Dimensions.Width, dimensioned.Dimensions.GetUnit().HasWidth);
- UpdateEditor(grid, x => x.Dimensions.Height, dimensioned.Dimensions.GetUnit().HasHeight);
- UpdateEditor(grid, x => x.Dimensions.Weight, dimensioned.Dimensions.GetUnit().HasWeight);
- }*/
- }
- private List<string>? ValidateData(T[] items)
- {
- var errors = new List<string>();
- DoValidate(items, errors);
- OnValidate?.Invoke(this, items, errors);
- return errors.Count != 0 ? errors : null;
- }
- protected virtual void DoValidate(T[] items, List<string> errors)
- {
- }
- protected virtual void AfterLoad(IDynamicEditorForm editor, T[] items)
- {
- editor.AfterLoad();
- }
- protected virtual void SelectPage(object sender, BaseObject[]? items)
- {
- }
- protected virtual Dictionary<string, object?> EditorValueChanged(IDynamicEditorForm editor, T[] items, string name, object value)
- {
- var result = DynamicGridUtils.UpdateEditorValue(items, name, value);
- if (OnEditorValueChanged != null)
- {
- var newchanges = OnEditorValueChanged(editor, name, value);
- foreach (var key in newchanges.Keys)
- result[key] = newchanges[key];
- }
- return result;
- }
- private readonly Dictionary<Tuple<Type, Type>, Dictionary<object, object>> _lookupcache = new();
- protected virtual void DefineLookups(ILookupEditorControl sender, T[] items, bool async = true)
- {
- if (sender.EditorDefinition is not ILookupEditor editor)
- return;
- var colname = sender.ColumnName;
- if (async)
- {
- Task.Run(() =>
- {
- try
- {
- var values = editor.Values(colname, items);
- Dispatcher.Invoke(
- () =>
- {
- try
- {
- //Logger.Send(LogType.Information, typeof(T).Name, "Dispatching Results" + colname);
- sender.LoadLookups(values);
- }
- catch (Exception e2)
- {
- Logger.Send(LogType.Information, typeof(T).Name,
- "Exception (2) in LoadLookups: " + e2.Message + "\n" + e2.StackTrace);
- }
- }
- );
- }
- catch (Exception e)
- {
- Logger.Send(LogType.Information, typeof(T).Name,
- "Exception (1) in LoadLookups: " + e.Message + "\n" + e.StackTrace);
- }
- });
- }
- else
- {
- var values = editor.Values(colname, items);
- sender.LoadLookups(values);
- }
- }
- /// <summary>
- /// Retrieves an editor to display for the given column of <paramref name="item"/>.
- /// </summary>
- /// <param name="item">The object being edited.</param>
- /// <param name="column">The column of the editor.</param>
- /// <returns>A new editor, or <see langword="null"/> if no editor defined and no sensible default exists.</returns>
- protected virtual BaseEditor? GetEditor(object item, DynamicGridColumn column)
- {
- return column.Editor ?? DatabaseSchema.Property(item.GetType(), column.ColumnName)?.Editor;
- }
- protected IFilter? DefineLookupFilter(Type type, string column, T[] items)
- {
- return LookupFactory.DefineLookupFilter(typeof(T), type, column, items);
- }
- protected virtual void SetEditorValue(object item, string name, object value)
- {
- try
- {
- CoreUtils.SetPropertyValue(item, name, value);
- }
- catch (Exception e)
- {
- Logger.Send(LogType.Error, "", string.Format("*** Unknown Error: {0}\n{1}", e.Message, e.StackTrace));
- }
- }
- protected virtual object? GetEditorValue(object item, string name)
- {
- return CoreUtils.GetPropertyValue(item, name);
- }
- protected override bool CanCreateRows() => CanCreateItems();
- protected virtual bool CanCreateItems()
- {
- return true;
- }
- protected override bool EditRows(CoreRow[]? rows)
- {
- if(rows is null)
- {
- if (!CanCreateItems())
- return false;
- var item = CreateItem();
-
- if (!AfterCreate(item))
- return false;
- if(Options.NonModalEditorHost is not null)
- {
- EditItemsNonModal(Options.NonModalEditorHost, [item], (items, result) =>
- {
- if (result)
- {
- var row = Data.NewRow();
- ObjectToRow(item, row);
- Data.Rows.Add(row);
- var masterrow = MasterData.NewRow();
- ObjectToRow(item, masterrow);
- MasterData.Rows.Add(masterrow);
- _recordmap[row] = masterrow;
- InvalidateGrid();
- SelectedRows = [row];
- DoChanged();
- }
- });
- return false;
- }
- else
- {
- if (EditItems([item]))
- {
- //_CurrentRow = Data.Rows.Count;
- var row = Data.NewRow();
- ObjectToRow(item, row);
- Data.Rows.Add(row);
- var masterrow = MasterData.NewRow();
- ObjectToRow(item, masterrow);
- MasterData.Rows.Add(masterrow);
- _recordmap[row] = masterrow;
- InvalidateGrid();
- SelectedRows = [row];
- DoChanged();
- return true;
- }
- return false;
- }
- }
- else
- {
- var items = Array.Empty<T>();
- using (new WaitCursor())
- {
- Stopwatch sw = new Stopwatch();
- sw.Start();
- items = LoadItems(rows);
- //Logger.Send(LogType.Information, "DG:LoadItems", String.Format("Loaded Items: {0}ms", sw.ElapsedMilliseconds));
- sw.Stop();
- }
- if (items.Length != 0)
- {
- var snaps = items.ToArray(x => x.TakeSnapshot());
- if(Options.NonModalEditorHost is not null)
- {
- EditItemsNonModal(Options.NonModalEditorHost, items, (items, result) =>
- {
- if (result)
- {
- var sel = SelectedRows;
- UpdateRows(rows, items, invalidateRows: false);
- InvalidateGrid();
- SelectedRows = sel;
- DoChanged();
- }
- else
- {
- foreach(var snap in snaps)
- {
- snap.ResetObject();
- }
- }
- });
- return false;
- }
- else
- {
- if (EditItems(items))
- {
- var sel = SelectedRows;
- UpdateRows(rows, items, invalidateRows: false);
- InvalidateGrid();
- SelectedRows = sel;
- DoChanged();
- return true;
- }
- else
- {
- foreach(var snap in snaps)
- {
- snap.ResetObject();
- }
- }
- }
- return false;
- }
- return false;
- }
- }
- #endregion
- #region Duplicate
- protected virtual IEnumerable<T> LoadDuplicatorItems(CoreRow[] rows)
- {
- return LoadItems(rows);
- }
- bool IDuplicateDynamicGrid.DoDuplicate(CoreRow[] rows) => DoDuplicate(rows);
- protected virtual bool DoDuplicate(CoreRow[] rows)
- {
- if (rows.Length == 0)
- {
- MessageBox.Show("Please select at least one record to duplicate!");
- return false;
- }
- /*var ids = ExtractValues(x => x.ID, Selection.Selected).ToArray();
- if (!ids.Any())
- {
- MessageBox.Show("Please select at least one record to duplicate!");
- return false;
- }*/
- var duplicator = (new T() as IDuplicatable)?.GetDuplicator();
- if (duplicator is null)
- {
- MessageBox.Show($"Cannot duplicate {typeof(T)}");
- return false;
- }
- duplicator.Duplicate(LoadDuplicatorItems(rows));// new Filter<T>(x => x.ID).InList(ids));
- return true;
- }
- #endregion
- public virtual T CreateItem()
- {
- var result = new T();
- OnCreateItem?.Invoke(this, result);
- return result;
- }
- public virtual bool AfterCreate(T item)
- {
- return OnAfterCreateItem?.Invoke(this, item) ?? true;
- }
- protected void ReloadForms<TTargetType, TTargetForm, TSourceForm>(IDynamicEditorForm editor, TTargetType item,
- Expression<Func<TSourceForm, object?>> sourcekey, Guid sourceid)
- where TTargetType : Entity, new()
- where TTargetForm : Entity, IRemotable, IPersistent, IDigitalFormInstance, new()
- where TSourceForm : Entity, IRemotable, IPersistent, IDigitalForm<TTargetType>, new()
- {
- var type = typeof(IDynamicOneToManyGrid<,>).MakeGenericType(typeof(TTargetType), typeof(TTargetForm));
- var page =
- editor.Pages?.FirstOrDefault(x => x.GetType().GetInterfaces().Contains(type)) as IDynamicOneToManyGrid<TTargetType, TTargetForm>;
- if (page != null && item != null)
- {
- if (!page.Ready)
- page.Load(item, null);
- CoreTable table;
- if (sourceid == Guid.Empty)
- {
- table = new CoreTable();
- table.LoadColumns(typeof(TSourceForm));
- }
- else
- {
- table = new Client<TSourceForm>().Query(
- new Filter<TSourceForm>(sourcekey).IsEqualTo(sourceid).And(x => x.Form.AppliesTo)
- .IsEqualTo(typeof(TTargetType).EntityName().Split('.').Last())
- );
- }
- var newforms = new List<TTargetForm>();
- foreach (var row in table.Rows)
- {
- var sourceform = row.ToObject<TSourceForm>();
- var targetform = new TTargetForm();
- targetform.Form.ID = sourceform.Form.ID;
- targetform.Form.Synchronise(sourceform.Form);
- newforms.Add(targetform);
- }
- page.Items.Clear();
- page.LoadItems(newforms.ToArray());
- }
- }
- #region ClipBuffer
- protected virtual bool BeforeCopy(IList<T> items)
- {
- return true;
- }
- protected override void MoveRows(CoreRow[] rows, int index, bool isCopy = false)
- {
- if (!Options.ReorderRows || !IsSequenced) return;
- var items = LoadItems(rows);
- if (isCopy)
- {
- if (!BeforeCopy(items))
- {
- return;
- }
- }
- var sequence = index < Data.Rows.Count
- ? Data.Rows[index].Get<ISequenceable, long>(x => x.Sequence)
- : Data.Rows[^1].Get<ISequenceable, long>(x => x.Sequence) + 1;
- var postRows = Data.Rows.Where(r => !rows.Contains(r) && r.Get<ISequenceable, long>(x => x.Sequence) >= sequence);
- var updates = items.Concatenate(LoadItems(postRows.ToArray()));
- foreach (var update in updates)
- {
- ((ISequenceable)update).Sequence = sequence;
- sequence++;
- }
- if (updates.Length != 0)
- {
- SaveItems(updates);
- DoChanged();
- Refresh(false, true);
- }
- }
- #endregion
- protected virtual void ObjectToRow(T obj, CoreRow row)
- {
- row.Table.FillRow(row, obj);
- }
- #region Import / Export
- protected virtual CoreTable LoadImportKeys(String[] fields)
- {
- var result = new CoreTable();
- result.LoadColumns(Columns.None<T>().Add(fields));
- return result;
- }
- protected virtual Guid GetImportID()
- {
- return Guid.Empty;
- }
- protected virtual bool CustomiseImportItem(T item)
- {
- if (IsSequenced)
- ((ISequenceable)item).Sequence = CoreUtils.GenerateSequence();
- return true;
- }
- protected virtual string CustomiseImportFileName(string filename)
- {
- return filename;
- }
- void IImportDynamicGrid.DoImport() => DoImport();
- protected virtual void DoImport()
- {
- var list = new DynamicImportList(
- typeof(T),
- GetImportID()
- );
- list.OnImportItem += o => { return CustomiseImportItem((T)o); };
- list.OnCustomiseImport += (o, args) => { args.FileName = CustomiseImportFileName(args.FileName); };
- list.OnSave += (sender, entity) => SaveItem(entity as T);
- list.OnLoad += (sender, type, fields, id) => LoadImportKeys(fields);
- list.ShowDialog();
- Refresh(false, true);
- }
- protected virtual void CustomiseExportColumns(List<string> columnnames)
- {
- }
- protected virtual string CustomiseExportFileName(string filename)
- {
- return filename;
- }
- protected virtual void CustomiseExportFilters(Filters<T> filters, CoreRow[] visiblerows)
- {
- }
- protected virtual void ApplyExportFilter(CoreTable table, object data)
- {
- }
- void IExportDynamicGrid.DoExport() => DoExport();
- protected virtual void DoExport()
- {
- var columnnames = VisibleColumns.Select(x => x.ColumnName).ToList();
- CustomiseExportColumns(columnnames);
- var form = new DynamicExportForm(typeof(T), columnnames);
- if (form.ShowDialog() != true)
- return;
- var filters = new Filters<T>();
- filters.Add(DefineFilter());
- var predicates = GetUIComponent().GetFilterPredicates();
- var visiblerows = GetVisibleRows();
- CustomiseExportFilters(filters, visiblerows);
- var columns = Columns.None<T>().Add(form.Fields);
- var otherColumns = form.GetChildFields()
- .Select(x => new Tuple<Type, IColumns>(
- x.Key,
- Columns.None(x.Key).Add(x.Value)))
- .Where(x => x.Item2.ColumnNames().Any()).ToList();
- var reloadColumns = Columns.None<T>();
- foreach (var column in columns)
- {
- reloadColumns.Add(column);
- }
- foreach (var column in HiddenColumns.ColumnNames)
- {
- reloadColumns.Add(column);
- }
- foreach (var column in LookupFactory.RequiredColumns<T>())
- {
- columns.Add(column);
- }
- foreach (var (column, _) in predicates)
- {
- reloadColumns.Add(column);
- }
- CoreTable? data = null;
- void LoadExport()
- {
- var newData = new CoreTable();
- newData.LoadColumns(columns);
- FilterRows(data.Rows, newData, filter: row =>
- {
- foreach(var (_, predicate) in predicates)
- {
- if (!predicate(row))
- {
- return false;
- }
- }
- return true;
- });
- var list = new List<Tuple<Type?, CoreTable>>() { new(typeof(T), newData) };
- list.AddRange(LoadExportTables(filters, otherColumns));
- DoExportTables(list);
- }
- var sort = LookupFactory.DefineSort<T>();
- Reload(filters, reloadColumns, ref sort, CancellationToken.None, (table, err) =>
- {
- if (table is not null)
- {
- if (table.Offset == 0 || data is null)
- {
- data = table;
- if (!IsPaging)
- {
- Dispatcher.Invoke(LoadExport);
- }
- }
- else
- {
- data.AddPage(table);
- if (!IsPaging)
- {
- Dispatcher.Invoke(LoadExport);
- }
- }
- }
- else if (err is not null)
- {
- Dispatcher.Invoke(() =>
- {
- MessageWindow.ShowError("Error in export.", err);
- });
- }
- });
- }
- /// <summary>
- /// Loads the child tables for an export, based on the filter of the parent table.
- /// </summary>
- /// <remarks>
- /// If not overriden, defaults to creating empty tables with no records.
- /// </remarks>
- /// <param name="filter">Filter for the parent table.</param>
- /// <param name="tableColumns">A list of the child table types, with columns to load for each</param>
- /// <returns>A list of tables, in the same order as they came in <paramref name="tableColumns"/></returns>
- protected virtual IEnumerable<Tuple<Type?, CoreTable>> LoadExportTables(Filters<T> filter, IEnumerable<Tuple<Type, IColumns>> tableColumns)
- {
- return tableColumns.Select(x =>
- {
- var table = new CoreTable();
- table.LoadColumns(x.Item2);
- return new Tuple<Type?, CoreTable>(x.Item1, table);
- });
- }
- private void DoExportTables(List<Tuple<Type?, CoreTable>> data)
- {
- var filename = CustomiseExportFileName(typeof(T).EntityName().Split('.').Last());
- ExcelExporter.DoExport(data, filename);
- }
- #endregion
- #endregion
- #region Header Actions
- protected override bool SelectColumns([NotNullWhen(true)] out DynamicGridColumns? columns)
- {
- var schema = new DynamicGridObjectColumnSchema(typeof(T));
- var editor = new DynamicGridColumnsEditor(schema, typeof(T)) { WindowStartupLocation = WindowStartupLocation.CenterScreen };
- editor.DirectEdit = IsDirectEditMode();
- editor.Columns.AddRange(VisibleColumns);
- schema.OnProcessColumns += args =>
- {
- GetAvailableColumns?.Invoke(args);
- };
- if (editor.ShowDialog().Equals(true))
- {
- columns = editor.Columns;
- return true;
- }
- else
- {
- columns = null;
- return false;
- }
- }
- #endregion
- #region Drag + Drop
- protected DragDropEffects DragTable(Type entity, CoreTable table)
- {
- Logger.Send(LogType.Information, "", "DragTable");
- var data = new DataObject();
- data.SetData(DynamicGridUtils.DragFormat, new DynamicGridDragFormat(table.ToDataTable(), entity));
- var effect = DragDrop.DoDragDrop(this, data, DragDropEffects.All);
- return effect;
- }
- protected override DragDropEffects OnRowsDragStart(CoreRow[] rows)
- {
- Logger.Send(LogType.Information, "", "OnRowsDragStart");
- var table = new CoreTable();
- table.LoadColumns(Data.Columns);
- table.LoadRows(rows);
- return DragTable(typeof(T), table);
- }
- #endregion
- }
- #region Styling
- public class DynamicGridRowStyle : DynamicGridStyle<VirtualizingCellsControl>
- {
- public DynamicGridRowStyle() : base(null)
- {
- }
- public DynamicGridRowStyle(IDynamicGridStyle source) : base(source)
- {
- }
- public override DependencyProperty FontSizeProperty => Control.FontSizeProperty;
- public override DependencyProperty FontStyleProperty => Control.FontStyleProperty;
- public override DependencyProperty FontWeightProperty => Control.FontWeightProperty;
- public override DependencyProperty BackgroundProperty => Control.BackgroundProperty;
- public override DependencyProperty ForegroundProperty => Control.ForegroundProperty;
- }
- public class DynamicGridCellStyle : DynamicGridStyle<Control>
- {
- public DynamicGridCellStyle() : base(null)
- {
- }
- public DynamicGridCellStyle(IDynamicGridStyle source) : base(source)
- {
- }
- public override DependencyProperty FontSizeProperty => Control.FontSizeProperty;
- public override DependencyProperty FontStyleProperty => Control.FontStyleProperty;
- public override DependencyProperty FontWeightProperty => Control.FontWeightProperty;
- public override DependencyProperty BackgroundProperty => Control.BackgroundProperty;
- public override DependencyProperty ForegroundProperty => Control.ForegroundProperty;
- }
- // Used to render boolean columns (the default "false" value shows what appears to be an intermediate state, which is ugly
- // This should show nothing for false, and a tick in a box for true
- public class BoolToImageConverter : AbstractConverter<bool, ImageSource?>
- {
- public ImageSource TrueValue { get; set; }
- public ImageSource? FalseValue { get; set; }
- public BoolToImageConverter()
- {
- TrueValue = Wpf.Resources.Bullet_Tick.AsBitmapImage();
- }
-
- public override ImageSource? Convert(bool value)
- {
- return value ? TrueValue : FalseValue;
- }
- public override bool Deconvert(ImageSource? value)
- {
- return ImageUtils.IsEqual(value as BitmapImage, TrueValue as BitmapImage);
- }
- }
- public class StringToColorImageConverter : IValueConverter
- {
- private readonly int _height = 50;
- private readonly int _width = 25;
- private readonly Dictionary<string, BitmapImage> cache = new();
- public StringToColorImageConverter(int width, int height)
- {
- _width = width;
- _height = height;
- }
- public object? Convert(object value, Type targetType, object parameter, CultureInfo culture)
- {
- var str = value?.ToString();
- if (str is null)
- return null;
- var colorcode = str.TrimStart('#');
- if (!cache.ContainsKey(colorcode))
- {
- var col = ImageUtils.StringToColor(colorcode);
- var bmp = ImageUtils.BitmapFromColor(col, _width, _height, Color.Black);
- cache[colorcode] = bmp.AsBitmapImage();
- }
- var result = cache[colorcode];
- return result;
- }
- public object? ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
- {
- return null;
- }
- }
- public class StringArrayConverter : IValueConverter
- {
- object? IValueConverter.Convert(object value, Type targetType, object parameter, CultureInfo culture)
- {
- if (value is string[] strArray)
- {
- return string.Join("\n", strArray);
- }
- Logger.Send(LogType.Error, "", $"Attempt to convert an object which is not a string array: {value}.");
- return null;
- }
- object IValueConverter.ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
- {
- return value;
- }
- }
- #endregion
- [Serializable]
- class DynamicGridDragFormat
- {
- private string entity;
- public DataTable Table { get; set; }
- public Type Entity
- {
- get => CoreUtils.GetEntity(entity);
- [MemberNotNull(nameof(entity))]
- set => entity = value.EntityName();
- }
- public DynamicGridDragFormat(DataTable table, Type entity)
- {
- Table = table;
- Entity = entity;
- }
- }
|