CsvEditor.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Data;
  4. using Terminal.Gui;
  5. using Terminal.Gui.Trees;
  6. using System.Linq;
  7. using System.Globalization;
  8. using System.IO;
  9. using System.Text;
  10. using NStack;
  11. namespace UICatalog.Scenarios {
  12. [ScenarioMetadata (Name: "Csv Editor", Description: "Open and edit simple CSV files")]
  13. [ScenarioCategory ("Controls")]
  14. [ScenarioCategory ("Dialogs")]
  15. [ScenarioCategory ("Text")]
  16. [ScenarioCategory ("Dialogs")]
  17. [ScenarioCategory ("TopLevel")]
  18. public class CsvEditor : Scenario
  19. {
  20. TableView tableView;
  21. private string currentFile;
  22. private MenuItem miLeft;
  23. private MenuItem miRight;
  24. private MenuItem miCentered;
  25. private Label selectedCellLabel;
  26. public override void Setup ()
  27. {
  28. Win.Title = this.GetName();
  29. Win.Y = 1; // menu
  30. Win.Height = Dim.Fill (1); // status bar
  31. Top.LayoutSubviews ();
  32. this.tableView = new TableView () {
  33. X = 0,
  34. Y = 0,
  35. Width = Dim.Fill (),
  36. Height = Dim.Fill (1),
  37. };
  38. var menu = new MenuBar (new MenuBarItem [] {
  39. new MenuBarItem ("_File", new MenuItem [] {
  40. new MenuItem ("_Open CSV", "", () => Open()),
  41. new MenuItem ("_Save", "", () => Save()),
  42. new MenuItem ("_Quit", "", () => Quit()),
  43. }),
  44. new MenuBarItem ("_Edit", new MenuItem [] {
  45. new MenuItem ("_New Column", "", () => AddColumn()),
  46. new MenuItem ("_New Row", "", () => AddRow()),
  47. new MenuItem ("_Rename Column", "", () => RenameColumn()),
  48. new MenuItem ("_Delete Column", "", () => DeleteColum()),
  49. new MenuItem ("_Move Column", "", () => MoveColumn()),
  50. new MenuItem ("_Move Row", "", () => MoveRow()),
  51. new MenuItem ("_Sort Asc", "", () => Sort(true)),
  52. new MenuItem ("_Sort Desc", "", () => Sort(false)),
  53. }),
  54. new MenuBarItem ("_View", new MenuItem [] {
  55. miLeft = new MenuItem ("_Align Left", "", () => Align(TextAlignment.Left)),
  56. miRight = new MenuItem ("_Align Right", "", () => Align(TextAlignment.Right)),
  57. miCentered = new MenuItem ("_Align Centered", "", () => Align(TextAlignment.Centered)),
  58. // Format requires hard typed data table, when we read a CSV everything is untyped (string) so this only works for new columns in this demo
  59. miCentered = new MenuItem ("_Set Format Pattern", "", () => SetFormat()),
  60. })
  61. });
  62. Top.Add (menu);
  63. var statusBar = new StatusBar (new StatusItem [] {
  64. new StatusItem(Key.CtrlMask | Key.O, "~^O~ Open", () => Open()),
  65. new StatusItem(Key.CtrlMask | Key.S, "~^S~ Save", () => Save()),
  66. new StatusItem(Key.CtrlMask | Key.Q, "~^Q~ Quit", () => Quit()),
  67. });
  68. Top.Add (statusBar);
  69. Win.Add (tableView);
  70. selectedCellLabel = new Label(){
  71. X = 0,
  72. Y = Pos.Bottom(tableView),
  73. Text = "0,0",
  74. Width = Dim.Fill(),
  75. TextAlignment = TextAlignment.Right
  76. };
  77. Win.Add(selectedCellLabel);
  78. tableView.SelectedCellChanged += OnSelectedCellChanged;
  79. tableView.CellActivated += EditCurrentCell;
  80. tableView.KeyPress += TableViewKeyPress;
  81. SetupScrollBar();
  82. }
  83. private void OnSelectedCellChanged (TableView.SelectedCellChangedEventArgs e)
  84. {
  85. selectedCellLabel.Text = $"{tableView.SelectedRow},{tableView.SelectedColumn}";
  86. if(tableView.Table == null || tableView.SelectedColumn == -1)
  87. return;
  88. var col = tableView.Table.Columns[tableView.SelectedColumn];
  89. var style = tableView.Style.GetColumnStyleIfAny(col);
  90. miLeft.Checked = style?.Alignment == TextAlignment.Left;
  91. miRight.Checked = style?.Alignment == TextAlignment.Right;
  92. miCentered.Checked = style?.Alignment == TextAlignment.Centered;
  93. }
  94. private void RenameColumn ()
  95. {
  96. if(NoTableLoaded()) {
  97. return;
  98. }
  99. var currentCol = tableView.Table.Columns[tableView.SelectedColumn];
  100. if(GetText("Rename Column","Name:",currentCol.ColumnName,out string newName)) {
  101. currentCol.ColumnName = newName;
  102. tableView.Update();
  103. }
  104. }
  105. private void DeleteColum()
  106. {
  107. if(NoTableLoaded()) {
  108. return;
  109. }
  110. if(tableView.SelectedColumn == -1) {
  111. MessageBox.ErrorQuery("No Column","No column selected", "Ok");
  112. return;
  113. }
  114. try {
  115. tableView.Table.Columns.RemoveAt(tableView.SelectedColumn);
  116. tableView.Update();
  117. } catch (Exception ex) {
  118. MessageBox.ErrorQuery("Could not remove column",ex.Message, "Ok");
  119. }
  120. }
  121. private void MoveColumn ()
  122. {
  123. if(NoTableLoaded()) {
  124. return;
  125. }
  126. if(tableView.SelectedColumn == -1) {
  127. MessageBox.ErrorQuery("No Column","No column selected", "Ok");
  128. return;
  129. }
  130. try{
  131. var currentCol = tableView.Table.Columns[tableView.SelectedColumn];
  132. if(GetText("Move Column","New Index:",currentCol.Ordinal.ToString(),out string newOrdinal)) {
  133. var newIdx = Math.Min(Math.Max(0,int.Parse(newOrdinal)),tableView.Table.Columns.Count-1);
  134. currentCol.SetOrdinal(newIdx);
  135. tableView.SetSelection(newIdx,tableView.SelectedRow,false);
  136. tableView.EnsureSelectedCellIsVisible();
  137. tableView.SetNeedsDisplay();
  138. }
  139. }catch(Exception ex)
  140. {
  141. MessageBox.ErrorQuery("Error moving column",ex.Message, "Ok");
  142. }
  143. }
  144. private void Sort (bool asc)
  145. {
  146. if(NoTableLoaded()) {
  147. return;
  148. }
  149. if(tableView.SelectedColumn == -1) {
  150. MessageBox.ErrorQuery("No Column","No column selected", "Ok");
  151. return;
  152. }
  153. var colName = tableView.Table.Columns[tableView.SelectedColumn].ColumnName;
  154. tableView.Table.DefaultView.Sort = colName + (asc ? " asc" : " desc");
  155. tableView.Table = tableView.Table.DefaultView.ToTable();
  156. }
  157. private void MoveRow ()
  158. {
  159. if(NoTableLoaded()) {
  160. return;
  161. }
  162. if(tableView.SelectedRow == -1) {
  163. MessageBox.ErrorQuery("No Rows","No row selected", "Ok");
  164. return;
  165. }
  166. try{
  167. int oldIdx = tableView.SelectedRow;
  168. var currentRow = tableView.Table.Rows[oldIdx];
  169. if(GetText("Move Row","New Row:",oldIdx.ToString(),out string newOrdinal)) {
  170. var newIdx = Math.Min(Math.Max(0,int.Parse(newOrdinal)),tableView.Table.Rows.Count-1);
  171. if(newIdx == oldIdx)
  172. return;
  173. var arrayItems = currentRow.ItemArray;
  174. tableView.Table.Rows.Remove(currentRow);
  175. // Removing and Inserting the same DataRow seems to result in it loosing its values so we have to create a new instance
  176. var newRow = tableView.Table.NewRow();
  177. newRow.ItemArray = arrayItems;
  178. tableView.Table.Rows.InsertAt(newRow,newIdx);
  179. tableView.SetSelection(tableView.SelectedColumn,newIdx,false);
  180. tableView.EnsureSelectedCellIsVisible();
  181. tableView.SetNeedsDisplay();
  182. }
  183. }catch(Exception ex)
  184. {
  185. MessageBox.ErrorQuery("Error moving column",ex.Message, "Ok");
  186. }
  187. }
  188. private void Align (TextAlignment newAlignment)
  189. {
  190. if (NoTableLoaded ()) {
  191. return;
  192. }
  193. var col = tableView.Table.Columns[tableView.SelectedColumn];
  194. var style = tableView.Style.GetOrCreateColumnStyle(col);
  195. style.Alignment = newAlignment;
  196. miLeft.Checked = style.Alignment == TextAlignment.Left;
  197. miRight.Checked = style.Alignment == TextAlignment.Right;
  198. miCentered.Checked = style.Alignment == TextAlignment.Centered;
  199. tableView.Update();
  200. }
  201. private void SetFormat()
  202. {
  203. if (NoTableLoaded ()) {
  204. return;
  205. }
  206. var col = tableView.Table.Columns[tableView.SelectedColumn];
  207. if(col.DataType == typeof(string)) {
  208. MessageBox.ErrorQuery("Cannot Format Column","String columns cannot be Formatted, try adding a new column to the table with a date/numerical Type","Ok");
  209. return;
  210. }
  211. var style = tableView.Style.GetOrCreateColumnStyle(col);
  212. if(GetText("Format","Pattern:",style.Format ?? "",out string newPattern)) {
  213. style.Format = newPattern;
  214. tableView.Update();
  215. }
  216. }
  217. private bool NoTableLoaded ()
  218. {
  219. if(tableView.Table == null) {
  220. MessageBox.ErrorQuery("No Table Loaded","No table has currently be opened","Ok");
  221. return true;
  222. }
  223. return false;
  224. }
  225. private void AddRow ()
  226. {
  227. if(NoTableLoaded()) {
  228. return;
  229. }
  230. var newRow = tableView.Table.NewRow();
  231. var newRowIdx = Math.Min(Math.Max(0,tableView.SelectedRow+1),tableView.Table.Rows.Count);
  232. tableView.Table.Rows.InsertAt(newRow,newRowIdx);
  233. tableView.Update();
  234. }
  235. private void AddColumn ()
  236. {
  237. if(NoTableLoaded()) {
  238. return;
  239. }
  240. if(GetText("Enter column name","Name:","",out string colName)) {
  241. var col = new DataColumn(colName);
  242. var newColIdx = Math.Min(Math.Max(0,tableView.SelectedColumn + 1),tableView.Table.Columns.Count);
  243. int result = MessageBox.Query(40,15,"Column Type","Pick a data type for the column",new ustring[]{"Date","Integer","Double","Text","Cancel"});
  244. if(result <= -1 || result >= 4)
  245. return;
  246. switch(result) {
  247. case 0: col.DataType = typeof(DateTime);
  248. break;
  249. case 1: col.DataType = typeof(int);
  250. break;
  251. case 2: col.DataType = typeof(double);
  252. break;
  253. case 3: col.DataType = typeof(string);
  254. break;
  255. }
  256. tableView.Table.Columns.Add(col);
  257. col.SetOrdinal(newColIdx);
  258. tableView.Update();
  259. }
  260. }
  261. private void Save()
  262. {
  263. if(tableView.Table == null || string.IsNullOrWhiteSpace(currentFile)) {
  264. MessageBox.ErrorQuery("No file loaded","No file is currently loaded","Ok");
  265. return;
  266. }
  267. var sb = new StringBuilder();
  268. sb.AppendLine(string.Join(",",tableView.Table.Columns.Cast<DataColumn>().Select(c=>c.ColumnName)));
  269. foreach(DataRow row in tableView.Table.Rows) {
  270. sb.AppendLine(string.Join(",",row.ItemArray));
  271. }
  272. File.WriteAllText(currentFile,sb.ToString());
  273. }
  274. private void Open()
  275. {
  276. var ofd = new FileDialog("Select File","Open","File","Select a CSV file to open (does not support newlines, escaping etc)");
  277. ofd.AllowedFileTypes = new string[]{".csv" };
  278. Application.Run(ofd);
  279. if(!ofd.Canceled && !string.IsNullOrWhiteSpace(ofd.FilePath?.ToString()))
  280. {
  281. Open(ofd.FilePath.ToString());
  282. }
  283. }
  284. private void Open(string filename)
  285. {
  286. int lineNumber = 0;
  287. currentFile = null;
  288. try {
  289. var dt = new DataTable();
  290. var lines = File.ReadAllLines(filename);
  291. foreach(var h in lines[0].Split(',')){
  292. dt.Columns.Add(h);
  293. }
  294. foreach(var line in lines.Skip(1)) {
  295. lineNumber++;
  296. dt.Rows.Add(line.Split(','));
  297. }
  298. tableView.Table = dt;
  299. // Only set the current filename if we succesfully loaded the entire file
  300. currentFile = filename;
  301. }
  302. catch(Exception ex) {
  303. MessageBox.ErrorQuery("Open Failed",$"Error on line {lineNumber}{Environment.NewLine}{ex.Message}","Ok");
  304. }
  305. }
  306. private void SetupScrollBar ()
  307. {
  308. var _scrollBar = new ScrollBarView (tableView, true);
  309. _scrollBar.ChangedPosition += () => {
  310. tableView.RowOffset = _scrollBar.Position;
  311. if (tableView.RowOffset != _scrollBar.Position) {
  312. _scrollBar.Position = tableView.RowOffset;
  313. }
  314. tableView.SetNeedsDisplay ();
  315. };
  316. /*
  317. _scrollBar.OtherScrollBarView.ChangedPosition += () => {
  318. _listView.LeftItem = _scrollBar.OtherScrollBarView.Position;
  319. if (_listView.LeftItem != _scrollBar.OtherScrollBarView.Position) {
  320. _scrollBar.OtherScrollBarView.Position = _listView.LeftItem;
  321. }
  322. _listView.SetNeedsDisplay ();
  323. };*/
  324. tableView.DrawContent += (e) => {
  325. _scrollBar.Size = tableView.Table?.Rows?.Count ??0;
  326. _scrollBar.Position = tableView.RowOffset;
  327. // _scrollBar.OtherScrollBarView.Size = _listView.Maxlength - 1;
  328. // _scrollBar.OtherScrollBarView.Position = _listView.LeftItem;
  329. _scrollBar.Refresh ();
  330. };
  331. }
  332. private void TableViewKeyPress (View.KeyEventEventArgs e)
  333. {
  334. if(e.KeyEvent.Key == Key.DeleteChar){
  335. if(tableView.FullRowSelect)
  336. {
  337. // Delete button deletes all rows when in full row mode
  338. foreach(int toRemove in tableView.GetAllSelectedCells().Select(p=>p.Y).Distinct().OrderByDescending(i=>i))
  339. tableView.Table.Rows.RemoveAt(toRemove);
  340. }
  341. else{
  342. // otherwise set all selected cells to null
  343. foreach(var pt in tableView.GetAllSelectedCells())
  344. {
  345. tableView.Table.Rows[pt.Y][pt.X] = DBNull.Value;
  346. }
  347. }
  348. tableView.Update();
  349. e.Handled = true;
  350. }
  351. }
  352. private void ClearColumnStyles ()
  353. {
  354. tableView.Style.ColumnStyles.Clear();
  355. tableView.Update();
  356. }
  357. private void CloseExample ()
  358. {
  359. tableView.Table = null;
  360. }
  361. private void Quit ()
  362. {
  363. Application.RequestStop ();
  364. }
  365. private bool GetText(string title, string label, string initialText, out string enteredText)
  366. {
  367. bool okPressed = false;
  368. var ok = new Button ("Ok", is_default: true);
  369. ok.Clicked += () => { okPressed = true; Application.RequestStop (); };
  370. var cancel = new Button ("Cancel");
  371. cancel.Clicked += () => { Application.RequestStop (); };
  372. var d = new Dialog (title, 60, 20, ok, cancel);
  373. var lbl = new Label() {
  374. X = 0,
  375. Y = 1,
  376. Text = label
  377. };
  378. var tf = new TextField()
  379. {
  380. Text = initialText,
  381. X = 0,
  382. Y = 2,
  383. Width = Dim.Fill()
  384. };
  385. d.Add (lbl,tf);
  386. tf.SetFocus();
  387. Application.Run (d);
  388. enteredText = okPressed? tf.Text.ToString() : null;
  389. return okPressed;
  390. }
  391. private void EditCurrentCell (TableView.CellActivatedEventArgs e)
  392. {
  393. if(e.Table == null)
  394. return;
  395. var oldValue = e.Table.Rows[e.Row][e.Col].ToString();
  396. if(GetText("Enter new value",e.Table.Columns[e.Col].ColumnName,oldValue, out string newText)) {
  397. try {
  398. e.Table.Rows[e.Row][e.Col] = string.IsNullOrWhiteSpace(newText) ? DBNull.Value : (object)newText;
  399. }
  400. catch(Exception ex) {
  401. MessageBox.ErrorQuery(60,20,"Failed to set text", ex.Message,"Ok");
  402. }
  403. tableView.Update();
  404. }
  405. }
  406. }
  407. }