TableEditor.cs 12 KB

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