Files
Terminal.Gui/UICatalog/Scenarios/TableEditor.cs
Tig dcb3b359ad Fixes #2926 - Refactor KeyEvent and KeyEventEventArgs to simplify (#2927)
* Adds basic MainLoop unit tests

* Remove WinChange action from Curses

* Remove WinChange action from Curses

* Remove ProcessInput action from Windows MainLoop

* Simplified MainLoop/ConsoleDriver by making MainLoop internal and moving impt fns to Application

* Modernized Terminal resize events

* Modernized Terminal resize events

* Removed un used property

* for _isWindowsTerminal devenv->wininit; not sure what changed

* Modernized mouse/keyboard events (Action->EventHandler)

* Updated OnMouseEvent API docs

* Using WT_SESSION to detect WT

* removes hacky GetParentProcess

* Updates to fix #2634 (clear last line)

* removes hacky GetParentProcess2

* Addressed mac resize issue

* Addressed mac resize issue

* Removes ConsoleDriver.PrepareToRun, has Init return MainLoop

* Removes unneeded Attribute methods

* Removed GetProcesssName

* Removed GetProcesssName

* Refactored KeyEvent and KeyEventEventArgs into a single class

* Revert "Refactored KeyEvent and KeyEventEventArgs into a single class"

This reverts commit 88a00658db.

* Fixed key repeat issue; reverted stupidity on 1049/1047 confusion

* Updated CSI API Docs

* merge

* Rearranged Event.cs to Keyboard.cs and Mouse.cs

* Renamed KeyEventEventArgs KeyEventArgs

* temp renamed KeyEvent OldKeyEvent

* Merged KeyEvent into KeyEventArgs

* Renamed Application.ProcessKey members

* Renamed Application.ProcessKey members

* Renamed Application.ProcessKey members

* Added Responder.KeyPressed

* Removed unused references

* Fixed arg naming

* InvokeKeybindings->InvokeKeyBindings

* InvokeKeybindings->InvokeKeyBindings

* Fixed unit tests fail

* More progress on refactoring key input; still broken and probably wrong

* Moved OnKeyPressed out of Responder and made ProcessKeyPrssed non-virtual

* Updated API docs

* Moved key handling from Responder to View

* Updated API docs

* Updated HotKey API docs

* Updated shortcut API docs

* Fixed responder unit tests

* Removed Shortcut from View as it is not used

* Removed unneeded OnHotKey override from Button

* Fixed BackTab logic

* Button now uses Key Bindings exclusively

* Button now uses Key Bindings exclusively

* Updated keyboard.md docs

* Fixed unit tests to account for Toplevel handling default button

* Added View.InvokeCommand API

* Modernized RadioGroup

* Removed ColdKey

* Modernized (partially) StatusBar

* Worked around FileDialog issue with Ctrl-F

* Fixed driver unit test; view must be focused to reciev key pressed

* Application code cleanup

* Start on updaing menu

* Menu now mostly works

* Menu Select refinement

* Fixed known menu bugs!

* Enabled HotKey to cause focus- experimental

* Fixes #3022 & adds unit test to prove it

* Actually Fixes #3022 & adds unit test to prove it

* Working through hotkey issues

* Misc fixes

* removed hot/cold key stuff from Keys scenario

* Fixed scenarios

* Simplified shortcut string handling

* Modernized Checkbox

* Modernized TileView

* Updated API docs

* Updated API docs

* attempting to publish v2 docs

* Revert "attempting to publish v2 docs"

This reverts commit 59dcec111b.

* Playing with api docs

* Removed Key.BackTab

* Removed Caps/Scroll/Numlock

* Partial removal of keymodifiers - unit tests pass

* Partial removal of keymodifiers - broke netdriver somewhere

* WindowsDriver & added KeyEventArgsTests

* Fixing menu shortcut/hotkeys - broke Menu.cs into separate files

* Fixed MenuBar!

* Finished modernizing Menu/MenuBar

* Removed Key.a-z. Broke lots of stuff

* checkout@v4

* progress on key mapping and formatting

* VK tests are still failing

* Fixed some unit tests

* Added Hotkey and Keybinding unit tests

* fixed unit test

* All unit tests pass again...

* Fixed broken unit tests

* KeyEventArgs.KeyValue -> AsRune

* Fixed bugs. Still some broken

* Added KeyEventArgs.IsAlpha. Added KeyEventArgs.cast ops. Fixed bugs. Unit tests pass

* Fixed WindowsDriver

* Oops.

* Refactoring based on bdisp's help. Not complete!

* removed calling into subviews from OnKeyBindings

* removed calling into subviews from OnKeyBindings

* Improved View KeyEvent unit tests

* More hotkey unit tests

* BIg change - Got rid of KeyPress w/in Application/Drivers

* Unit tests now pass again

* Refreshed API docs

* Better HotKey logic. More progress. Getting close.

* Fixed handling of shifted chars like ö

* Minor code cleanup

* Minor code cleanup2

* Why is build Action failing?

* Why is build Action failing??

* upgraded to .net8 to try to fix weird CI/CD build errors

* upgraded to .net8 to try to fix weird CI/CD build errors2

* Disabling TextViewTests to diagnose build errors

* reenable TextViewTests to diagnose build errors

* Arrrrrrg

* Merged v2_develop

* Fixed uppercase accented keys in WindowsDriver

* Fixed key binding api docs

* Experimental impl of CommandScope.SubViews for MenuBar

* Removed dead code from application.cs

* Removed dead code from application.cs

* Removed dead code from ConsoleDriver.cs

* Cleaned up some key binding stuff

* Disabled Alt to activate menu for now

* Updated label commands

* Fixed menu bugs. Upgraded menu unit tests

* Fixed unit tests

* Working on NetDriver

* fixed netdriver

* Fixed issues called out by @bdisp CR

* fixed CursesDriver

* added todo to netdriver

* Cherry picked treeview test fix 1b415e5

* Fix NetDriver.

* CommandScope->KeyBindingScope

* Address some tznind feedback

* Refactored KeyBindings big time!

* Added key consts to KeyEventArgs and renamed Key to ConsoleDriverKey

* Fixed some API docs

* Moved ConsoleDriverKey to ConsoleDriver.cs

* Renamed Key->ConsoleDriverKey

* Renamed Key->ConsoleDriverKey

* Renamed Key->ConsoleDriverKey

* renamed file I forgot to rename before

* Updated name and API docs of KeyEventArgs.isAlpha

* Fixed issues with OnKeyUp not doing the right thing.

* Fixed MainLoop.Running never being used

* Fixed MainLoop.Running never being used - unit tests

* Claned up BUGBUG comments

* Disabled a unit test to see why ci/cd tests are failing

* Removed defunct commented code

* Removed more defunct commented code

