| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480 | using System;using System.Collections.Generic;using System.Drawing;using System.IO;using System.Linq;using System.Text.RegularExpressions;using System.Windows;using System.Windows.Controls;using System.Windows.Media.Imaging;using InABox.Core;using InABox.DynamicGrid;using InABox.Scripting;using InABox.WPF;using Microsoft.Win32;using Org.BouncyCastle.Asn1.Mozilla;using Syncfusion.UI.Xaml.Spreadsheet;using Syncfusion.Windows.Shared;using UnderlineType = InABox.Core.UnderlineType;namespace InABox.DynamicGrid{    public static class FormLayoutImporter    {        private class Cell        {            public string Content { get; set; }            public int Row { get; set; }            public int Column { get; set; }            public int RowSpan { get; set; } = 1;            public int ColumnSpan { get; set; } = 1;            public ICell InnerCell { get; set; }            public Cell(int row, int column, string content, ICell cell)            {                Row = row;                Column = column;                Content = content;                InnerCell = cell;            }        }        private static void DeleteColumn(List<Cell> cells, int column)        {            foreach(var cell in cells)            {                if(cell.Column <= column && cell.Column + cell.ColumnSpan - 1 >= column)                {                    --cell.ColumnSpan;                }                else if(cell.Column > column)                {                    --cell.Column;                }            }            cells.RemoveAll(x => x.ColumnSpan < 0);        }        private static List<Cell> GetCells(ISheet sheet)        {            var grid = new Dictionary<int, Dictionary<int, Cell>>();            for (int rowIdx = sheet.FirstRow; rowIdx <= sheet.LastRow; ++rowIdx)            {                var row = sheet.GetRow(rowIdx);                if (row is not null && row.FirstColumn >= 0)                {                    var rowCells = new Dictionary<int, Cell>();                    for (int colIdx = row.FirstColumn; colIdx <= row.LastColumn; ++colIdx)                    {                        var cell = row.GetCell(colIdx);                        if (cell is not null)                        {                            rowCells.Add(colIdx, new Cell(rowIdx, colIdx, cell.GetValue(), cell));                        }                    }                    grid.Add(rowIdx, rowCells);                }            }            foreach (var region in sheet.GetMergedCells())            {                for (int r = region.FirstRow; r <= region.LastRow; ++r)                {                    if (!grid.TryGetValue(r, out var row)) continue;                    for (int c = region.FirstColumn; c <= region.LastColumn; ++c)                    {                        if ((r - region.FirstRow) + (c - region.FirstColumn) != 0)                        {                            row.Remove(c);                        }                    }                    if (row.Count == 0)                    {                        grid.Remove(r);                    }                }                if (grid.TryGetValue(region.FirstRow, out var cRow) && cRow.TryGetValue(region.FirstColumn, out var cCell))                {                    cCell.RowSpan = region.LastRow - region.FirstRow + 1;                    cCell.ColumnSpan = region.LastColumn - region.FirstColumn + 1;                }            }            var cells = new List<Cell>();            foreach (var row in grid.Values)            {                foreach (var cell in row.Values)                {                    cells.Add(cell);                }            }            return cells;        }        private static Regex VariableRegex = new("^\\s*\\[(?<VAR>[^:\\]]+)(?::(?<TYPE>[^\\]]+))?\\]$");        public static DFLayout LoadLayout(ISpreadsheet spreadsheet)        {            var sheet = spreadsheet.GetSheet(0);            var cells = GetCells(sheet);            int firstRow = int.MaxValue;            int lastRow = 0;            int firstCol = int.MaxValue;            int lastCol = 0;            foreach (var cell in cells)            {                firstCol = Math.Min(cell.Column, firstCol);                lastCol = Math.Max(cell.Column + cell.ColumnSpan - 1, lastCol);                firstRow = Math.Min(cell.Row, firstRow);                lastRow = Math.Max(cell.Row + cell.RowSpan - 1, lastRow);            }            var layout = new DFLayout();            var columnWidths = new Dictionary<int, float>();            var colOffset = 0;            for (int col = firstCol; col <= lastCol; ++col)            {                var width = sheet.GetColumnWidth(col);                if(width == float.MinValue)                {                    layout.ColumnWidths.Add("10*");                }                else if(width <= 0f)                {                    DeleteColumn(cells, col);                }                else                {                    layout.ColumnWidths.Add($"{width}*");                }            }            /*var rowHeights = new Dictionary<int, float>();            for (int row = firstRow; row <= lastRow; ++row)            {                rowHeights[row] = sheet.GetRowHeight(row);            }            var totalHeight = rowHeights.Values.Sum();*/            for(int row = firstRow; row <= lastRow; ++row)            {                layout.RowHeights.Add("Auto");            }            foreach(var cell in cells)            {                var style = cell.InnerCell.GetStyle();                if (string.IsNullOrWhiteSpace(cell.Content) && style.Foreground == Color.Empty) continue;                DFLayoutControl? control;                var match = VariableRegex.Match(cell.Content);                if (match.Success)                {                    var variableName = match.Groups["VAR"];                    var variableType = match.Groups["TYPE"];                    Type? fieldType = null;                    if (variableType.Success)                    {                        fieldType = DFUtils.GetFieldType(variableType.Value);                    }                    fieldType ??= typeof(DFLayoutStringField);                    var field = (Activator.CreateInstance(fieldType) as DFLayoutField)!;                    field.Name = variableName.Value;                    control = field;                }                else                {                    var font = style.Font;                    control = new DFLayoutLabel                    {                        Caption = cell.Content,                        Style = new DFLayoutTextStyle                        {                            FontSize = font.FontSize,                            IsItalic = font.Italic,                            IsBold = font.Bold,                            Underline = font.Underline switch                            {                                Scripting.UnderlineType.None => UnderlineType.None,                                Scripting.UnderlineType.Single or Scripting.UnderlineType.SingleAccounting => UnderlineType.Single,                                Scripting.UnderlineType.Double or Scripting.UnderlineType.DoubleAccounting => UnderlineType.Double,                                _ => UnderlineType.None                            },                            BackgroundColour = style.Foreground,                            ForegroundColour = font.Colour                        }                    };                }                if(control is not null)                {                    control.Row = cell.Row - firstRow + 1;                    control.Column = cell.Column - firstCol + 1 - colOffset;                    control.RowSpan = cell.RowSpan;                    control.ColumnSpan = cell.ColumnSpan;                    layout.Elements.Add(control);                }            }            return layout;        }    }    public abstract class DynamicFormLayoutGrid<T> : DynamicOneToManyGrid<DigitalForm, DigitalFormLayout> where T : Entity, IRemotable, IPersistent, new()    {        private readonly BitmapImage design = Properties.Resources.design.AsBitmapImage();        public DynamicFormLayoutGrid()        {            Options.AddRange(DynamicGridOption.RecordCount, DynamicGridOption.ImportData);            ActionColumns.Add(new DynamicImageColumn(DesignImage, DesignClick));            //AddButton("Design", PRSDesktop.Resources.design.AsBitmapImage(), DesignClick);            HiddenColumns.Add(x => x.Layout);            AddButton("Auto Generate", null, AutoGenerate_Click);            AddButton("Duplicate", null, Duplicate_Click);        }        private DFLayout LoadLayoutFromSpreadsheet(ISpreadsheet spreadsheet)        {            return FormLayoutImporter.LoadLayout(spreadsheet);        }        protected override void DoImport()        {            var dialog = new OpenFileDialog();            dialog.Filter = "Excel Spreadsheet (.xlsx)|*.xlsx";            if (dialog.ShowDialog() == true)            {                try                {                    DFLayout layout;                    using (var fs = new FileStream(dialog.FileName, FileMode.Open, FileAccess.Read, FileShare.Read))                    {                        layout = LoadLayoutFromSpreadsheet(new Spreadsheet(fs));                    }                    var dfLayout = CreateItem();                    dfLayout.Layout = layout.SaveLayout();                    if(EditItems(new DigitalFormLayout[] { dfLayout }))                    {                        var newVariables = new List<DigitalFormVariable>();                        foreach (var element in layout.Elements)                        {                            if(element is DFLayoutField field)                            {                                var variable = new DigitalFormVariable                                {                                    Code = field.Name,                                    Description = field.Name                                };                                variable.SetFieldType(field.GetType());                                newVariables.Add(variable);                            }                        }                        if(newVariables.Count > 0)                        {                            var variables = GetVariableGrid();                            if (variables is not null)                            {                                var save = new List<DigitalFormVariable>();                                foreach(var newVariable in newVariables)                                {                                    var variable = variables.GetVariable(newVariable.Code);                                    if(variable is not null)                                    {                                        if(variable.FieldType() != newVariable.FieldType())                                        {                                            MessageBox.Show($"Variable [{newVariable.Code}] already exists with a different type!");                                        }                                    }                                    else                                    {                                        save.Add(newVariable);                                    }                                }                                variables.SaveItems(save.ToArray());                                variables.Refresh(false, true);                            }                        }                        Refresh(false, true);                    }                }                catch(Exception e)                {                    Logger.Send(LogType.Error, "", CoreUtils.FormatException(e));                    MessageBox.Show($"Error: {e.Message}");                }            }        }        private bool Duplicate_Click(Button btn, CoreRow[] rows)        {            if (!rows.Any()) return false;            SaveItems(rows.Select(x =>            {                var layout = x.ToObject<DigitalFormLayout>();                layout.ID = Guid.Empty;                return layout;            }).ToArray());            return true;        }        private bool AutoGenerate_Click(Button btn, CoreRow[] rows)        {            var menu = new ContextMenu();            menu.AddItem("Desktop Layout", null, AddDesktop_Click);            menu.AddItem("Mobile Layout", null, AddMobile_Click);            menu.IsOpen = true;            return false;        }        private BitmapImage? DesignImage(CoreRow? row)        {            return row != null ? design : null;        }        private void AddMobile_Click()        {            var item = CreateItem();            item.Layout = DFLayout.GenerateAutoMobileLayout(GetVariables()).SaveLayout();            item.Type = DFLayoutType.Mobile;            if (EditItems(new[] { item }))            {                SaveItem(item);                Refresh(false, true);                DoChanged();            }        }        private void AddDesktop_Click()        {            var item = CreateItem();            item.Layout = DFLayout.GenerateAutoDesktopLayout(GetVariables()).SaveLayout();            item.Type = DFLayoutType.Desktop;            if (EditItems(new[] { item }))            {                SaveItem(item);                Refresh(false, true);                DoChanged();            }        }        private DynamicVariableGrid? GetVariableGrid()            => EditorGrid.Pages?.FirstOrDefault(x => x is DynamicVariableGrid)                as DynamicVariableGrid;        private List<DigitalFormVariable> GetVariables()            => GetVariableGrid()?.Items.ToList() ?? new List<DigitalFormVariable>();        private void Design(DigitalFormLayout layout)        {            var variables = GetVariables();            var newVariables = new List<DigitalFormVariable>();            var form = new DynamicFormDesignWindow            {                Type = layout.Type            };            form.OnCreateVariable += (fieldType) =>            {                if (DynamicVariableUtils.CreateAndEdit(Item, GetVariables(), fieldType, out var variable))                {                    newVariables.Add(variable);                    return variable;                }                return null;            };            /*form.OnEditVariable += (variable) =>            {                var properties = variable.CreateProperties();                if (DynamicVariableUtils.EditProperties(Item, GetVariables(), properties.GetType(), properties))                {                    variable.SaveProperties(properties);                    return true;                }                return false;            };*/            form.LoadLayout(layout, variables);            form.Initialize();            if (form.ShowDialog() == true)            {                layout.Layout = form.SaveLayout();                SaveItem(layout);                var grid = GetVariableGrid();                if (grid is not null)                {                    grid.SaveItems(newVariables.ToArray());                    grid.Refresh(false, true);                }            }        }        private bool DesignClick(CoreRow? row)        {            if (row == null)                return false;            Design(LoadItem(row));            return false;        }        //public override void SaveItem(DigitalFormLayout item)        //{        //    bool bActive = item.Active;        //    foreach (var other in Items.Where(x=>(x != item) && (x.Type == item.Type)))        //    {        //        if (item.Active)        //        {        //            if (other.Active)        //                other.Active = false;        //        }        //        else        //            bActive = bActive || other.Active;        //    }        //    if (!bActive)        //        item.Active = true;        //    base.SaveItem(item);        //}        protected override void DoDoubleClick(object sender)        {            DesignClick(SelectedRows.FirstOrDefault());        }    }}
 |