Notepad.cs 9.5 KB

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