* Re-eanbled unit test; jsut removing one test case...

* Disabled more...

* Renambed Global->Applicaton and updated scope API docs

* Disabled more unit tests...

* Removed dead code

* Disabled more unit tests...2

* Disabled more unit tests...3

* Renambed Global->Applicaton and updated scope API docs 2

* Added more KeyBinding scope tests

* Added more KeyBinding scope tests2

* ConsoleDriverKey too long. Key too ambiguous. Settled on KeyCode. (Partialy because eventually I want to intro a class named Key).

* KeyEventArgs improvements. cast to Rune must be explicit as it's lossy

* Fixed warnings

* Renamed KeyEventArgs to Key... progress on fixing broken stuff that resulted

* Fix ConsoleKeyMapping bugs.

* Fix NetDriver issue from converting a lower case to a upper case.

* Started migration to Key from KeyCode - e.g. made HotKeys all consistent.

* Fixed build warnings

* Added key defns to Key

* KeyBindings now uses Key vs. KeyCode

* Verified by tweaking UICatalog

* Fixed treeview test ... again

* Renamed ProcessKeyDown/Up to NewKeyDown/Up and OnKeyPressed to OnProcessKeyDown to make things more clear

* Added test AllViews_KeyDown_All_EventsFire unit tests and fixed a few Views that were wrong

* fixed stupid KeyUp event bug

* If key not handled, return false for datefield

* dotnet test --no-restore --verbosity diag

* dotnet test --blame

* run tests on windows

* Fix TestVKPacket unit test and move it to ConsoleKeyMappingTests.cs file.

* Remove unnecessary commented code.

* Tweaked unit tests and removed Key.BareKey

* Fixed little details and updated api docs

* updated api docs

* AddKeyBindingsForHotKey: KeyCode->Key

* Cleaned up more old KeyCode usages. Added TODOs

---------

Co-authored-by: BDisp <bd.bdisp@gmail.com>
2023-12-16 12:04:23 -07:00

1108 lines
37 KiB
C#

