TableEditor.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Data;
  4. using Terminal.Gui;
  5. using System.Linq;
  6. using System.Globalization;
  7. namespace UICatalog.Scenarios {
  8. [ScenarioMetadata (Name: "TableEditor", Description: "A Terminal.Gui DataTable editor via TableView")]
  9. [ScenarioCategory ("Controls")]
  10. [ScenarioCategory ("Dialogs")]
  11. [ScenarioCategory ("Text")]
  12. [ScenarioCategory ("Dialogs")]
  13. [ScenarioCategory ("TopLevel"), ScenarioCategory ("TableView")]
  14. public class TableEditor : Scenario
  15. {
  16. TableView tableView;
  17. private MenuItem miAlwaysShowHeaders;
  18. private MenuItem miHeaderOverline;
  19. private MenuItem miHeaderMidline;
  20. private MenuItem miHeaderUnderline;
  21. private MenuItem miCellLines;
  22. private MenuItem miFullRowSelect;
  23. private MenuItem miExpandLastColumn;
  24. private MenuItem miAlternatingColors;
  25. private MenuItem miCursor;
  26. ColorScheme redColorScheme;
  27. ColorScheme redColorSchemeAlt;
  28. ColorScheme alternatingColorScheme;
  29. public override void Setup ()
  30. {
  31. Win.Title = this.GetName();
  32. Win.Y = 1; // menu
  33. Win.Height = Dim.Fill (1); // status bar
  34. Top.LayoutSubviews ();
  35. this.tableView = new TableView () {
  36. X = 0,
  37. Y = 0,
  38. Width = Dim.Fill (),
  39. Height = Dim.Fill (1),
  40. };
  41. var menu = new MenuBar (new MenuBarItem [] {
  42. new MenuBarItem ("_File", new MenuItem [] {
  43. new MenuItem ("_OpenBigExample", "", () => OpenExample(true)),
  44. new MenuItem ("_OpenSmallExample", "", () => OpenExample(false)),
  45. new MenuItem ("_CloseExample", "", () => CloseExample()),
  46. new MenuItem ("_Quit", "", () => Quit()),
  47. }),
  48. new MenuBarItem ("_View", new MenuItem [] {
  49. miAlwaysShowHeaders = new MenuItem ("_AlwaysShowHeaders", "", () => ToggleAlwaysShowHeader()){Checked = tableView.Style.AlwaysShowHeaders, CheckType = MenuItemCheckStyle.Checked },
  50. miHeaderOverline = new MenuItem ("_HeaderOverLine", "", () => ToggleOverline()){Checked = tableView.Style.ShowHorizontalHeaderOverline, CheckType = MenuItemCheckStyle.Checked },
  51. miHeaderMidline = new MenuItem ("_HeaderMidLine", "", () => ToggleHeaderMidline()){Checked = tableView.Style.ShowVerticalHeaderLines, CheckType = MenuItemCheckStyle.Checked },
  52. miHeaderUnderline =new MenuItem ("_HeaderUnderLine", "", () => ToggleUnderline()){Checked = tableView.Style.ShowHorizontalHeaderUnderline, CheckType = MenuItemCheckStyle.Checked },
  53. miFullRowSelect =new MenuItem ("_FullRowSelect", "", () => ToggleFullRowSelect()){Checked = tableView.FullRowSelect, CheckType = MenuItemCheckStyle.Checked },
  54. miCellLines =new MenuItem ("_CellLines", "", () => ToggleCellLines()){Checked = tableView.Style.ShowVerticalCellLines, CheckType = MenuItemCheckStyle.Checked },
  55. miExpandLastColumn = new MenuItem ("_ExpandLastColumn", "", () => ToggleExpandLastColumn()){Checked = tableView.Style.ExpandLastColumn, CheckType = MenuItemCheckStyle.Checked },
  56. new MenuItem ("_AllLines", "", () => ToggleAllCellLines()),
  57. new MenuItem ("_NoLines", "", () => ToggleNoCellLines()),
  58. miAlternatingColors = new MenuItem ("Alternating Colors", "", () => ToggleAlternatingColors()){CheckType = MenuItemCheckStyle.Checked},
  59. miCursor = new MenuItem ("Invert Selected Cell First Character", "", () => ToggleInvertSelectedCellFirstCharacter()){Checked = tableView.Style.InvertSelectedCellFirstCharacter,CheckType = MenuItemCheckStyle.Checked},
  60. new MenuItem ("_ClearColumnStyles", "", () => ClearColumnStyles()),
  61. }),
  62. });
  63. Top.Add (menu);
  64. var statusBar = new StatusBar (new StatusItem [] {
  65. new StatusItem(Key.F2, "~F2~ OpenExample", () => OpenExample(true)),
  66. new StatusItem(Key.F3, "~F3~ CloseExample", () => CloseExample()),
  67. new StatusItem(Key.F4, "~F4~ OpenSimple", () => OpenSimple(true)),
  68. new StatusItem(Key.CtrlMask | Key.Q, "~^Q~ Quit", () => Quit()),
  69. });
  70. Top.Add (statusBar);
  71. Win.Add (tableView);
  72. var selectedCellLabel = new Label(){
  73. X = 0,
  74. Y = Pos.Bottom(tableView),
  75. Text = "0,0",
  76. Width = Dim.Fill(),
  77. TextAlignment = TextAlignment.Right
  78. };
  79. Win.Add(selectedCellLabel);
  80. tableView.SelectedCellChanged += (e)=>{selectedCellLabel.Text = $"{tableView.SelectedRow},{tableView.SelectedColumn}";};
  81. tableView.CellActivated += EditCurrentCell;
  82. tableView.KeyPress += TableViewKeyPress;
  83. SetupScrollBar();
  84. redColorScheme = new ColorScheme(){
  85. Disabled = Win.ColorScheme.Disabled,
  86. HotFocus = Win.ColorScheme.HotFocus,
  87. Focus = Win.ColorScheme.Focus,
  88. Normal = Application.Driver.MakeAttribute(Color.Red,Win.ColorScheme.Normal.Background)
  89. };
  90. alternatingColorScheme = new ColorScheme(){
  91. Disabled = Win.ColorScheme.Disabled,
  92. HotFocus = Win.ColorScheme.HotFocus,
  93. Focus = Win.ColorScheme.Focus,
  94. Normal = Application.Driver.MakeAttribute(Color.White,Color.BrightBlue)
  95. };
  96. redColorSchemeAlt = new ColorScheme(){
  97. Disabled = Win.ColorScheme.Disabled,
  98. HotFocus = Win.ColorScheme.HotFocus,
  99. Focus = Win.ColorScheme.Focus,
  100. Normal = Application.Driver.MakeAttribute(Color.Red,Color.BrightBlue)
  101. };
  102. }
  103. private void SetupScrollBar ()
  104. {
  105. var _scrollBar = new ScrollBarView (tableView, true);
  106. _scrollBar.ChangedPosition += () => {
  107. tableView.RowOffset = _scrollBar.Position;
  108. if (tableView.RowOffset != _scrollBar.Position) {
  109. _scrollBar.Position = tableView.RowOffset;
  110. }
  111. tableView.SetNeedsDisplay ();
  112. };
  113. /*
  114. _scrollBar.OtherScrollBarView.ChangedPosition += () => {
  115. _listView.LeftItem = _scrollBar.OtherScrollBarView.Position;
  116. if (_listView.LeftItem != _scrollBar.OtherScrollBarView.Position) {
  117. _scrollBar.OtherScrollBarView.Position = _listView.LeftItem;
  118. }
  119. _listView.SetNeedsDisplay ();
  120. };*/
  121. tableView.DrawContent += (e) => {
  122. _scrollBar.Size = tableView.Table?.Rows?.Count ??0;
  123. _scrollBar.Position = tableView.RowOffset;
  124. // _scrollBar.OtherScrollBarView.Size = _listView.Maxlength - 1;
  125. // _scrollBar.OtherScrollBarView.Position = _listView.LeftItem;
  126. _scrollBar.Refresh ();
  127. };
  128. }
  129. private void TableViewKeyPress (View.KeyEventEventArgs e)
  130. {
  131. if(e.KeyEvent.Key == Key.DeleteChar){
  132. if(tableView.FullRowSelect)
  133. {
  134. // Delete button deletes all rows when in full row mode
  135. foreach(int toRemove in tableView.GetAllSelectedCells().Select(p=>p.Y).Distinct().OrderByDescending(i=>i))
  136. tableView.Table.Rows.RemoveAt(toRemove);
  137. }
  138. else{
  139. // otherwise set all selected cells to null
  140. foreach(var pt in tableView.GetAllSelectedCells())
  141. {
  142. tableView.Table.Rows[pt.Y][pt.X] = DBNull.Value;
  143. }
  144. }
  145. tableView.Update();
  146. e.Handled = true;
  147. }
  148. }
  149. private void ClearColumnStyles ()
  150. {
  151. tableView.Style.ColumnStyles.Clear();
  152. tableView.Update();
  153. }
  154. private void ToggleAlwaysShowHeader ()
  155. {
  156. miAlwaysShowHeaders.Checked = !miAlwaysShowHeaders.Checked;
  157. tableView.Style.AlwaysShowHeaders = miAlwaysShowHeaders.Checked;
  158. tableView.Update();
  159. }
  160. private void ToggleOverline ()
  161. {
  162. miHeaderOverline.Checked = !miHeaderOverline.Checked;
  163. tableView.Style.ShowHorizontalHeaderOverline = miHeaderOverline.Checked;
  164. tableView.Update();
  165. }
  166. private void ToggleHeaderMidline ()
  167. {
  168. miHeaderMidline.Checked = !miHeaderMidline.Checked;
  169. tableView.Style.ShowVerticalHeaderLines = miHeaderMidline.Checked;
  170. tableView.Update();
  171. }
  172. private void ToggleUnderline ()
  173. {
  174. miHeaderUnderline.Checked = !miHeaderUnderline.Checked;
  175. tableView.Style.ShowHorizontalHeaderUnderline = miHeaderUnderline.Checked;
  176. tableView.Update();
  177. }
  178. private void ToggleFullRowSelect ()
  179. {
  180. miFullRowSelect.Checked = !miFullRowSelect.Checked;
  181. tableView.FullRowSelect= miFullRowSelect.Checked;
  182. tableView.Update();
  183. }
  184. private void ToggleExpandLastColumn()
  185. {
  186. miExpandLastColumn.Checked = !miExpandLastColumn.Checked;
  187. tableView.Style.ExpandLastColumn = miExpandLastColumn.Checked;
  188. tableView.Update();
  189. }
  190. private void ToggleCellLines()
  191. {
  192. miCellLines.Checked = !miCellLines.Checked;
  193. tableView.Style.ShowVerticalCellLines = miCellLines.Checked;
  194. tableView.Update();
  195. }
  196. private void ToggleAllCellLines()
  197. {
  198. tableView.Style.ShowHorizontalHeaderOverline = true;
  199. tableView.Style.ShowVerticalHeaderLines = true;
  200. tableView.Style.ShowHorizontalHeaderUnderline = true;
  201. tableView.Style.ShowVerticalCellLines = true;
  202. miHeaderOverline.Checked = true;
  203. miHeaderMidline.Checked = true;
  204. miHeaderUnderline.Checked = true;
  205. miCellLines.Checked = true;
  206. tableView.Update();
  207. }
  208. private void ToggleNoCellLines()
  209. {
  210. tableView.Style.ShowHorizontalHeaderOverline = false;
  211. tableView.Style.ShowVerticalHeaderLines = false;
  212. tableView.Style.ShowHorizontalHeaderUnderline = false;
  213. tableView.Style.ShowVerticalCellLines = false;
  214. miHeaderOverline.Checked = false;
  215. miHeaderMidline.Checked = false;
  216. miHeaderUnderline.Checked = false;
  217. miCellLines.Checked = false;
  218. tableView.Update();
  219. }
  220. private void ToggleAlternatingColors()
  221. {
  222. //toggle menu item
  223. miAlternatingColors.Checked = !miAlternatingColors.Checked;
  224. if(miAlternatingColors.Checked){
  225. tableView.Style.RowColorGetter = (a)=> {return a.RowIndex%2==0 ? alternatingColorScheme : null;};
  226. }
  227. else
  228. {
  229. tableView.Style.RowColorGetter = null;
  230. }
  231. tableView.SetNeedsDisplay();
  232. }
  233. private void ToggleInvertSelectedCellFirstCharacter ()
  234. {
  235. //toggle menu item
  236. miCursor.Checked = !miCursor.Checked;
  237. tableView.Style.InvertSelectedCellFirstCharacter = miCursor.Checked;
  238. tableView.SetNeedsDisplay ();
  239. }
  240. private void CloseExample ()
  241. {
  242. tableView.Table = null;
  243. }
  244. private void Quit ()
  245. {
  246. Application.RequestStop ();
  247. }
  248. private void OpenExample (bool big)
  249. {
  250. tableView.Table = BuildDemoDataTable(big ? 30 : 5, big ? 1000 : 5);
  251. SetDemoTableStyles();
  252. }
  253. private void SetDemoTableStyles ()
  254. {
  255. var alignMid = new TableView.ColumnStyle () {
  256. Alignment = TextAlignment.Centered
  257. };
  258. var alignRight = new TableView.ColumnStyle () {
  259. Alignment = TextAlignment.Right
  260. };
  261. var dateFormatStyle = new TableView.ColumnStyle () {
  262. Alignment = TextAlignment.Right,
  263. RepresentationGetter = (v)=> v is DateTime d ? d.ToString("yyyy-MM-dd"):v.ToString()
  264. };
  265. var negativeRight = new TableView.ColumnStyle () {
  266. Format = "0.##",
  267. MinWidth = 10,
  268. AlignmentGetter = (v)=>v is double d ?
  269. // align negative values right
  270. d < 0 ? TextAlignment.Right :
  271. // align positive values left
  272. TextAlignment.Left:
  273. // not a double
  274. TextAlignment.Left,
  275. ColorGetter = (a)=> a.CellValue is double d ?
  276. // color 0 and negative values red
  277. d <= 0.0000001 ? a.RowIndex%2==0 && miAlternatingColors.Checked ? redColorSchemeAlt: redColorScheme :
  278. // use normal scheme for positive values
  279. null:
  280. // not a double
  281. null
  282. };
  283. tableView.Style.ColumnStyles.Add(tableView.Table.Columns["DateCol"],dateFormatStyle);
  284. tableView.Style.ColumnStyles.Add(tableView.Table.Columns["DoubleCol"],negativeRight);
  285. tableView.Style.ColumnStyles.Add(tableView.Table.Columns["NullsCol"],alignMid);
  286. tableView.Style.ColumnStyles.Add(tableView.Table.Columns["IntCol"],alignRight);
  287. tableView.Update();
  288. }
  289. private void OpenSimple (bool big)
  290. {
  291. tableView.Table = BuildSimpleDataTable(big ? 30 : 5, big ? 1000 : 5);
  292. }
  293. private void EditCurrentCell (TableView.CellActivatedEventArgs e)
  294. {
  295. if(e.Table == null)
  296. return;
  297. var oldValue = e.Table.Rows[e.Row][e.Col].ToString();
  298. bool okPressed = false;
  299. var ok = new Button ("Ok", is_default: true);
  300. ok.Clicked += () => { okPressed = true; Application.RequestStop (); };
  301. var cancel = new Button ("Cancel");
  302. cancel.Clicked += () => { Application.RequestStop (); };
  303. var d = new Dialog ("Enter new value", 60, 20, ok, cancel);
  304. var lbl = new Label() {
  305. X = 0,
  306. Y = 1,
  307. Text = e.Table.Columns[e.Col].ColumnName
  308. };
  309. var tf = new TextField()
  310. {
  311. Text = oldValue,
  312. X = 0,
  313. Y = 2,
  314. Width = Dim.Fill()
  315. };
  316. d.Add (lbl,tf);
  317. tf.SetFocus();
  318. Application.Run (d);
  319. if(okPressed) {
  320. try {
  321. e.Table.Rows[e.Row][e.Col] = string.IsNullOrWhiteSpace(tf.Text.ToString()) ? DBNull.Value : (object)tf.Text;
  322. }
  323. catch(Exception ex) {
  324. MessageBox.ErrorQuery(60,20,"Failed to set text", ex.Message,"Ok");
  325. }
  326. tableView.Update();
  327. }
  328. }
  329. /// <summary>
  330. /// Generates a new demo <see cref="DataTable"/> with the given number of <paramref name="cols"/> (min 5) and <paramref name="rows"/>
  331. /// </summary>
  332. /// <param name="cols"></param>
  333. /// <param name="rows"></param>
  334. /// <returns></returns>
  335. public static DataTable BuildDemoDataTable(int cols, int rows)
  336. {
  337. var dt = new DataTable();
  338. int explicitCols = 6;
  339. dt.Columns.Add(new DataColumn("StrCol",typeof(string)));
  340. dt.Columns.Add(new DataColumn("DateCol",typeof(DateTime)));
  341. dt.Columns.Add(new DataColumn("IntCol",typeof(int)));
  342. dt.Columns.Add(new DataColumn("DoubleCol",typeof(double)));
  343. dt.Columns.Add(new DataColumn("NullsCol",typeof(string)));
  344. dt.Columns.Add(new DataColumn("Unicode",typeof(string)));
  345. for(int i=0;i< cols -explicitCols; i++) {
  346. dt.Columns.Add("Column" + (i+explicitCols));
  347. }
  348. var r = new Random(100);
  349. for(int i=0;i< rows;i++) {
  350. List<object> row = new List<object>(){
  351. "Some long text that is super cool",
  352. new DateTime(2000+i,12,25),
  353. r.Next(i),
  354. (r.NextDouble()*i)-0.5 /*add some negatives to demo styles*/,
  355. DBNull.Value,
  356. "Les Mise" + Char.ConvertFromUtf32(Int32.Parse("0301", NumberStyles.HexNumber)) + "rables"
  357. };
  358. for(int j=0;j< cols -explicitCols; j++) {
  359. row.Add("SomeValue" + r.Next(100));
  360. }
  361. dt.Rows.Add(row.ToArray());
  362. }
  363. return dt;
  364. }
  365. /// <summary>
  366. /// 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
  367. /// </summary>
  368. /// <param name="cols"></param>
  369. /// <param name="rows"></param>
  370. /// <returns></returns>
  371. public static DataTable BuildSimpleDataTable(int cols, int rows)
  372. {
  373. var dt = new DataTable();
  374. for(int c = 0; c < cols; c++) {
  375. dt.Columns.Add("Col"+c);
  376. }
  377. for(int r = 0; r < rows; r++) {
  378. var newRow = dt.NewRow();
  379. for(int c = 0; c < cols; c++) {
  380. newRow[c] = $"R{r}C{c}";
  381. }
  382. dt.Rows.Add(newRow);
  383. }
  384. return dt;
  385. }
  386. }
  387. }