Notepad.cs 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416
  1. using System;
  2. using System.IO;
  3. using System.Linq;
  4. using Terminal.Gui;
  5. using Terminal.Gui.Graphs;
  6. namespace UICatalog.Scenarios {
  7. [ScenarioMetadata (Name: "Notepad", Description: "Multi-tab text editor uising the TabView control.")]
  8. [ScenarioCategory ("Controls"), ScenarioCategory ("TabView")]
  9. public class Notepad : Scenario {
  10. TabView tabView;
  11. private int numbeOfNewTabs = 1;
  12. private TabView focusedTabView;
  13. private StatusItem lenStatusItem;
  14. // Don't create a Window, just return the top-level view
  15. public override void Init (ColorScheme colorScheme)
  16. {
  17. Application.Init ();
  18. Application.Top.ColorScheme = Colors.Base;
  19. }
  20. public override void Setup ()
  21. {
  22. var menu = new MenuBar (new MenuBarItem [] {
  23. new MenuBarItem ("_File", new MenuItem [] {
  24. new MenuItem ("_New", "", () => New()),
  25. new MenuItem ("_Open", "", () => Open()),
  26. new MenuItem ("_Save", "", () => Save()),
  27. new MenuItem ("Save _As", "", () => SaveAs()),
  28. new MenuItem ("_Close", "", () => Close()),
  29. new MenuItem ("_Quit", "", () => Quit()),
  30. })
  31. });
  32. Application.Top.Add (menu);
  33. tabView = CreateNewTabView ();
  34. tabView.Style.ShowBorder = true;
  35. tabView.ApplyStyleChanges ();
  36. // Start with only a single view but support splitting to show side by side
  37. var split = new TileView(1) {
  38. X = 0,
  39. Y = 1,
  40. Width = Dim.Fill (),
  41. Height = Dim.Fill (1),
  42. };
  43. split.Tiles.ElementAt(0).ContentView.Add (tabView);
  44. split.Border.BorderStyle = BorderStyle.None;
  45. Application.Top.Add (split);
  46. lenStatusItem = new StatusItem (Key.CharMask, "Len: ", null);
  47. var statusBar = new StatusBar (new StatusItem [] {
  48. new StatusItem(Key.CtrlMask | Key.Q, "~^Q~ Quit", () => Quit()),
  49. // These shortcut keys don't seem to work correctly in linux
  50. //new StatusItem(Key.CtrlMask | Key.N, "~^O~ Open", () => Open()),
  51. //new StatusItem(Key.CtrlMask | Key.N, "~^N~ New", () => New()),
  52. new StatusItem(Key.CtrlMask | Key.S, "~^S~ Save", () => Save()),
  53. new StatusItem(Key.CtrlMask | Key.W, "~^W~ Close", () => Close()),
  54. lenStatusItem,
  55. });
  56. focusedTabView = tabView;
  57. tabView.SelectedTabChanged += TabView_SelectedTabChanged;
  58. tabView.Enter += (s, e) => focusedTabView = tabView;
  59. Application.Top.Add (statusBar);
  60. New ();
  61. }
  62. private void TabView_SelectedTabChanged (object sender, TabChangedEventArgs e)
  63. {
  64. lenStatusItem.Title = $"Len:{e.NewTab?.View?.Text?.Length ?? 0}";
  65. }
  66. private void TabView_TabClicked (object sender, TabMouseEventArgs e)
  67. {
  68. // we are only interested in right clicks
  69. if(!e.MouseEvent.Flags.HasFlag(MouseFlags.Button3Clicked)) {
  70. return;
  71. }
  72. MenuBarItem items;
  73. if (e.Tab == null) {
  74. items = new MenuBarItem (new MenuItem [] {
  75. new MenuItem ($"Open", "", () => Open()),
  76. });
  77. } else {
  78. var tv = (TabView)sender;
  79. var t = (OpenedFile)e.Tab;
  80. items = new MenuBarItem (new MenuItem [] {
  81. new MenuItem ($"Save", "", () => Save(focusedTabView, e.Tab)),
  82. new MenuItem ($"Close", "", () => Close(tv, e.Tab)),
  83. null,
  84. new MenuItem ($"Split Up", "", () => SplitUp(tv,t)),
  85. new MenuItem ($"Split Down", "", () => SplitDown(tv,t)),
  86. new MenuItem ($"Split Right", "", () => SplitRight(tv,t)),
  87. new MenuItem ($"Split Left", "", () => SplitLeft(tv,t)),
  88. });
  89. }
  90. ((View)sender).ViewToScreen (e.MouseEvent.X, e.MouseEvent.Y, out int screenX, out int screenY,true);
  91. var contextMenu = new ContextMenu (screenX,screenY, items);
  92. contextMenu.Show ();
  93. e.MouseEvent.Handled = true;
  94. }
  95. private void SplitUp (TabView sender, OpenedFile tab)
  96. {
  97. Split(0, Orientation.Horizontal,sender,tab);
  98. }
  99. private void SplitDown (TabView sender, OpenedFile tab)
  100. {
  101. Split(1, Orientation.Horizontal,sender,tab);
  102. }
  103. private void SplitLeft (TabView sender, OpenedFile tab)
  104. {
  105. Split(0, Orientation.Vertical,sender,tab);
  106. }
  107. private void SplitRight (TabView sender, OpenedFile tab)
  108. {
  109. Split(1, Orientation.Vertical,sender,tab);
  110. }
  111. private void Split (int offset, Orientation orientation,TabView sender, OpenedFile tab)
  112. {
  113. var split = (TileView)sender.SuperView.SuperView;
  114. var tileIndex = split.IndexOf(sender);
  115. if(tileIndex == -1)
  116. {
  117. return;
  118. }
  119. if(orientation != split.Orientation)
  120. {
  121. split.TrySplitTile(tileIndex,1,out split);
  122. split.Orientation = orientation;
  123. tileIndex = 0;
  124. }
  125. var newTile = split.InsertTile(tileIndex + offset);
  126. var newTabView = CreateNewTabView ();
  127. tab.CloneTo (newTabView);
  128. newTile.ContentView.Add(newTabView);
  129. newTabView.EnsureFocus();
  130. newTabView.FocusFirst();
  131. newTabView.FocusNext();
  132. }
  133. private TabView CreateNewTabView ()
  134. {
  135. var tv = new TabView () {
  136. X = 0,
  137. Y = 0,
  138. Width = Dim.Fill (),
  139. Height = Dim.Fill (),
  140. };
  141. tv.TabClicked += TabView_TabClicked;
  142. tv.SelectedTabChanged += TabView_SelectedTabChanged;
  143. tv.Enter += (s, e) => focusedTabView = tv;
  144. return tv;
  145. }
  146. private void New ()
  147. {
  148. Open (null, $"new {numbeOfNewTabs++}");
  149. }
  150. private void Close ()
  151. {
  152. Close (focusedTabView, focusedTabView.SelectedTab);
  153. }
  154. private void Close (TabView tv, TabView.Tab tabToClose)
  155. {
  156. var tab = tabToClose as OpenedFile;
  157. if (tab == null) {
  158. return;
  159. }
  160. focusedTabView = tv;
  161. if (tab.UnsavedChanges) {
  162. int result = MessageBox.Query ("Save Changes", $"Save changes to {tab.Text.ToString ().TrimEnd ('*')}", "Yes", "No", "Cancel");
  163. if (result == -1 || result == 2) {
  164. // user cancelled
  165. return;
  166. }
  167. if (result == 0) {
  168. if(tab.File == null) {
  169. SaveAs ();
  170. } else {
  171. tab.Save ();
  172. }
  173. }
  174. }
  175. // close and dispose the tab
  176. tv.RemoveTab (tab);
  177. tab.View.Dispose ();
  178. focusedTabView = tv;
  179. if(tv.Tabs.Count == 0) {
  180. var split = (TileView)tv.SuperView.SuperView;
  181. // if it is the last TabView on screen don't drop it or we will
  182. // be unable to open new docs!
  183. if(split.IsRootTileView() && split.Tiles.Count == 1) {
  184. return;
  185. }
  186. var tileIndex = split.IndexOf (tv);
  187. split.RemoveTile (tileIndex);
  188. if(split.Tiles.Count == 0) {
  189. var parent = split.GetParentTileView ();
  190. if (parent == null) {
  191. return;
  192. }
  193. var idx = parent.IndexOf (split);
  194. if (idx == -1) {
  195. return;
  196. }
  197. parent.RemoveTile (idx);
  198. }
  199. }
  200. }
  201. private void Open ()
  202. {
  203. var open = new OpenDialog ("Open", "Open a file") { AllowsMultipleSelection = true };
  204. Application.Run (open);
  205. if (!open.Canceled) {
  206. foreach (var path in open.FilePaths) {
  207. if (string.IsNullOrEmpty (path) || !File.Exists (path)) {
  208. return;
  209. }
  210. // TODO should open in focused TabView
  211. Open (new FileInfo (path), Path.GetFileName (path));
  212. }
  213. }
  214. }
  215. /// <summary>
  216. /// Creates a new tab with initial text
  217. /// </summary>
  218. /// <param name="fileInfo">File that was read or null if a new blank document</param>
  219. private void Open (FileInfo fileInfo, string tabName)
  220. {
  221. var tab = new OpenedFile (focusedTabView, tabName, fileInfo);
  222. focusedTabView.AddTab (tab, true);
  223. }
  224. public void Save ()
  225. {
  226. Save (focusedTabView, focusedTabView.SelectedTab);
  227. }
  228. public void Save (TabView tabViewToSave, TabView.Tab tabToSave)
  229. {
  230. var tab = tabToSave as OpenedFile;
  231. if (tab == null) {
  232. return;
  233. }
  234. if (tab.File == null) {
  235. SaveAs ();
  236. }
  237. tab.Save ();
  238. tabViewToSave.SetNeedsDisplay ();
  239. }
  240. public bool SaveAs ()
  241. {
  242. var tab = focusedTabView.SelectedTab as OpenedFile;
  243. if (tab == null) {
  244. return false;
  245. }
  246. var fd = new SaveDialog ();
  247. Application.Run (fd);
  248. if (string.IsNullOrWhiteSpace (fd.FilePath?.ToString ())) {
  249. return false;
  250. }
  251. if(fd.Canceled) {
  252. return false;
  253. }
  254. tab.File = new FileInfo (fd.FilePath.ToString ());
  255. tab.Text = fd.FileName.ToString ();
  256. tab.Save ();
  257. return true;
  258. }
  259. private class OpenedFile : TabView.Tab {
  260. public FileInfo File { get; set; }
  261. /// <summary>
  262. /// The text of the tab the last time it was saved
  263. /// </summary>
  264. /// <value></value>
  265. public string SavedText { get; set; }
  266. public bool UnsavedChanges => !string.Equals (SavedText, View.Text.ToString ());
  267. public OpenedFile (TabView parent, string name, FileInfo file)
  268. : base (name, CreateTextView(file))
  269. {
  270. File = file;
  271. SavedText = View.Text.ToString ();
  272. RegisterTextViewEvents (parent);
  273. }
  274. private void RegisterTextViewEvents (TabView parent)
  275. {
  276. var textView = (TextView)View;
  277. // when user makes changes rename tab to indicate unsaved
  278. textView.KeyUp += (s, k) => {
  279. // if current text doesn't match saved text
  280. var areDiff = this.UnsavedChanges;
  281. if (areDiff) {
  282. if (!this.Text.ToString ().EndsWith ('*')) {
  283. this.Text = this.Text.ToString () + '*';
  284. parent.SetNeedsDisplay ();
  285. }
  286. } else {
  287. if (Text.ToString ().EndsWith ('*')) {
  288. Text = Text.ToString ().TrimEnd ('*');
  289. parent.SetNeedsDisplay ();
  290. }
  291. }
  292. };
  293. }
  294. private static View CreateTextView (FileInfo file)
  295. {
  296. string initialText = string.Empty;
  297. if(file != null && file.Exists) {
  298. initialText = System.IO.File.ReadAllText (file.FullName);
  299. }
  300. return new TextView () {
  301. X = 0,
  302. Y = 0,
  303. Width = Dim.Fill (),
  304. Height = Dim.Fill (),
  305. Text = initialText,
  306. AllowsTab = false,
  307. };
  308. }
  309. public OpenedFile CloneTo(TabView other)
  310. {
  311. var newTab = new OpenedFile (other, base.Text.ToString(), File);
  312. other.AddTab (newTab, true);
  313. return newTab;
  314. }
  315. internal void Save ()
  316. {
  317. var newText = View.Text.ToString ();
  318. System.IO.File.WriteAllText (File.FullName, newText);
  319. SavedText = newText;
  320. Text = Text.ToString ().TrimEnd ('*');
  321. }
  322. }
  323. private void Quit ()
  324. {
  325. Application.RequestStop ();
  326. }
  327. }
  328. }