using System;
using System.Collections.Generic;
using System.Data;
using Terminal.Gui;
using System.Linq;
using System.Globalization;
using System.Text;
using System.IO;
namespace UICatalog.Scenarios {
[ScenarioMetadata (Name: "TableEditor", Description: "Implements data table editor using the TableView control.")]
[ScenarioCategory ("TableView")]
[ScenarioCategory ("Controls")]
[ScenarioCategory ("Dialogs")]
[ScenarioCategory ("Text and Formatting")]
[ScenarioCategory ("Top Level Windows")]
public class TableEditor : Scenario {
TableView tableView;
DataTable currentTable;
private MenuItem _miShowHeaders;
private MenuItem _miAlwaysShowHeaders;
private MenuItem _miHeaderOverline;
private MenuItem _miHeaderMidline;
private MenuItem _miHeaderUnderline;
private MenuItem _miShowHorizontalScrollIndicators;
private MenuItem _miCellLines;
private MenuItem _miFullRowSelect;
private MenuItem _miExpandLastColumn;
private MenuItem _miAlwaysUseNormalColorForVerticalCellLines;
private MenuItem _miSmoothScrolling;
private MenuItem _miAlternatingColors;
private MenuItem _miCursor;
private MenuItem _miBottomline;
private MenuItem _miCheckboxes;
private MenuItem _miRadioboxes;
private List<IDisposable> toDispose = new List<IDisposable> ();
ColorScheme redColorScheme;
ColorScheme redColorSchemeAlt;
ColorScheme alternatingColorScheme;
HashSet<FileSystemInfo> _checkedFileSystemInfos = new HashSet<FileSystemInfo> ();
public override void Setup ()
{
Win.Title = this.GetName ();
Win.Y = 1; // menu
Win.Height = Dim.Fill (1); // status bar
Application.Top.LayoutSubviews ();
this.tableView = new TableView () {
X = 0,
Y = 0,
Width = Dim.Fill (),
Height = Dim.Fill (1),
};
var menu = new MenuBar (new MenuBarItem [] {
new MenuBarItem ("_File", new MenuItem [] {
new MenuItem ("_OpenBigExample", "", () => OpenExample(true)),
new MenuItem ("_OpenSmallExample", "", () => OpenExample(false)),
new MenuItem ("OpenCharacter_Map","",()=>OpenUnicodeMap()),
new MenuItem ("OpenTreeExample","",()=>OpenTreeExample()),
new MenuItem ("_CloseExample", "", () => CloseExample()),
new MenuItem ("_Quit", "", () => Quit()),
}),
new MenuBarItem ("_View", new MenuItem [] {
_miShowHeaders = new MenuItem ("_ShowHeaders", "", () => ToggleShowHeaders()){Checked = tableView.Style.ShowHeaders, CheckType = MenuItemCheckStyle.Checked },
_miAlwaysShowHeaders = new MenuItem ("_AlwaysShowHeaders", "", () => ToggleAlwaysShowHeaders()){Checked = tableView.Style.AlwaysShowHeaders, CheckType = MenuItemCheckStyle.Checked },
_miHeaderOverline = new MenuItem ("_HeaderOverLine", "", () => ToggleOverline()){Checked = tableView.Style.ShowHorizontalHeaderOverline, CheckType = MenuItemCheckStyle.Checked },
_miHeaderMidline = new MenuItem ("_HeaderMidLine", "", () => ToggleHeaderMidline()){Checked = tableView.Style.ShowVerticalHeaderLines, CheckType = MenuItemCheckStyle.Checked },
_miHeaderUnderline = new MenuItem ("_HeaderUnderLine", "", () => ToggleUnderline()){Checked = tableView.Style.ShowHorizontalHeaderUnderline, CheckType = MenuItemCheckStyle.Checked },
_miBottomline = new MenuItem ("_BottomLine", "", () => ToggleBottomline()){Checked = tableView.Style.ShowHorizontalBottomline, CheckType = MenuItemCheckStyle.Checked },
_miShowHorizontalScrollIndicators = new MenuItem ("_HorizontalScrollIndicators", "", () => ToggleHorizontalScrollIndicators()){Checked = tableView.Style.ShowHorizontalScrollIndicators, CheckType = MenuItemCheckStyle.Checked },
_miFullRowSelect =new MenuItem ("_FullRowSelect", "", () => ToggleFullRowSelect()){Checked = tableView.FullRowSelect, CheckType = MenuItemCheckStyle.Checked },
_miCellLines =new MenuItem ("_CellLines", "", () => ToggleCellLines()){Checked = tableView.Style.ShowVerticalCellLines, CheckType = MenuItemCheckStyle.Checked },
_miExpandLastColumn = new MenuItem ("_ExpandLastColumn", "", () => ToggleExpandLastColumn()){Checked = tableView.Style.ExpandLastColumn, CheckType = MenuItemCheckStyle.Checked },
_miAlwaysUseNormalColorForVerticalCellLines = new MenuItem ("_AlwaysUseNormalColorForVerticalCellLines", "", () => ToggleAlwaysUseNormalColorForVerticalCellLines()){Checked = tableView.Style.AlwaysUseNormalColorForVerticalCellLines, CheckType = MenuItemCheckStyle.Checked },
_miSmoothScrolling = new MenuItem ("_SmoothHorizontalScrolling", "", () => ToggleSmoothScrolling()){Checked = tableView.Style.SmoothHorizontalScrolling, CheckType = MenuItemCheckStyle.Checked },
new MenuItem ("_AllLines", "", () => ToggleAllCellLines()),
new MenuItem ("_NoLines", "", () => ToggleNoCellLines()),
_miCheckboxes = new MenuItem ("_Checkboxes", "", () => ToggleCheckboxes(false)){Checked = false, CheckType = MenuItemCheckStyle.Checked },
_miRadioboxes = new MenuItem ("_Radioboxes", "", () => ToggleCheckboxes(true)){Checked = false, CheckType = MenuItemCheckStyle.Checked },
_miAlternatingColors = new MenuItem ("Alternating Colors", "", () => ToggleAlternatingColors()){CheckType = MenuItemCheckStyle.Checked},
_miCursor = new MenuItem ("Invert Selected Cell First Character", "", () => ToggleInvertSelectedCellFirstCharacter()){Checked = tableView.Style.InvertSelectedCellFirstCharacter,CheckType = MenuItemCheckStyle.Checked},
new MenuItem ("_ClearColumnStyles", "", () => ClearColumnStyles()),
new MenuItem ("Sho_w All Columns", "", ()=>ShowAllColumns())
}),
new MenuBarItem ("_Column", new MenuItem [] {
new MenuItem ("_Set Max Width", "", SetMaxWidth),
new MenuItem ("_Set Min Width", "", SetMinWidth),
new MenuItem ("_Set MinAcceptableWidth", "",SetMinAcceptableWidth),
new MenuItem ("_Set All MinAcceptableWidth=1", "",SetMinAcceptableWidthToOne),
}),
});
Application.Top.Add (menu);
var statusBar = new StatusBar (new StatusItem [] {
new StatusItem(KeyCode.F2, "~F2~ OpenExample", () => OpenExample(true)),
new StatusItem(KeyCode.F3, "~F3~ CloseExample", () => CloseExample()),
new StatusItem(KeyCode.F4, "~F4~ OpenSimple", () => OpenSimple(true)),
new StatusItem(Application.QuitKey, $"{Application.QuitKey} to Quit", () => Quit()),
});
Application.Top.Add (statusBar);
Win.Add (tableView);
var selectedCellLabel = new Label () {
X = 0,
Y = Pos.Bottom (tableView),
Text = "0,0",
Width = Dim.Fill (),
TextAlignment = TextAlignment.Right
};
Win.Add (selectedCellLabel);
tableView.SelectedCellChanged += (s, e) => { selectedCellLabel.Text = $"{tableView.SelectedRow},{tableView.SelectedColumn}"; };
tableView.CellActivated += EditCurrentCell;
tableView.KeyDown += TableViewKeyPress;
SetupScrollBar ();
redColorScheme = new ColorScheme () {
Disabled = Win.ColorScheme.Disabled,
HotFocus = Win.ColorScheme.HotFocus,
Focus = Win.ColorScheme.Focus,
Normal = new Attribute (Color.Red, Win.ColorScheme.Normal.Background)
};
alternatingColorScheme = new ColorScheme () {
Disabled = Win.ColorScheme.Disabled,
HotFocus = Win.ColorScheme.HotFocus,
Focus = Win.ColorScheme.Focus,
Normal = new Attribute (Color.White, Color.BrightBlue)
};
redColorSchemeAlt = new ColorScheme () {
Disabled = Win.ColorScheme.Disabled,
HotFocus = Win.ColorScheme.HotFocus,
Focus = Win.ColorScheme.Focus,
Normal = new Attribute (Color.Red, Color.BrightBlue)
};
// if user clicks the mouse in TableView
tableView.MouseClick += (s, e) => {
if(currentTable == null) {
return;
}
tableView.ScreenToCell (e.MouseEvent.X, e.MouseEvent.Y, out int? clickedCol);
if (clickedCol != null) {
if (e.MouseEvent.Flags.HasFlag (MouseFlags.Button1Clicked)) {
// left click in a header
SortColumn (clickedCol.Value);
} else if (e.MouseEvent.Flags.HasFlag (MouseFlags.Button3Clicked)) {
// right click in a header
ShowHeaderContextMenu (clickedCol.Value, e);
}
}
};
tableView.KeyBindings.Add (KeyCode.Space, Command.ToggleChecked);
}
private void ShowAllColumns ()
{
foreach (var colStyle in tableView.Style.ColumnStyles) {
colStyle.Value.Visible = true;
}
tableView.Update ();
}
private void SortColumn (int clickedCol)
{
var sort = GetProposedNewSortOrder (clickedCol, out var isAsc);
// don't try to sort on the toggled column
if (HasCheckboxes () && clickedCol == 0) {
return;
}
SortColumn (clickedCol, sort, isAsc);
}
private void SortColumn (int clickedCol, string sort, bool isAsc)
{
if(currentTable == null) {
return;
}
// set a sort order
currentTable.DefaultView.Sort = sort;
// copy the rows from the view
var sortedCopy = currentTable.DefaultView.ToTable ();
currentTable.Rows.Clear ();
foreach (DataRow r in sortedCopy.Rows) {
currentTable.ImportRow (r);
}
foreach (DataColumn col in currentTable.Columns) {
// remove any lingering sort indicator
col.ColumnName = TrimArrows (col.ColumnName);
// add a new one if this the one that is being sorted
if (col.Ordinal == clickedCol) {
col.ColumnName += isAsc ? CM.Glyphs.UpArrow : CM.Glyphs.DownArrow;
}
}
tableView.Update ();
}
private string TrimArrows (string columnName)
{
return columnName.TrimEnd ((char)CM.Glyphs.UpArrow.Value, (char)CM.Glyphs.DownArrow.Value);
}
private string StripArrows (string columnName)
{
return columnName.Replace ($"{CM.Glyphs.DownArrow}", "").Replace ($"{CM.Glyphs.UpArrow}", "");
}
private string GetProposedNewSortOrder (int clickedCol, out bool isAsc)
{
// work out new sort order
var sort = currentTable.DefaultView.Sort;
var colName = tableView.Table.ColumnNames [clickedCol];
if (sort?.EndsWith ("ASC") ?? false) {
sort = $"{colName} DESC";
isAsc = false;
} else {
sort = $"{colName} ASC";
isAsc = true;
}
return sort;
}
private void ShowHeaderContextMenu (int clickedCol, MouseEventEventArgs e)
{
if (HasCheckboxes () && clickedCol == 0) {
return;
}
var sort = GetProposedNewSortOrder (clickedCol, out var isAsc);
var colName = tableView.Table.ColumnNames [clickedCol];
var contextMenu = new ContextMenu (e.MouseEvent.X + 1, e.MouseEvent.Y + 1,
new MenuBarItem (new MenuItem [] {
new MenuItem ($"Hide {TrimArrows(colName)}", "", () => HideColumn(clickedCol)),
new MenuItem ($"Sort {StripArrows(sort)}","",()=>SortColumn(clickedCol,sort,isAsc)),
})
);
contextMenu.Show ();
}
private void HideColumn (int clickedCol)
{
var style = tableView.Style.GetOrCreateColumnStyle (clickedCol);
style.Visible = false;
tableView.Update ();
}
private int? GetColumn ()
{
if (tableView.Table == null)
return null;
if (tableView.SelectedColumn < 0 || tableView.SelectedColumn > tableView.Table.Columns)
return null;
return tableView.SelectedColumn;
}
private void SetMinAcceptableWidthToOne ()
{
for(int i =0;i<tableView.Table.Columns;i++) {
var style = tableView.Style.GetOrCreateColumnStyle (i);
style.MinAcceptableWidth = 1;
}
}
private void SetMinAcceptableWidth ()
{
var col = GetColumn ();
RunColumnWidthDialog (col, "MinAcceptableWidth", (s, v) => s.MinAcceptableWidth = v, (s) => s.MinAcceptableWidth);
}
private void SetMinWidth ()
{
var col = GetColumn ();
RunColumnWidthDialog (col, "MinWidth", (s, v) => s.MinWidth = v, (s) => s.MinWidth);
}
private void SetMaxWidth ()
{
var col = GetColumn ();
RunColumnWidthDialog (col, "MaxWidth", (s, v) => s.MaxWidth = v, (s) => s.MaxWidth);
}
private void RunColumnWidthDialog (int? col, string prompt, Action<ColumnStyle, int> setter, Func<ColumnStyle, int> getter)
{
if (col == null) {
return;
}
var accepted = false;
var ok = new Button ("Ok", is_default: true);
ok.Clicked += (s, e) => { accepted = true; Application.RequestStop (); };
var cancel = new Button ("Cancel");
cancel.Clicked += (s, e) => { Application.RequestStop (); };
var d = new Dialog (ok, cancel) { Title = prompt };
var style = tableView.Style.GetOrCreateColumnStyle (col.Value);
var lbl = new Label () {
X = 0,
Y = 1,
Text = tableView.Table.ColumnNames [col.Value]
};
var tf = new TextField () {
Text = getter (style).ToString (),
X = 0,
Y = 2,
Width = Dim.Fill ()
};
d.Add (lbl, tf);
tf.SetFocus ();
Application.Run (d);
if (accepted) {
try {
setter (style, int.Parse (tf.Text));
} catch (Exception ex) {
MessageBox.ErrorQuery (60, 20, "Failed to set", ex.Message, "Ok");
}
tableView.Update ();
}
}
private void SetupScrollBar ()
{
var scrollBar = new ScrollBarView (tableView, true);
scrollBar.ChangedPosition += (s, e) => {
tableView.RowOffset = scrollBar.Position;
if (tableView.RowOffset != scrollBar.Position) {
scrollBar.Position = tableView.RowOffset;
}
tableView.SetNeedsDisplay ();
};
/*
scrollBar.OtherScrollBarView.ChangedPosition += (s,e) => {
tableView.LeftItem = scrollBar.OtherScrollBarView.Position;
if (tableView.LeftItem != scrollBar.OtherScrollBarView.Position) {
scrollBar.OtherScrollBarView.Position = tableView.LeftItem;
}
tableView.SetNeedsDisplay ();
};*/
tableView.DrawContent += (s, e) => {
scrollBar.Size = tableView.Table?.Rows ?? 0;
scrollBar.Position = tableView.RowOffset;
//scrollBar.OtherScrollBarView.Size = tableView.Maxlength - 1;
//scrollBar.OtherScrollBarView.Position = tableView.LeftItem;
scrollBar.Refresh ();
};
}
private void TableViewKeyPress (object sender, Key e)
{
if(currentTable == null) {
return;
}
if (e.KeyCode == KeyCode.DeleteChar) {
if (tableView.FullRowSelect) {
// Delete button deletes all rows when in full row mode
foreach (int toRemove in tableView.GetAllSelectedCells ().Select (p => p.Y).Distinct ().OrderByDescending (i => i))
currentTable.Rows.RemoveAt (toRemove);
} else {
// otherwise set all selected cells to null
foreach (var pt in tableView.GetAllSelectedCells ()) {
currentTable.Rows [pt.Y] [pt.X] = DBNull.Value;
}
}
tableView.Update ();
e.Handled = true;
}
}
private void ClearColumnStyles ()
{
tableView.Style.ColumnStyles.Clear ();
tableView.Update ();
}
private void ToggleShowHeaders ()
{
_miShowHeaders.Checked = !_miShowHeaders.Checked;
tableView.Style.ShowHeaders = (bool)_miShowHeaders.Checked;
tableView.Update ();
}
private void ToggleAlwaysShowHeaders ()
{
_miAlwaysShowHeaders.Checked = !_miAlwaysShowHeaders.Checked;
tableView.Style.AlwaysShowHeaders = (bool)_miAlwaysShowHeaders.Checked;
tableView.Update ();
}
private void ToggleOverline ()
{
_miHeaderOverline.Checked = !_miHeaderOverline.Checked;
tableView.Style.ShowHorizontalHeaderOverline = (bool)_miHeaderOverline.Checked;
tableView.Update ();
}
private void ToggleHeaderMidline ()
{
_miHeaderMidline.Checked = !_miHeaderMidline.Checked;
tableView.Style.ShowVerticalHeaderLines = (bool)_miHeaderMidline.Checked;
tableView.Update ();
}
private void ToggleUnderline ()
{
_miHeaderUnderline.Checked = !_miHeaderUnderline.Checked;
tableView.Style.ShowHorizontalHeaderUnderline = (bool)_miHeaderUnderline.Checked;
tableView.Update ();
}
private void ToggleBottomline ()
{
_miBottomline.Checked = !_miBottomline.Checked;
tableView.Style.ShowHorizontalBottomline = (bool)_miBottomline.Checked;
tableView.Update ();
}
private void ToggleHorizontalScrollIndicators ()
{
_miShowHorizontalScrollIndicators.Checked = !_miShowHorizontalScrollIndicators.Checked;
tableView.Style.ShowHorizontalScrollIndicators = (bool)_miShowHorizontalScrollIndicators.Checked;
tableView.Update ();
}
private void ToggleFullRowSelect ()
{
_miFullRowSelect.Checked = !_miFullRowSelect.Checked;
tableView.FullRowSelect = (bool)_miFullRowSelect.Checked;
tableView.Update ();
}
private void ToggleExpandLastColumn ()
{
_miExpandLastColumn.Checked = !_miExpandLastColumn.Checked;
tableView.Style.ExpandLastColumn = (bool)_miExpandLastColumn.Checked;
tableView.Update ();
}
private void ToggleCheckboxes (bool radio)
{
if (tableView.Table is CheckBoxTableSourceWrapperBase wrapper) {
// unwrap it to remove check boxes
tableView.Table = wrapper.Wrapping;
_miCheckboxes.Checked = false;
_miRadioboxes.Checked = false;
// if toggling off checkboxes/radio
if (wrapper.UseRadioButtons == radio) {
return;
}
}
ITableSource source;
// Either toggling on checkboxes/radio or switching from radio to checkboxes (or vice versa)
if (tableView.Table is TreeTableSource<FileSystemInfo> treeSource) {
source = new CheckBoxTableSourceWrapperByObject<FileSystemInfo> (tableView, treeSource,
this._checkedFileSystemInfos.Contains,
this.CheckOrUncheckFile
) {
UseRadioButtons = radio
};
} else {
source = new CheckBoxTableSourceWrapperByIndex (tableView, tableView.Table) {
UseRadioButtons = radio
};
}
tableView.Table = source;
if (radio) {
_miRadioboxes.Checked = true;
_miCheckboxes.Checked = false;
} else {
_miRadioboxes.Checked = false;
_miCheckboxes.Checked = true;
}
}
private void CheckOrUncheckFile (FileSystemInfo info, bool check)
{
if (check) {
_checkedFileSystemInfos.Add (info);
} else {
_checkedFileSystemInfos.Remove (info);
}
}
private void ToggleAlwaysUseNormalColorForVerticalCellLines ()
{
_miAlwaysUseNormalColorForVerticalCellLines.Checked = !_miAlwaysUseNormalColorForVerticalCellLines.Checked;
tableView.Style.AlwaysUseNormalColorForVerticalCellLines = (bool)_miAlwaysUseNormalColorForVerticalCellLines.Checked;
tableView.Update ();
}
private void ToggleSmoothScrolling ()
{
_miSmoothScrolling.Checked = !_miSmoothScrolling.Checked;
tableView.Style.SmoothHorizontalScrolling = (bool)_miSmoothScrolling.Checked;
tableView.Update ();
}
private void ToggleCellLines ()
{
_miCellLines.Checked = !_miCellLines.Checked;
tableView.Style.ShowVerticalCellLines = (bool)_miCellLines.Checked;
tableView.Update ();
}
private void ToggleAllCellLines ()
{
tableView.Style.ShowHorizontalHeaderOverline = true;
tableView.Style.ShowVerticalHeaderLines = true;
tableView.Style.ShowHorizontalHeaderUnderline = true;
tableView.Style.ShowVerticalCellLines = true;
_miHeaderOverline.Checked = true;
_miHeaderMidline.Checked = true;
_miHeaderUnderline.Checked = true;
_miCellLines.Checked = true;
tableView.Update ();
}
private void ToggleNoCellLines ()
{
tableView.Style.ShowHorizontalHeaderOverline = false;
tableView.Style.ShowVerticalHeaderLines = false;
tableView.Style.ShowHorizontalHeaderUnderline = false;
tableView.Style.ShowVerticalCellLines = false;
_miHeaderOverline.Checked = false;
_miHeaderMidline.Checked = false;
_miHeaderUnderline.Checked = false;
_miCellLines.Checked = false;
tableView.Update ();
}
private void ToggleAlternatingColors ()
{
//toggle menu item
_miAlternatingColors.Checked = !_miAlternatingColors.Checked;
if (_miAlternatingColors.Checked == true) {
tableView.Style.RowColorGetter = (a) => { return a.RowIndex % 2 == 0 ? alternatingColorScheme : null; };
} else {
tableView.Style.RowColorGetter = null;
}
tableView.SetNeedsDisplay ();
}
private void ToggleInvertSelectedCellFirstCharacter ()
{
//toggle menu item
_miCursor.Checked = !_miCursor.Checked;
tableView.Style.InvertSelectedCellFirstCharacter = (bool)_miCursor.Checked;
tableView.SetNeedsDisplay ();
}
private void CloseExample ()
{
tableView.Table = null;
}
private void Quit ()
{
Application.RequestStop ();
}
private void OpenExample (bool big)
{
SetTable (BuildDemoDataTable (big ? 30 : 5, big ? 1000 : 5));
SetDemoTableStyles ();
}
private void SetTable (DataTable dataTable)
{
tableView.Table = new DataTableSource (currentTable = dataTable);
}
private void OpenUnicodeMap ()
{
SetTable (BuildUnicodeMap ());
tableView.Update ();
}
private IEnumerable<FileSystemInfo> GetChildren (FileSystemInfo arg)
{
try {
return arg is DirectoryInfo d ?
d.GetFileSystemInfos () :
Enumerable.Empty<FileSystemInfo> ();
} catch (Exception) {
// Permission denied etc
return Enumerable.Empty<FileSystemInfo> ();
}
}
protected override void Dispose (bool disposing)
{
base.Dispose (disposing);
foreach (var d in toDispose) {
d.Dispose ();
}
}
private void OpenTreeExample ()
{
tableView.Style.ColumnStyles.Clear ();
var tree = new TreeView<FileSystemInfo> {
AspectGetter = (f) => f.Name,
TreeBuilder = new DelegateTreeBuilder<FileSystemInfo> (GetChildren)
};
var source = new TreeTableSource<FileSystemInfo> (tableView, "Name", tree, new (){
{"Extension", f=>f.Extension},
{"CreationTime", f=>f.CreationTime},
{"FileSize", GetHumanReadableFileSize}
});
var seen = new HashSet<string> ();
try {
foreach (var path in Environment.GetLogicalDrives ()) {
tree.AddObject (new DirectoryInfo (path));
}
} catch (Exception e) {
MessageBox.ErrorQuery ("Could not find local drives", e.Message, "Ok");
}
tableView.Table = source;
toDispose.Add (tree);
}
private string GetHumanReadableFileSize (FileSystemInfo fsi)
{
if (fsi is not FileInfo fi) {
return null;
}
long value = fi.Length;
var culture = CultureInfo.CurrentUICulture;
return GetHumanReadableFileSize (value, culture);
}
private string GetHumanReadableFileSize (long value, CultureInfo culture)
{
const long ByteConversion = 1024;
string [] SizeSuffixes = { "bytes", "KB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB" };
if (value < 0) {
return "-" + GetHumanReadableFileSize (-value, culture);
}
if (value == 0) {
return "0.0 bytes";
}
int mag = (int)Math.Log (value, ByteConversion);
double adjustedSize = value / Math.Pow (1000, mag);
return string.Format (culture.NumberFormat, "{0:n2} {1}", adjustedSize, SizeSuffixes [mag]);
}
private DataTable BuildUnicodeMap ()
{
var dt = new DataTable ();
// add cols called 0 to 9
for (int i = 0; i < 10; i++) {
var col = dt.Columns.Add (i.ToString (), typeof (uint));
var style = tableView.Style.GetOrCreateColumnStyle (col.Ordinal);
style.RepresentationGetter = (o) => new Rune ((uint)o).ToString ();
}
// add cols called a to z
for (int i = 'a'; i < 'a' + 26; i++) {
var col = dt.Columns.Add (((char)i).ToString (), typeof (uint));
var style = tableView.Style.GetOrCreateColumnStyle (col.Ordinal);
style.RepresentationGetter = (o) => new Rune ((uint)o).ToString ();
}
// now add table contents
List<uint> runes = new List<uint> ();
foreach (var range in Ranges) {
for (uint i = range.Start; i <= range.End; i++) {
runes.Add (i);
}
}
DataRow dr = null;
for (int i = 0; i < runes.Count; i++) {
if (dr == null || i % dt.Columns.Count == 0) {
dr = dt.Rows.Add ();
}
dr [i % dt.Columns.Count] = runes [i].ToString ();
}
return dt;
}
class UnicodeRange {
public uint Start;
public uint End;
public string Category;
public UnicodeRange (uint start, uint end, string category)
{
this.Start = start;
this.End = end;
this.Category = category;
}
}
List<UnicodeRange> Ranges = new List<UnicodeRange> {
new UnicodeRange (0x0000, 0x001F, "ASCII Control Characters"),
new UnicodeRange (0x0080, 0x009F, "C0 Control Characters"),
new UnicodeRange(0x1100, 0x11ff,"Hangul Jamo"), // This is where wide chars tend to start
new UnicodeRange(0x20A0, 0x20CF,"Currency Symbols"),
new UnicodeRange(0x2100, 0x214F,"Letterlike Symbols"),
new UnicodeRange(0x2190, 0x21ff,"Arrows" ),
new UnicodeRange(0x2200, 0x22ff,"Mathematical symbols"),
new UnicodeRange(0x2300, 0x23ff,"Miscellaneous Technical"),
new UnicodeRange(0x2500, 0x25ff,"Box Drawing & Geometric Shapes"),
new UnicodeRange(0x2600, 0x26ff,"Miscellaneous Symbols"),
new UnicodeRange(0x2700, 0x27ff,"Dingbats"),
new UnicodeRange(0x2800, 0x28ff,"Braille"),
new UnicodeRange(0x2b00, 0x2bff,"Miscellaneous Symbols and Arrows"),
new UnicodeRange(0xFB00, 0xFb4f,"Alphabetic Presentation Forms"),
new UnicodeRange(0x12400, 0x1240f,"Cuneiform Numbers and Punctuation"),
new UnicodeRange((uint)(CharMap.MaxCodePoint - 16), (uint)CharMap.MaxCodePoint,"End"),
new UnicodeRange (0x0020 ,0x007F ,"Basic Latin"),
new UnicodeRange (0x00A0 ,0x00FF ,"Latin-1 Supplement"),
new UnicodeRange (0x0100 ,0x017F ,"Latin Extended-A"),
new UnicodeRange (0x0180 ,0x024F ,"Latin Extended-B"),
new UnicodeRange (0x0250 ,0x02AF ,"IPA Extensions"),
new UnicodeRange (0x02B0 ,0x02FF ,"Spacing Modifier Letters"),
new UnicodeRange (0x0300 ,0x036F ,"Combining Diacritical Marks"),
new UnicodeRange (0x0370 ,0x03FF ,"Greek and Coptic"),
new UnicodeRange (0x0400 ,0x04FF ,"Cyrillic"),
new UnicodeRange (0x0500 ,0x052F ,"Cyrillic Supplementary"),
new UnicodeRange (0x0530 ,0x058F ,"Armenian"),
new UnicodeRange (0x0590 ,0x05FF ,"Hebrew"),
new UnicodeRange (0x0600 ,0x06FF ,"Arabic"),
new UnicodeRange (0x0700 ,0x074F ,"Syriac"),
new UnicodeRange (0x0780 ,0x07BF ,"Thaana"),
new UnicodeRange (0x0900 ,0x097F ,"Devanagari"),
new UnicodeRange (0x0980 ,0x09FF ,"Bengali"),
new UnicodeRange (0x0A00 ,0x0A7F ,"Gurmukhi"),
new UnicodeRange (0x0A80 ,0x0AFF ,"Gujarati"),
new UnicodeRange (0x0B00 ,0x0B7F ,"Oriya"),
new UnicodeRange (0x0B80 ,0x0BFF ,"Tamil"),
new UnicodeRange (0x0C00 ,0x0C7F ,"Telugu"),
new UnicodeRange (0x0C80 ,0x0CFF ,"Kannada"),
new UnicodeRange (0x0D00 ,0x0D7F ,"Malayalam"),
new UnicodeRange (0x0D80 ,0x0DFF ,"Sinhala"),
new UnicodeRange (0x0E00 ,0x0E7F ,"Thai"),
new UnicodeRange (0x0E80 ,0x0EFF ,"Lao"),
new UnicodeRange (0x0F00 ,0x0FFF ,"Tibetan"),
new UnicodeRange (0x1000 ,0x109F ,"Myanmar"),
new UnicodeRange (0x10A0 ,0x10FF ,"Georgian"),
new UnicodeRange (0x1100 ,0x11FF ,"Hangul Jamo"),
new UnicodeRange (0x1200 ,0x137F ,"Ethiopic"),
new UnicodeRange (0x13A0 ,0x13FF ,"Cherokee"),
new UnicodeRange (0x1400 ,0x167F ,"Unified Canadian Aboriginal Syllabics"),
new UnicodeRange (0x1680 ,0x169F ,"Ogham"),
new UnicodeRange (0x16A0 ,0x16FF ,"Runic"),
new UnicodeRange (0x1700 ,0x171F ,"Tagalog"),
new UnicodeRange (0x1720 ,0x173F ,"Hanunoo"),
new UnicodeRange (0x1740 ,0x175F ,"Buhid"),
new UnicodeRange (0x1760 ,0x177F ,"Tagbanwa"),
new UnicodeRange (0x1780 ,0x17FF ,"Khmer"),
new UnicodeRange (0x1800 ,0x18AF ,"Mongolian"),
new UnicodeRange (0x1900 ,0x194F ,"Limbu"),
new UnicodeRange (0x1950 ,0x197F ,"Tai Le"),
new UnicodeRange (0x19E0 ,0x19FF ,"Khmer Symbols"),
new UnicodeRange (0x1D00 ,0x1D7F ,"Phonetic Extensions"),
new UnicodeRange (0x1E00 ,0x1EFF ,"Latin Extended Additional"),
new UnicodeRange (0x1F00 ,0x1FFF ,"Greek Extended"),
new UnicodeRange (0x2000 ,0x206F ,"General Punctuation"),
new UnicodeRange (0x2070 ,0x209F ,"Superscripts and Subscripts"),
new UnicodeRange (0x20A0 ,0x20CF ,"Currency Symbols"),
new UnicodeRange (0x20D0 ,0x20FF ,"Combining Diacritical Marks for Symbols"),
new UnicodeRange (0x2100 ,0x214F ,"Letterlike Symbols"),
new UnicodeRange (0x2150 ,0x218F ,"Number Forms"),
new UnicodeRange (0x2190 ,0x21FF ,"Arrows"),
new UnicodeRange (0x2200 ,0x22FF ,"Mathematical Operators"),
new UnicodeRange (0x2300 ,0x23FF ,"Miscellaneous Technical"),
new UnicodeRange (0x2400 ,0x243F ,"Control Pictures"),
new UnicodeRange (0x2440 ,0x245F ,"Optical Character Recognition"),
new UnicodeRange (0x2460 ,0x24FF ,"Enclosed Alphanumerics"),
new UnicodeRange (0x2500 ,0x257F ,"Box Drawing"),
new UnicodeRange (0x2580 ,0x259F ,"Block Elements"),
new UnicodeRange (0x25A0 ,0x25FF ,"Geometric Shapes"),
new UnicodeRange (0x2600 ,0x26FF ,"Miscellaneous Symbols"),
new UnicodeRange (0x2700 ,0x27BF ,"Dingbats"),
new UnicodeRange (0x27C0 ,0x27EF ,"Miscellaneous Mathematical Symbols-A"),
new UnicodeRange (0x27F0 ,0x27FF ,"Supplemental Arrows-A"),
new UnicodeRange (0x2800 ,0x28FF ,"Braille Patterns"),
new UnicodeRange (0x2900 ,0x297F ,"Supplemental Arrows-B"),
new UnicodeRange (0x2980 ,0x29FF ,"Miscellaneous Mathematical Symbols-B"),
new UnicodeRange (0x2A00 ,0x2AFF ,"Supplemental Mathematical Operators"),
new UnicodeRange (0x2B00 ,0x2BFF ,"Miscellaneous Symbols and Arrows"),
new UnicodeRange (0x2E80 ,0x2EFF ,"CJK Radicals Supplement"),
new UnicodeRange (0x2F00 ,0x2FDF ,"Kangxi Radicals"),
new UnicodeRange (0x2FF0 ,0x2FFF ,"Ideographic Description Characters"),
new UnicodeRange (0x3000 ,0x303F ,"CJK Symbols and Punctuation"),
new UnicodeRange (0x3040 ,0x309F ,"Hiragana"),
new UnicodeRange (0x30A0 ,0x30FF ,"Katakana"),
new UnicodeRange (0x3100 ,0x312F ,"Bopomofo"),
new UnicodeRange (0x3130 ,0x318F ,"Hangul Compatibility Jamo"),
new UnicodeRange (0x3190 ,0x319F ,"Kanbun"),
new UnicodeRange (0x31A0 ,0x31BF ,"Bopomofo Extended"),
new UnicodeRange (0x31F0 ,0x31FF ,"Katakana Phonetic Extensions"),
new UnicodeRange (0x3200 ,0x32FF ,"Enclosed CJK Letters and Months"),
new UnicodeRange (0x3300 ,0x33FF ,"CJK Compatibility"),
new UnicodeRange (0x3400 ,0x4DBF ,"CJK Unified Ideographs Extension A"),
new UnicodeRange (0x4DC0 ,0x4DFF ,"Yijing Hexagram Symbols"),
new UnicodeRange (0x4E00 ,0x9FFF ,"CJK Unified Ideographs"),
new UnicodeRange (0xA000 ,0xA48F ,"Yi Syllables"),
new UnicodeRange (0xA490 ,0xA4CF ,"Yi Radicals"),
new UnicodeRange (0xAC00 ,0xD7AF ,"Hangul Syllables"),
new UnicodeRange (0xD800 ,0xDB7F ,"High Surrogates"),
new UnicodeRange (0xDB80 ,0xDBFF ,"High Private Use Surrogates"),
new UnicodeRange (0xDC00 ,0xDFFF ,"Low Surrogates"),
new UnicodeRange (0xE000 ,0xF8FF ,"Private Use Area"),
new UnicodeRange (0xF900 ,0xFAFF ,"CJK Compatibility Ideographs"),
new UnicodeRange (0xFB00 ,0xFB4F ,"Alphabetic Presentation Forms"),
new UnicodeRange (0xFB50 ,0xFDFF ,"Arabic Presentation Forms-A"),
new UnicodeRange (0xFE00 ,0xFE0F ,"Variation Selectors"),
new UnicodeRange (0xFE20 ,0xFE2F ,"Combining Half Marks"),
new UnicodeRange (0xFE30 ,0xFE4F ,"CJK Compatibility Forms"),
new UnicodeRange (0xFE50 ,0xFE6F ,"Small Form Variants"),
new UnicodeRange (0xFE70 ,0xFEFF ,"Arabic Presentation Forms-B"),
new UnicodeRange (0xFF00 ,0xFFEF ,"Halfwidth and Fullwidth Forms"),
new UnicodeRange (0xFFF0 ,0xFFFF ,"Specials"),
new UnicodeRange (0x10000, 0x1007F ,"Linear B Syllabary"),
new UnicodeRange (0x10080, 0x100FF ,"Linear B Ideograms"),
new UnicodeRange (0x10100, 0x1013F ,"Aegean Numbers"),
new UnicodeRange (0x10300, 0x1032F ,"Old Italic"),
new UnicodeRange (0x10330, 0x1034F ,"Gothic"),
new UnicodeRange (0x10380, 0x1039F ,"Ugaritic"),
new UnicodeRange (0x10400, 0x1044F ,"Deseret"),
new UnicodeRange (0x10450, 0x1047F ,"Shavian"),
new UnicodeRange (0x10480, 0x104AF ,"Osmanya"),
new UnicodeRange (0x10800, 0x1083F ,"Cypriot Syllabary"),
new UnicodeRange (0x1D000, 0x1D0FF ,"Byzantine Musical Symbols"),
new UnicodeRange (0x1D100, 0x1D1FF ,"Musical Symbols"),
new UnicodeRange (0x1D300, 0x1D35F ,"Tai Xuan Jing Symbols"),
new UnicodeRange (0x1D400, 0x1D7FF ,"Mathematical Alphanumeric Symbols"),
new UnicodeRange (0x1F600, 0x1F532 ,"Emojis Symbols"),
new UnicodeRange (0x20000, 0x2A6DF ,"CJK Unified Ideographs Extension B"),
new UnicodeRange (0x2F800, 0x2FA1F ,"CJK Compatibility Ideographs Supplement"),
new UnicodeRange (0xE0000, 0xE007F ,"Tags"),
};
private void SetDemoTableStyles ()
{
tableView.Style.ColumnStyles.Clear ();
var alignMid = new ColumnStyle () {
Alignment = TextAlignment.Centered
};
var alignRight = new ColumnStyle () {
Alignment = TextAlignment.Right
};
var dateFormatStyle = new ColumnStyle () {
Alignment = TextAlignment.Right,
RepresentationGetter = (v) => v is DateTime d ? d.ToString ("yyyy-MM-dd") : v.ToString ()
};
var negativeRight = new ColumnStyle () {
Format = "0.##",
MinWidth = 10,
AlignmentGetter = (v) => v is double d ?
// align negative values right
d < 0 ? TextAlignment.Right :
// align positive values left
TextAlignment.Left :
// not a double
TextAlignment.Left,
ColorGetter = (a) => a.CellValue is double d ?
// color 0 and negative values red
d <= 0.0000001 ? a.RowIndex % 2 == 0 && _miAlternatingColors.Checked == true ? redColorSchemeAlt : redColorScheme :
// use normal scheme for positive values
null :
// not a double
null
};
tableView.Style.ColumnStyles.Add (currentTable.Columns ["DateCol"].Ordinal, dateFormatStyle);
tableView.Style.ColumnStyles.Add (currentTable.Columns ["DoubleCol"].Ordinal, negativeRight);
tableView.Style.ColumnStyles.Add (currentTable.Columns ["NullsCol"].Ordinal, alignMid);
tableView.Style.ColumnStyles.Add (currentTable.Columns ["IntCol"].Ordinal, alignRight);
tableView.Update ();
}
private void OpenSimple (bool big)
{
SetTable (BuildSimpleDataTable (big ? 30 : 5, big ? 1000 : 5));
}
private void EditCurrentCell (object sender, CellActivatedEventArgs e)
{
if (e.Table as DataTableSource == null || currentTable == null)
return;
var tableCol = ToTableCol (e.Col);
if (tableCol < 0) {
return;
}
var o = currentTable.Rows [e.Row] [tableCol];
var title = o is uint u ? GetUnicodeCategory (u) + $"(0x{o:X4})" : "Enter new value";
var oldValue = currentTable.Rows [e.Row] [tableCol].ToString ();
bool okPressed = false;
var ok = new Button ("Ok", is_default: true);
ok.Clicked += (s, e) => { okPressed = true; Application.RequestStop (); };
var cancel = new Button ("Cancel");
cancel.Clicked += (s, e) => { Application.RequestStop (); };
var d = new Dialog (ok, cancel) { Title = title };
var lbl = new Label () {
X = 0,
Y = 1,
Text = tableView.Table.ColumnNames [e.Col]
};
var tf = new TextField () {
Text = oldValue,
X = 0,
Y = 2,
Width = Dim.Fill ()
};
d.Add (lbl, tf);
tf.SetFocus ();
Application.Run (d);
if (okPressed) {
try {
currentTable.Rows [e.Row] [tableCol] = string.IsNullOrWhiteSpace (tf.Text) ? DBNull.Value : (object)tf.Text;
} catch (Exception ex) {
MessageBox.ErrorQuery (60, 20, "Failed to set text", ex.Message, "Ok");
}
tableView.Update ();
}
}
private int ToTableCol (int col)
{
if (HasCheckboxes ()) {
return col - 1;
}
return col;
}
private bool HasCheckboxes ()
{
return tableView.Table is CheckBoxTableSourceWrapperBase;
}
private string GetUnicodeCategory (uint u)
{
return Ranges.FirstOrDefault (r => u >= r.Start && u <= r.End)?.Category ?? "Unknown";
}
/// <summary>
/// Generates a new demo <see cref="DataTable"/> with the given number of <paramref name="cols"/> (min 5) and <paramref name="rows"/>
/// </summary>
/// <param name="cols"></param>
/// <param name="rows"></param>
/// <returns></returns>
public static DataTable BuildDemoDataTable (int cols, int rows)
{
var dt = new DataTable ();
int explicitCols = 6;
dt.Columns.Add (new DataColumn ("StrCol", typeof (string)));
dt.Columns.Add (new DataColumn ("DateCol", typeof (DateTime)));
dt.Columns.Add (new DataColumn ("IntCol", typeof (int)));
dt.Columns.Add (new DataColumn ("DoubleCol", typeof (double)));
dt.Columns.Add (new DataColumn ("NullsCol", typeof (string)));
dt.Columns.Add (new DataColumn ("Unicode", typeof (string)));
for (int i = 0; i < cols - explicitCols; i++) {
dt.Columns.Add ("Column" + (i + explicitCols));
}
var r = new Random (100);
for (int i = 0; i < rows; i++) {
List<object> row = new List<object> (){
"Some long text that is super cool",
new DateTime(2000+i,12,25),
r.Next(i),
(r.NextDouble()*i)-0.5 /*add some negatives to demo styles*/,
DBNull.Value,
"Les Mise" + Char.ConvertFromUtf32(Int32.Parse("0301", NumberStyles.HexNumber)) + "rables"
};
for (int j = 0; j < cols - explicitCols; j++) {
row.Add ("SomeValue" + r.Next (100));
}
dt.Rows.Add (row.ToArray ());
}
return dt;
}
/// <summary>
/// Builds a simple table in which cell values contents are the index of the cell. This helps testing that scrolling etc is working correctly and not skipping out any rows/columns when paging
/// </summary>
/// <param name="cols"></param>
/// <param name="rows"></param>
/// <returns></returns>
public static DataTable BuildSimpleDataTable (int cols, int rows)
{
var dt = new DataTable ();
for (int c = 0; c < cols; c++) {
dt.Columns.Add ("Col" + c);
}
for (int r = 0; r < rows; r++) {
var newRow = dt.NewRow ();
for (int c = 0; c < cols; c++) {
newRow [c] = $"R{r}C{c}";
}
dt.Rows.Add (newRow);
}
return dt;
}
}
}