Menu.cs 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998
  1. //
  2. // Menu.cs: application menus and submenus
  3. //
  4. // Authors:
  5. // Miguel de Icaza ([email protected])
  6. //
  7. // TODO:
  8. // Add accelerator support, but should also support chords (ShortCut in MenuItem)
  9. // Allow menus inside menus
  10. using System;
  11. using NStack;
  12. using System.Linq;
  13. using System.Collections.Generic;
  14. namespace Terminal.Gui {
  15. /// <summary>
  16. /// A menu item has a title, an associated help text, and an action to execute on activation.
  17. /// </summary>
  18. public class MenuItem {
  19. /// <summary>
  20. /// Initializes a new <see cref="T:Terminal.Gui.MenuItem"/>.
  21. /// </summary>
  22. /// <param name="title">Title for the menu item.</param>
  23. /// <param name="help">Help text to display.</param>
  24. /// <param name="action">Action to invoke when the menu item is activated.</param>
  25. /// <param name="canExecute">Function to determine if the action can currently be executred.</param>
  26. public MenuItem (ustring title, string help, Action action, Func<bool> canExecute = null)
  27. {
  28. Title = title ?? "";
  29. Help = help ?? "";
  30. Action = action;
  31. CanExecute = canExecute;
  32. bool nextIsHot = false;
  33. foreach (var x in Title) {
  34. if (x == '_')
  35. nextIsHot = true;
  36. else {
  37. if (nextIsHot) {
  38. HotKey = Char.ToUpper ((char)x);
  39. break;
  40. }
  41. nextIsHot = false;
  42. }
  43. }
  44. }
  45. /// <summary>
  46. /// Initializes a new <see cref="T:Terminal.Gui.MenuItem"/>.
  47. /// </summary>
  48. /// <param name="title">Title for the menu item.</param>
  49. /// <param name="subMenu">The menu sub-menu.</param>
  50. public MenuItem (ustring title, MenuBarItem subMenu) : this (title, "", null)
  51. {
  52. SubMenu = subMenu;
  53. IsFromSubMenu = true;
  54. }
  55. //
  56. //
  57. /// <summary>
  58. /// The hotkey is used when the menu is active, the shortcut can be triggered when the menu is not active.
  59. /// For example HotKey would be "N" when the File Menu is open (assuming there is a "_New" entry
  60. /// if the ShortCut is set to "Control-N", this would be a global hotkey that would trigger as well
  61. /// </summary>
  62. public Rune HotKey;
  63. /// <summary>
  64. /// This is the global setting that can be used as a global shortcut to invoke the action on the menu.
  65. /// </summary>
  66. public Key ShortCut;
  67. /// <summary>
  68. /// Gets or sets the title.
  69. /// </summary>
  70. /// <value>The title.</value>
  71. public ustring Title { get; set; }
  72. /// <summary>
  73. /// Gets or sets the help text for the menu item.
  74. /// </summary>
  75. /// <value>The help text.</value>
  76. public ustring Help { get; set; }
  77. /// <summary>
  78. /// Gets or sets the action to be invoked when the menu is triggered
  79. /// </summary>
  80. /// <value>Method to invoke.</value>
  81. public Action Action { get; set; }
  82. /// <summary>
  83. /// Gets or sets the action to be invoked if the menu can be triggered
  84. /// </summary>
  85. /// <value>Function to determine if action is ready to be executed.</value>
  86. public Func<bool> CanExecute { get; set; }
  87. /// <summary>
  88. /// Shortcut to check if the menu item is enabled
  89. /// </summary>
  90. public bool IsEnabled ()
  91. {
  92. return CanExecute == null ? true : CanExecute ();
  93. }
  94. internal int Width => Title.Length + Help.Length + 1 + 2;
  95. /// <summary>
  96. /// Gets or sets the parent for this MenuBarItem
  97. /// </summary>
  98. /// <value>The parent.</value>
  99. internal MenuBarItem SubMenu { get; set; }
  100. internal bool IsFromSubMenu { get; set; }
  101. /// <summary>
  102. /// Merely a debugging aid to see the interaction with main
  103. /// </summary>
  104. public MenuItem GetMenuItem ()
  105. {
  106. return this;
  107. }
  108. /// <summary>
  109. /// Merely a debugging aid to see the interaction with main
  110. /// </summary>
  111. public bool GetMenuBarItem ()
  112. {
  113. return IsFromSubMenu;
  114. }
  115. }
  116. /// <summary>
  117. /// A menu bar item contains other menu items.
  118. /// </summary>
  119. public class MenuBarItem {
  120. /// <summary>
  121. /// Initializes a new <see cref="T:Terminal.Gui.MenuBarItem"/>.
  122. /// </summary>
  123. /// <param name="title">Title for the menu item.</param>
  124. /// <param name="children">The items in the current menu.</param>
  125. public MenuBarItem (ustring title, MenuItem [] children)
  126. {
  127. SetTitle (title ?? "");
  128. Children = children;
  129. }
  130. /// <summary>
  131. /// Initializes a new <see cref="T:Terminal.Gui.MenuBarItem"/>.
  132. /// </summary>
  133. /// <param name="children">The items in the current menu.</param>
  134. public MenuBarItem (MenuItem[] children) : this (new string (' ', GetMaxTitleLength (children)), children)
  135. {
  136. }
  137. static int GetMaxTitleLength (MenuItem[] children)
  138. {
  139. int maxLength = 0;
  140. foreach (var item in children) {
  141. int len = GetMenuBarItemLength (item.Title);
  142. if (len > maxLength)
  143. maxLength = len;
  144. item.IsFromSubMenu = true;
  145. }
  146. return maxLength;
  147. }
  148. void SetTitle (ustring title)
  149. {
  150. if (title == null)
  151. title = "";
  152. Title = title;
  153. TitleLength = GetMenuBarItemLength(Title);
  154. }
  155. static int GetMenuBarItemLength(ustring title)
  156. {
  157. int len = 0;
  158. foreach (var ch in title) {
  159. if (ch == '_')
  160. continue;
  161. len++;
  162. }
  163. return len;
  164. }
  165. /// <summary>
  166. /// Gets or sets the title to display.
  167. /// </summary>
  168. /// <value>The title.</value>
  169. public ustring Title { get; set; }
  170. /// <summary>
  171. /// Gets or sets the children for this MenuBarItem
  172. /// </summary>
  173. /// <value>The children.</value>
  174. public MenuItem [] Children { get; set; }
  175. internal int TitleLength { get; private set; }
  176. }
  177. class Menu : View {
  178. internal MenuBarItem barItems;
  179. MenuBar host;
  180. internal int current;
  181. internal View previousSubFocused;
  182. static Rect MakeFrame (int x, int y, MenuItem [] items)
  183. {
  184. int maxW = items.Max(z => z?.Width) ?? 0;
  185. return new Rect (x, y, maxW + 2, items.Length + 2);
  186. }
  187. public Menu (MenuBar host, int x, int y, MenuBarItem barItems) : base (MakeFrame (x, y, barItems.Children))
  188. {
  189. this.barItems = barItems;
  190. this.host = host;
  191. current = -1;
  192. for (int i = 0; i < barItems.Children.Length; i++) {
  193. if (barItems.Children[i] != null) {
  194. current = i;
  195. break;
  196. }
  197. }
  198. ColorScheme = Colors.Menu;
  199. CanFocus = true;
  200. WantMousePositionReports = host.WantMousePositionReports;
  201. }
  202. internal Attribute DetermineColorSchemeFor (MenuItem item, int index)
  203. {
  204. if (item != null) {
  205. if (index == current) return ColorScheme.Focus;
  206. if (!item.IsEnabled ()) return ColorScheme.Disabled;
  207. }
  208. return ColorScheme.Normal;
  209. }
  210. public override void Redraw (Rect region)
  211. {
  212. Driver.SetAttribute (ColorScheme.Normal);
  213. DrawFrame (region, padding: 0, fill: true);
  214. for (int i = 0; i < barItems.Children.Length; i++) {
  215. var item = barItems.Children [i];
  216. Driver.SetAttribute (item == null ? ColorScheme.Normal : i == current ? ColorScheme.Focus : ColorScheme.Normal);
  217. if (item == null) {
  218. Move (0, i + 1);
  219. Driver.AddRune (Driver.LeftTee);
  220. } else
  221. Move (1, i + 1);
  222. Driver.SetAttribute (DetermineColorSchemeFor (item, i));
  223. for (int p = 0; p < Frame.Width - 2; p++)
  224. if (item == null)
  225. Driver.AddRune (Driver.HLine);
  226. else if (p == Frame.Width - 3 && barItems.Children [i].SubMenu != null)
  227. Driver.AddRune ('>');
  228. else
  229. Driver.AddRune (' ');
  230. if (item == null) {
  231. Move (Frame.Right - 1, i + 1);
  232. Driver.AddRune (Driver.RightTee);
  233. continue;
  234. }
  235. Move (2, i + 1);
  236. if (!item.IsEnabled ())
  237. DrawHotString (item.Title, ColorScheme.Disabled, ColorScheme.Disabled);
  238. else
  239. DrawHotString (item.Title,
  240. i == current ? ColorScheme.HotFocus : ColorScheme.HotNormal,
  241. i == current ? ColorScheme.Focus : ColorScheme.Normal);
  242. // The help string
  243. var l = item.Help.Length;
  244. Move (Frame.Width - l - 2, 1 + i);
  245. Driver.AddStr (item.Help);
  246. }
  247. PositionCursor ();
  248. }
  249. public override void PositionCursor ()
  250. {
  251. if (!host.isMenuClosed)
  252. Move (2, 1 + current);
  253. else
  254. host.PositionCursor ();
  255. }
  256. void Run (Action action)
  257. {
  258. if (action == null)
  259. return;
  260. Application.UngrabMouse ();
  261. host.CloseAllMenus ();
  262. Application.Refresh ();
  263. Application.MainLoop.AddIdle (() => {
  264. action ();
  265. return false;
  266. });
  267. }
  268. public override bool ProcessKey (KeyEvent kb)
  269. {
  270. bool disabled;
  271. switch (kb.Key) {
  272. case Key.CursorUp:
  273. if (current == -1)
  274. break;
  275. do {
  276. disabled = false;
  277. current--;
  278. if (host.UseKeysUpDownAsKeysLeftRight) {
  279. if (current == -1 && barItems.Children [current + 1].IsFromSubMenu && host.selectedSub > -1) {
  280. current++;
  281. host.PreviousMenu (true);
  282. break;
  283. }
  284. }
  285. if (current < 0)
  286. current = barItems.Children.Length - 1;
  287. var item = barItems.Children [current];
  288. if (item == null || !item.IsEnabled ()) disabled = true;
  289. } while (barItems.Children [current] == null || disabled);
  290. SetNeedsDisplay ();
  291. break;
  292. case Key.CursorDown:
  293. do {
  294. current++;
  295. disabled = false;
  296. if (current == barItems.Children.Length)
  297. current = 0;
  298. var item = barItems.Children [current];
  299. if (item == null || !item.IsEnabled ()) disabled = true;
  300. if (host.UseKeysUpDownAsKeysLeftRight && barItems.Children [current]?.SubMenu != null &&
  301. !disabled && !host.isMenuClosed) {
  302. CheckSubMenu ();
  303. break;
  304. }
  305. if (host.isMenuClosed)
  306. host.OpenMenu (host.selected);
  307. } while (barItems.Children [current] == null || disabled);
  308. SetNeedsDisplay ();
  309. break;
  310. case Key.CursorLeft:
  311. host.PreviousMenu (true);
  312. break;
  313. case Key.CursorRight:
  314. host.NextMenu (barItems.Children [current].IsFromSubMenu ? true : false);
  315. break;
  316. case Key.Esc:
  317. Application.UngrabMouse ();
  318. host.CloseAllMenus ();
  319. break;
  320. case Key.Enter:
  321. CheckSubMenu ();
  322. Run (barItems.Children [current].Action);
  323. break;
  324. default:
  325. // TODO: rune-ify
  326. if (Char.IsLetterOrDigit ((char)kb.KeyValue)) {
  327. var x = Char.ToUpper ((char)kb.KeyValue);
  328. foreach (var item in barItems.Children) {
  329. if (item == null) continue;
  330. if (item.IsEnabled () && item.HotKey == x) {
  331. host.CloseMenu ();
  332. Run (item.Action);
  333. return true;
  334. }
  335. }
  336. }
  337. break;
  338. }
  339. return true;
  340. }
  341. public override bool MouseEvent(MouseEvent me)
  342. {
  343. if (!host.handled && !host.HandleGrabView (me, this)) {
  344. return false;
  345. }
  346. host.handled = false;
  347. bool disabled;
  348. if (me.Flags == MouseFlags.Button1Clicked || me.Flags == MouseFlags.Button1Released) {
  349. disabled = false;
  350. if (me.Y < 1)
  351. return true;
  352. var meY = me.Y - 1;
  353. if (meY >= barItems.Children.Length)
  354. return true;
  355. var item = barItems.Children [meY];
  356. if (item == null || !item.IsEnabled ()) disabled = true;
  357. if (item != null && !disabled)
  358. Run (barItems.Children [meY].Action);
  359. return true;
  360. } else if (me.Flags == MouseFlags.Button1Pressed || me.Flags == MouseFlags.ReportMousePosition) {
  361. disabled = false;
  362. if (me.Y < 1)
  363. return true;
  364. if (me.Y - 1 >= barItems.Children.Length)
  365. return true;
  366. var item = barItems.Children [me.Y - 1];
  367. if (item == null || !item.IsEnabled ()) disabled = true;
  368. if (item != null && !disabled)
  369. current = me.Y - 1;
  370. HasFocus = true;
  371. SetNeedsDisplay ();
  372. CheckSubMenu ();
  373. return true;
  374. }
  375. return false;
  376. }
  377. internal void CheckSubMenu ()
  378. {
  379. if (barItems.Children [current] == null)
  380. return;
  381. var subMenu = barItems.Children [current].SubMenu;
  382. if (subMenu != null) {
  383. int pos = -1;
  384. if (host.openSubMenu != null)
  385. pos = host.openSubMenu.FindIndex (o => o?.barItems == subMenu);
  386. host.Activate (host.selected, pos, subMenu);
  387. } else if (host.openSubMenu != null && !barItems.Children [current].IsFromSubMenu)
  388. host.CloseMenu (false, true);
  389. }
  390. int GetSubMenuIndex (MenuBarItem subMenu)
  391. {
  392. int pos = -1;
  393. if (this != null && Subviews.Count > 0) {
  394. Menu v = null;
  395. foreach (var menu in Subviews) {
  396. if (((Menu)menu).barItems == subMenu)
  397. v = (Menu)menu;
  398. }
  399. if (v != null)
  400. pos = Subviews.IndexOf (v);
  401. }
  402. return pos;
  403. }
  404. }
  405. /// <summary>
  406. /// A menu bar for your application.
  407. /// </summary>
  408. public class MenuBar : View {
  409. /// <summary>
  410. /// The menus that were defined when the menubar was created. This can be updated if the menu is not currently visible.
  411. /// </summary>
  412. /// <value>The menu array.</value>
  413. public MenuBarItem [] Menus { get; set; }
  414. internal int selected;
  415. internal int selectedSub;
  416. Action action;
  417. /// <summary>
  418. /// Used for change the navigation key style.
  419. /// </summary>
  420. public bool UseKeysUpDownAsKeysLeftRight { get; set; } = true;
  421. /// <summary>
  422. /// Initializes a new instance of the <see cref="T:Terminal.Gui.MenuBar"/> class with the specified set of toplevel menu items.
  423. /// </summary>
  424. /// <param name="menus">Individual menu items, if one of those contains a null, then a separator is drawn.</param>
  425. public MenuBar (MenuBarItem [] menus) : base ()
  426. {
  427. X = 0;
  428. Y = 0;
  429. Width = Dim.Fill ();
  430. Height = 1;
  431. Menus = menus;
  432. CanFocus = false;
  433. selected = -1;
  434. selectedSub = -1;
  435. ColorScheme = Colors.Menu;
  436. WantMousePositionReports = true;
  437. isMenuClosed = true;
  438. }
  439. public override void Redraw (Rect region)
  440. {
  441. Move (0, 0);
  442. Driver.SetAttribute (Colors.Menu.Normal);
  443. for (int i = 0; i < Frame.Width; i++)
  444. Driver.AddRune (' ');
  445. Move (1, 0);
  446. int pos = 1;
  447. for (int i = 0; i < Menus.Length; i++) {
  448. var menu = Menus [i];
  449. Move (pos, 0);
  450. Attribute hotColor, normalColor;
  451. if (i == selected) {
  452. hotColor = i == selected ? ColorScheme.HotFocus : ColorScheme.HotNormal;
  453. normalColor = i == selected ? ColorScheme.Focus : ColorScheme.Normal;
  454. } else {
  455. hotColor = Colors.Base.Focus;
  456. normalColor = Colors.Base.Focus;
  457. }
  458. DrawHotString (" " + menu.Title + " " + " ", hotColor, normalColor);
  459. pos += menu.TitleLength + 3;
  460. }
  461. PositionCursor ();
  462. }
  463. public override void PositionCursor ()
  464. {
  465. int pos = 0;
  466. for (int i = 0; i < Menus.Length; i++) {
  467. if (i == selected) {
  468. pos++;
  469. if (!isMenuClosed)
  470. Move (pos, 0);
  471. else
  472. Move (pos + 1, 0);
  473. return;
  474. } else {
  475. if (!isMenuClosed)
  476. pos += Menus [i].TitleLength + 4;
  477. else
  478. pos += 2 + Menus [i].TitleLength + 1;
  479. }
  480. }
  481. Move (0, 0);
  482. }
  483. void Selected (MenuItem item)
  484. {
  485. // TODO: Running = false;
  486. action = item.Action;
  487. }
  488. public event EventHandler OnOpenMenu;
  489. public event EventHandler OnCloseMenu;
  490. internal Menu openMenu;
  491. Menu openCurrentMenu;
  492. internal List<Menu> openSubMenu;
  493. View previousFocused;
  494. internal bool isMenuOpening;
  495. internal bool isMenuClosing;
  496. internal bool isMenuClosed;
  497. public bool MenuOpen;
  498. View lastFocused;
  499. /// <summary>
  500. /// Get the lasted focused view before open the menu.
  501. /// </summary>
  502. public View LastFocused { get; private set; }
  503. internal void OpenMenu (int index, int sIndex = -1, MenuBarItem subMenu = null)
  504. {
  505. isMenuOpening = true;
  506. OnOpenMenu?.Invoke (this, null);
  507. int pos = 0;
  508. switch (subMenu) {
  509. case null:
  510. lastFocused = lastFocused ?? SuperView.MostFocused;
  511. if (openSubMenu != null)
  512. CloseMenu (false, true);
  513. if (openMenu != null)
  514. SuperView.Remove (openMenu);
  515. for (int i = 0; i < index; i++)
  516. pos += Menus [i].Title.Length + 2;
  517. openMenu = new Menu (this, pos, 1, Menus [index]);
  518. openCurrentMenu = openMenu;
  519. openCurrentMenu.previousSubFocused = openMenu;
  520. SuperView.Add (openMenu);
  521. SuperView.SetFocus (openMenu);
  522. break;
  523. default:
  524. if (openSubMenu == null)
  525. openSubMenu = new List<Menu> ();
  526. if (sIndex > -1) {
  527. RemoveSubMenu (sIndex);
  528. } else {
  529. var last = openSubMenu.Count > 0 ? openSubMenu.Last () : openMenu;
  530. openCurrentMenu = new Menu (this, last.Frame.Left + last.Frame.Width, last.Frame.Top + 1 + last.current, subMenu);
  531. openCurrentMenu.previousSubFocused = last.previousSubFocused;
  532. openSubMenu.Add (openCurrentMenu);
  533. SuperView.Add (openCurrentMenu);
  534. }
  535. selectedSub = openSubMenu.Count - 1;
  536. SuperView?.SetFocus (openCurrentMenu);
  537. break;
  538. }
  539. isMenuOpening = false;
  540. isMenuClosed = false;
  541. MenuOpen = true;
  542. }
  543. // Starts the menu from a hotkey
  544. void StartMenu ()
  545. {
  546. if (openMenu != null)
  547. return;
  548. selected = 0;
  549. SetNeedsDisplay ();
  550. previousFocused = SuperView.Focused;
  551. OpenMenu (selected);
  552. Application.GrabMouse (this);
  553. }
  554. // Activates the menu, handles either first focus, or activating an entry when it was already active
  555. // For mouse events.
  556. internal void Activate (int idx, int sIdx = -1, MenuBarItem subMenu = null)
  557. {
  558. selected = idx;
  559. selectedSub = sIdx;
  560. if (openMenu == null)
  561. previousFocused = SuperView.Focused;
  562. OpenMenu (idx, sIdx, subMenu);
  563. SetNeedsDisplay ();
  564. }
  565. internal void CloseMenu (bool reopen = false, bool isSubMenu = false)
  566. {
  567. isMenuClosing = true;
  568. OnCloseMenu?.Invoke (this, null);
  569. switch (isSubMenu) {
  570. case false:
  571. if (openMenu != null)
  572. SuperView.Remove (openMenu);
  573. SetNeedsDisplay ();
  574. if (previousFocused != null && openMenu != null && previousFocused.ToString () != openCurrentMenu.ToString ())
  575. previousFocused?.SuperView?.SetFocus (previousFocused);
  576. openMenu = null;
  577. if (lastFocused is Menu) {
  578. lastFocused = null;
  579. }
  580. LastFocused = lastFocused;
  581. lastFocused = null;
  582. if (LastFocused != null) {
  583. if (!reopen)
  584. selected = -1;
  585. LastFocused.SuperView?.SetFocus (LastFocused);
  586. } else {
  587. SuperView.SetFocus (this);
  588. isMenuClosed = true;
  589. PositionCursor ();
  590. }
  591. isMenuClosed = true;
  592. break;
  593. case true:
  594. selectedSub = -1;
  595. SetNeedsDisplay ();
  596. RemoveAllOpensSubMenus ();
  597. openCurrentMenu.previousSubFocused?.SuperView?.SetFocus (openCurrentMenu.previousSubFocused);
  598. openSubMenu = null;
  599. break;
  600. }
  601. isMenuClosing = false;
  602. MenuOpen = false;
  603. }
  604. void RemoveSubMenu (int index)
  605. {
  606. if (openSubMenu == null)
  607. return;
  608. for (int i = openSubMenu.Count - 1; i > index; i--) {
  609. isMenuClosing = true;
  610. if (openSubMenu.Count - 1 > 0)
  611. SuperView.SetFocus (openSubMenu [i - 1]);
  612. else
  613. SuperView.SetFocus (openMenu);
  614. if (openSubMenu != null) {
  615. SuperView.Remove (openSubMenu [i]);
  616. openSubMenu.Remove (openSubMenu [i]);
  617. }
  618. RemoveSubMenu (i);
  619. }
  620. if (openSubMenu.Count > 0)
  621. openCurrentMenu = openSubMenu.Last ();
  622. //if (openMenu.Subviews.Count == 0)
  623. // return;
  624. //if (index == 0) {
  625. // //SuperView.SetFocus (previousSubFocused);
  626. // FocusPrev ();
  627. // return;
  628. //}
  629. //for (int i = openMenu.Subviews.Count - 1; i > index; i--) {
  630. // isMenuClosing = true;
  631. // if (openMenu.Subviews.Count - 1 > 0)
  632. // SuperView.SetFocus (openMenu.Subviews [i - 1]);
  633. // else
  634. // SuperView.SetFocus (openMenu);
  635. // if (openMenu != null) {
  636. // Remove (openMenu.Subviews [i]);
  637. // openMenu.Remove (openMenu.Subviews [i]);
  638. // }
  639. // RemoveSubMenu (i);
  640. //}
  641. isMenuClosing = false;
  642. }
  643. internal void RemoveAllOpensSubMenus ()
  644. {
  645. if (openSubMenu != null) {
  646. foreach (var item in openSubMenu) {
  647. SuperView.Remove (item);
  648. }
  649. }
  650. }
  651. internal void CloseAllMenus ()
  652. {
  653. if (!isMenuOpening && !isMenuClosing) {
  654. if (openSubMenu != null)
  655. CloseMenu (false, true);
  656. CloseMenu ();
  657. if (LastFocused != null && LastFocused != this)
  658. selected = -1;
  659. }
  660. isMenuClosed = true;
  661. }
  662. View FindDeepestMenu (View view, ref int count)
  663. {
  664. count = count > 0 ? count : 0;
  665. foreach (var menu in view.Subviews) {
  666. if (menu is Menu) {
  667. count++;
  668. return FindDeepestMenu ((Menu)menu, ref count);
  669. }
  670. }
  671. return view;
  672. }
  673. internal void PreviousMenu (bool isSubMenu = false)
  674. {
  675. switch (isSubMenu) {
  676. case false:
  677. if (selected <= 0)
  678. selected = Menus.Length - 1;
  679. else
  680. selected--;
  681. if (selected > -1)
  682. CloseMenu (true, false);
  683. OpenMenu (selected);
  684. break;
  685. case true:
  686. if (selectedSub > -1) {
  687. selectedSub--;
  688. RemoveSubMenu (selectedSub);
  689. SetNeedsDisplay ();
  690. } else
  691. PreviousMenu ();
  692. break;
  693. }
  694. }
  695. internal void NextMenu (bool isSubMenu = false)
  696. {
  697. switch (isSubMenu) {
  698. case false:
  699. if (selected == -1)
  700. selected = 0;
  701. else if (selected + 1 == Menus.Length)
  702. selected = 0;
  703. else
  704. selected++;
  705. if (selected > -1)
  706. CloseMenu (true);
  707. OpenMenu (selected);
  708. break;
  709. case true:
  710. if (UseKeysUpDownAsKeysLeftRight) {
  711. CloseMenu (false, true);
  712. NextMenu ();
  713. } else {
  714. if ((selectedSub == -1 || openSubMenu == null || openSubMenu?.Count == selectedSub) && openCurrentMenu.barItems.Children [openCurrentMenu.current].SubMenu == null) {
  715. if (openSubMenu != null)
  716. CloseMenu (false, true);
  717. NextMenu ();
  718. } else if (openCurrentMenu.barItems.Children [openCurrentMenu.current].SubMenu != null ||
  719. !openCurrentMenu.barItems.Children [openCurrentMenu.current].IsFromSubMenu)
  720. selectedSub++;
  721. else
  722. return;
  723. SetNeedsDisplay ();
  724. openCurrentMenu.CheckSubMenu ();
  725. }
  726. break;
  727. }
  728. }
  729. internal bool FindAndOpenMenuByHotkey(KeyEvent kb)
  730. {
  731. int pos = 0;
  732. var c = ((uint)kb.Key & (uint)Key.CharMask);
  733. for (int i = 0; i < Menus.Length; i++)
  734. {
  735. // TODO: this code is duplicated, hotkey should be part of the MenuBarItem
  736. var mi = Menus[i];
  737. int p = mi.Title.IndexOf('_');
  738. if (p != -1 && p + 1 < mi.Title.Length) {
  739. if (mi.Title[p + 1] == c) {
  740. Application.GrabMouse (this);
  741. selected = i;
  742. OpenMenu (i);
  743. return true;
  744. }
  745. }
  746. }
  747. return false;
  748. }
  749. public override bool ProcessHotKey (KeyEvent kb)
  750. {
  751. if (kb.Key == Key.F9) {
  752. StartMenu ();
  753. return true;
  754. }
  755. if (kb.IsAlt)
  756. {
  757. if (FindAndOpenMenuByHotkey(kb)) return true;
  758. }
  759. var kc = kb.KeyValue;
  760. return base.ProcessHotKey (kb);
  761. }
  762. public override bool ProcessKey (KeyEvent kb)
  763. {
  764. switch (kb.Key) {
  765. case Key.CursorLeft:
  766. selected--;
  767. if (selected < 0)
  768. selected = Menus.Length - 1;
  769. break;
  770. case Key.CursorRight:
  771. selected = (selected + 1) % Menus.Length;
  772. break;
  773. case Key.Esc:
  774. case Key.ControlC:
  775. //TODO: Running = false;
  776. CloseMenu ();
  777. break;
  778. default:
  779. var key = kb.KeyValue;
  780. if ((key >= 'a' && key <= 'z') || (key >= 'A' && key <= 'Z') || (key >= '0' && key <= '9')) {
  781. char c = Char.ToUpper ((char)key);
  782. if (Menus [selected].Children == null)
  783. return false;
  784. foreach (var mi in Menus [selected].Children) {
  785. int p = mi.Title.IndexOf ('_');
  786. if (p != -1 && p + 1 < mi.Title.Length) {
  787. if (mi.Title [p + 1] == c) {
  788. Selected (mi);
  789. return true;
  790. }
  791. }
  792. }
  793. }
  794. return false;
  795. }
  796. SetNeedsDisplay ();
  797. return true;
  798. }
  799. public override bool MouseEvent(MouseEvent me)
  800. {
  801. if (!handled && !HandleGrabView (me, this)) {
  802. return false;
  803. }
  804. handled = false;
  805. if (me.Flags == MouseFlags.Button1Clicked ||
  806. (me.Flags == MouseFlags.ReportMousePosition && selected > -1)) {
  807. int pos = 1;
  808. int cx = me.X;
  809. for (int i = 0; i < Menus.Length; i++) {
  810. if (cx > pos && me.X < pos + 1 + Menus [i].TitleLength) {
  811. if (selected == i && me.Flags == MouseFlags.Button1Clicked && !isMenuClosed) {
  812. Application.UngrabMouse ();
  813. CloseMenu ();
  814. } else if (me.Flags == MouseFlags.Button1Clicked && isMenuClosed) {
  815. Activate (i);
  816. }
  817. else if (selected != i && selected > -1 && me.Flags == MouseFlags.ReportMousePosition) {
  818. if (!isMenuClosed) {
  819. CloseMenu ();
  820. Activate (i);
  821. }
  822. } else {
  823. if (!isMenuClosed)
  824. Activate (i);
  825. }
  826. return true;
  827. }
  828. pos += 2 + Menus [i].TitleLength + 1;
  829. }
  830. }
  831. return false;
  832. }
  833. internal bool handled;
  834. internal bool HandleGrabView (MouseEvent me, View current)
  835. {
  836. if (Application.mouseGrabView != null) {
  837. if (me.View is MenuBar || me.View is Menu) {
  838. if(me.View != current) {
  839. Application.UngrabMouse ();
  840. Application.GrabMouse (me.View);
  841. me.View.MouseEvent (me);
  842. }
  843. } else if (!(me.View is MenuBar || me.View is Menu) && me.Flags.HasFlag (MouseFlags.Button1Clicked)) {
  844. Application.UngrabMouse ();
  845. CloseAllMenus ();
  846. handled = false;
  847. return false;
  848. } else {
  849. handled = false;
  850. return false;
  851. }
  852. } else if (isMenuClosed && me.Flags.HasFlag (MouseFlags.Button1Clicked)) {
  853. Application.GrabMouse (current);
  854. } else {
  855. handled = false;
  856. return false;
  857. }
  858. //if (me.View != this && me.Flags != MouseFlags.Button1Clicked)
  859. // return true;
  860. //else if (me.View != this && me.Flags == MouseFlags.Button1Clicked) {
  861. // Application.UngrabMouse ();
  862. // host.CloseAllMenus ();
  863. // return true;
  864. //}
  865. //if (!(me.View is MenuBar) && !(me.View is Menu) && me.Flags != MouseFlags.Button1Clicked)
  866. // return false;
  867. //if (Application.mouseGrabView != null) {
  868. // if (me.View is MenuBar || me.View is Menu) {
  869. // me.X -= me.OfX;
  870. // me.Y -= me.OfY;
  871. // me.View.MouseEvent (me);
  872. // return true;
  873. // } else if (!(me.View is MenuBar || me.View is Menu) && me.Flags == MouseFlags.Button1Clicked) {
  874. // Application.UngrabMouse ();
  875. // CloseAllMenus ();
  876. // }
  877. //} else if (!isMenuClosed && selected == -1 && me.Flags == MouseFlags.Button1Clicked) {
  878. // Application.GrabMouse (this);
  879. // return true;
  880. //}
  881. //if (Application.mouseGrabView != null) {
  882. // if (Application.mouseGrabView == me.View && me.View == current) {
  883. // me.X -= me.OfX;
  884. // me.Y -= me.OfY;
  885. // } else if (me.View != current && me.View is MenuBar && me.View is Menu) {
  886. // Application.UngrabMouse ();
  887. // Application.GrabMouse (me.View);
  888. // } else if (me.Flags == MouseFlags.Button1Clicked) {
  889. // Application.UngrabMouse ();
  890. // CloseMenu ();
  891. // }
  892. //} else if ((!isMenuClosed && selected > -1)) {
  893. // Application.GrabMouse (current);
  894. //}
  895. handled = true;
  896. return true;
  897. }
  898. }
  899. }