Menu.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578
  1. // Permission is hereby granted, free of charge, to any person obtaining
  2. // a copy of this software and associated documentation files (the
  3. // "Software"), to deal in the Software without restriction, including
  4. // without limitation the rights to use, copy, modify, merge, publish,
  5. // distribute, sublicense, and/or sell copies of the Software, and to
  6. // permit persons to whom the Software is furnished to do so, subject to
  7. // the following conditions:
  8. //
  9. // The above copyright notice and this permission notice shall be
  10. // included in all copies or substantial portions of the Software.
  11. //
  12. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  13. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  14. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  15. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  16. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  17. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  18. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  19. //
  20. // Copyright (c) 2004-2005 Novell, Inc.
  21. //
  22. // Authors:
  23. // Jordi Mas i Hernandez, [email protected]
  24. //
  25. // TODO:
  26. // - FindMenuItem
  27. // - MdiListItem
  28. //
  29. using System.Collections;
  30. using System.ComponentModel;
  31. using System.ComponentModel.Design;
  32. using System.Reflection;
  33. using System.Runtime.InteropServices;
  34. namespace System.Windows.Forms
  35. {
  36. #if !NET_2_0
  37. [Designer ("Microsoft.VisualStudio.Windows.Forms.MenuDesigner, " + Consts.AssemblyMicrosoft_VisualStudio, "System.ComponentModel.Design.IDesigner")]
  38. #endif
  39. [ToolboxItemFilter("System.Windows.Forms", ToolboxItemFilterType.Allow)]
  40. [ListBindable(false)]
  41. public abstract class Menu : Component
  42. {
  43. internal MenuItemCollection menu_items;
  44. internal IntPtr menu_handle = IntPtr.Zero;
  45. internal Menu parent_menu = null;
  46. System.Drawing.Rectangle rect;
  47. internal Control Wnd;
  48. internal MenuTracker tracker;
  49. #if NET_2_0
  50. private string control_name;
  51. private object control_tag;
  52. #endif
  53. public const int FindHandle = 0;
  54. public const int FindShortcut = 1;
  55. protected Menu (MenuItem[] items)
  56. {
  57. menu_items = new MenuItemCollection (this);
  58. if (items != null)
  59. menu_items.AddRange (items);
  60. }
  61. #region Public Properties
  62. [BrowsableAttribute(false)]
  63. [EditorBrowsableAttribute(EditorBrowsableState.Advanced)]
  64. [DesignerSerializationVisibilityAttribute(DesignerSerializationVisibility.Hidden)]
  65. public IntPtr Handle {
  66. get { return menu_handle; }
  67. }
  68. internal virtual void OnMenuChanged (EventArgs e)
  69. {
  70. EventHandler eh = (EventHandler)(Events [MenuChangedEvent]);
  71. if (eh != null)
  72. eh (this, e);
  73. }
  74. [BrowsableAttribute(false)]
  75. [DesignerSerializationVisibilityAttribute(DesignerSerializationVisibility.Hidden)]
  76. public virtual bool IsParent {
  77. get {
  78. if (menu_items != null && menu_items.Count > 0)
  79. return true;
  80. else
  81. return false;
  82. }
  83. }
  84. [BrowsableAttribute(false)]
  85. [DesignerSerializationVisibilityAttribute(DesignerSerializationVisibility.Hidden)]
  86. public MenuItem MdiListItem {
  87. get {
  88. throw new NotImplementedException ();
  89. }
  90. }
  91. [BrowsableAttribute(false)]
  92. [DesignerSerializationVisibilityAttribute(DesignerSerializationVisibility.Content)]
  93. [MergableProperty(false)]
  94. public MenuItemCollection MenuItems {
  95. get { return menu_items; }
  96. }
  97. #if NET_2_0
  98. [BrowsableAttribute(false)]
  99. [DesignerSerializationVisibilityAttribute(DesignerSerializationVisibility.Hidden)]
  100. public string Name {
  101. get { return control_name; }
  102. set { control_name = value; }
  103. }
  104. [Localizable(false)]
  105. [Bindable(true)]
  106. [TypeConverter(typeof(StringConverter))]
  107. [DefaultValue(null)]
  108. [MWFCategory("Data")]
  109. public object Tag {
  110. get { return control_tag; }
  111. set { control_tag = value; }
  112. }
  113. #endif
  114. #endregion Public Properties
  115. #region Private Properties
  116. internal System.Drawing.Rectangle Rect {
  117. get { return rect; }
  118. }
  119. internal MenuItem SelectedItem {
  120. get {
  121. foreach (MenuItem item in MenuItems)
  122. if (item.Selected)
  123. return item;
  124. return null;
  125. }
  126. }
  127. internal int Height {
  128. get { return rect.Height; }
  129. set { rect.Height = value; }
  130. }
  131. internal int Width {
  132. get { return rect.Width; }
  133. set { rect.Width = value; }
  134. }
  135. internal int X {
  136. get { return rect.X; }
  137. set { rect.X = value; }
  138. }
  139. internal int Y {
  140. get { return rect.Y; }
  141. set { rect.Y = value; }
  142. }
  143. internal MenuTracker Tracker {
  144. get {
  145. Menu top = this;
  146. while (top.parent_menu != null)
  147. top = top.parent_menu;
  148. return top.tracker;
  149. }
  150. }
  151. #endregion Private Properties
  152. #region Public Methods
  153. protected void CloneMenu (Menu menuSrc)
  154. {
  155. Dispose (true);
  156. menu_items = new MenuItemCollection (this);
  157. for (int i = 0; i < menuSrc.MenuItems.Count ; i++)
  158. menu_items.Add (menuSrc.MenuItems [i].CloneMenu ());
  159. }
  160. protected virtual IntPtr CreateMenuHandle ()
  161. {
  162. return IntPtr.Zero;
  163. }
  164. protected override void Dispose (bool disposing)
  165. {
  166. if (disposing) {
  167. if (menu_handle != IntPtr.Zero) {
  168. menu_handle = IntPtr.Zero;
  169. }
  170. }
  171. }
  172. // From Microsoft documentation is impossible to guess that
  173. // this method is supossed to do
  174. //
  175. // update: according to MS documentation, first parameter is on of this
  176. // constant values FindHandle or FindShortcut, value depends from what
  177. // you what to search, by shortcut or handle. FindHandle and FindShortcut
  178. // is a constant fields and was defined for this class.
  179. public MenuItem FindMenuItem (int type, IntPtr value)
  180. {
  181. return null;
  182. }
  183. protected int FindMergePosition (int mergeOrder)
  184. {
  185. int cnt = MenuItems.Count, cur, pos;
  186. for (pos = 0; pos < cnt; ) {
  187. cur = (pos + cnt) /2;
  188. if (MenuItems[cur].MergeOrder > mergeOrder) {
  189. cnt = cur;
  190. } else {
  191. pos = cur +1;
  192. }
  193. }
  194. return pos;
  195. }
  196. public ContextMenu GetContextMenu ()
  197. {
  198. for (Menu item = this; item != null; item = item.parent_menu) {
  199. if (item is ContextMenu) {
  200. return (ContextMenu) item;
  201. }
  202. }
  203. return null;
  204. }
  205. public MainMenu GetMainMenu ()
  206. {
  207. for (Menu item = this; item != null; item = item.parent_menu) {
  208. if (item is MainMenu) {
  209. return (MainMenu) item;
  210. }
  211. }
  212. return null;
  213. }
  214. internal virtual void InvalidateItem (MenuItem item)
  215. {
  216. if (Wnd != null)
  217. Wnd.Invalidate (item.bounds);
  218. }
  219. public virtual void MergeMenu (Menu menuSrc)
  220. {
  221. if (menuSrc == this)
  222. throw new ArgumentException ("The menu cannot be merged with itself");
  223. for (int i = 0; i < menuSrc.MenuItems.Count; i++) {
  224. MenuItem sourceitem = menuSrc.MenuItems[i];
  225. switch (sourceitem.MergeType) {
  226. case MenuMerge.Remove: // Item not included
  227. break;
  228. case MenuMerge.Add:
  229. {
  230. int pos = FindMergePosition (sourceitem.MergeOrder);
  231. MenuItems.Add (pos, sourceitem.CloneMenu ());
  232. break;
  233. }
  234. case MenuMerge.Replace:
  235. case MenuMerge.MergeItems:
  236. {
  237. for (int pos = FindMergePosition (sourceitem.MergeOrder-1); pos <= MenuItems.Count; pos++) {
  238. if ((pos >= MenuItems.Count) || (MenuItems[pos].MergeOrder != sourceitem.MergeOrder)) {
  239. MenuItems.Add (pos, sourceitem.CloneMenu ());
  240. break;
  241. }
  242. MenuItem mergeitem = MenuItems[pos];
  243. if (mergeitem.MergeType != MenuMerge.Add) {
  244. if ((sourceitem.MergeType == MenuMerge.MergeItems) && (mergeitem.MergeType == MenuMerge.MergeItems)) {
  245. mergeitem.MergeMenu (sourceitem);
  246. } else {
  247. MenuItems.Remove (sourceitem);
  248. MenuItems.Add (pos, sourceitem.CloneMenu ());
  249. }
  250. break;
  251. }
  252. }
  253. break;
  254. }
  255. default:
  256. break;
  257. }
  258. }
  259. }
  260. protected internal virtual bool ProcessCmdKey (ref Message msg, Keys keyData)
  261. {
  262. if (tracker == null)
  263. return false;
  264. return tracker.ProcessKeys (ref msg, keyData);
  265. }
  266. public override string ToString ()
  267. {
  268. return base.ToString () + ", Items.Count: " + MenuItems.Count;
  269. }
  270. #endregion Public Methods
  271. static object MenuChangedEvent = new object ();
  272. internal event EventHandler MenuChanged {
  273. add { Events.AddHandler (MenuChangedEvent, value); }
  274. remove { Events.RemoveHandler (MenuChangedEvent, value); }
  275. }
  276. [ListBindable(false)]
  277. public class MenuItemCollection : IList, ICollection, IEnumerable
  278. {
  279. private Menu owner;
  280. private ArrayList items = new ArrayList ();
  281. public MenuItemCollection (Menu owner)
  282. {
  283. this.owner = owner;
  284. }
  285. #region Public Properties
  286. public int Count {
  287. get { return items.Count;}
  288. }
  289. public bool IsReadOnly {
  290. get { return false; }
  291. }
  292. bool ICollection.IsSynchronized {
  293. get { return false;}
  294. }
  295. object ICollection.SyncRoot {
  296. get { return this;}
  297. }
  298. bool IList.IsFixedSize {
  299. get { return false;}
  300. }
  301. public virtual MenuItem this [int index] {
  302. get {
  303. if (index < 0 || index >= Count)
  304. throw new ArgumentOutOfRangeException ("Index of out range");
  305. return (MenuItem) items[index];
  306. }
  307. }
  308. object IList.this[int index] {
  309. get { return items[index]; }
  310. set { throw new NotSupportedException (); }
  311. }
  312. #endregion Public Properties
  313. #region Public Methods
  314. public virtual int Add (MenuItem mi)
  315. {
  316. if (mi.Parent != null)
  317. mi.Parent.MenuItems.Remove (mi);
  318. items.Add (mi);
  319. mi.Index = items.Count - 1;
  320. UpdateItem (mi);
  321. owner.OnMenuChanged (EventArgs.Empty);
  322. if (owner.parent_menu != null)
  323. owner.parent_menu.OnMenuChanged (EventArgs.Empty);
  324. return items.Count - 1;
  325. }
  326. internal void AddNoEvents (MenuItem mi)
  327. {
  328. if (mi.Parent != null)
  329. mi.Parent.MenuItems.Remove (mi);
  330. items.Add (mi);
  331. mi.Index = items.Count - 1;
  332. mi.parent_menu = owner;
  333. }
  334. public virtual MenuItem Add (string s)
  335. {
  336. MenuItem item = new MenuItem (s);
  337. Add (item);
  338. return item;
  339. }
  340. public virtual int Add (int index, MenuItem mi)
  341. {
  342. if (index < 0 || index > Count)
  343. throw new ArgumentOutOfRangeException ("Index of out range");
  344. ArrayList new_items = new ArrayList (Count + 1);
  345. for (int i = 0; i < index; i++)
  346. new_items.Add (items[i]);
  347. new_items.Add (mi);
  348. for (int i = index; i < Count; i++)
  349. new_items.Add (items[i]);
  350. items = new_items;
  351. UpdateItemsIndices ();
  352. UpdateItem (mi);
  353. return index;
  354. }
  355. private void UpdateItem (MenuItem mi)
  356. {
  357. mi.parent_menu = owner;
  358. owner.OnMenuChanged (EventArgs.Empty);
  359. if (owner.parent_menu != null)
  360. owner.parent_menu.OnMenuChanged (EventArgs.Empty);
  361. if (owner.Tracker != null)
  362. owner.Tracker.AddShortcuts (mi);
  363. }
  364. internal void Insert (int index, MenuItem mi)
  365. {
  366. if (index < 0 || index > Count)
  367. throw new ArgumentOutOfRangeException ("Index of out range");
  368. items.Insert (index, mi);
  369. UpdateItemsIndices ();
  370. UpdateItem (mi);
  371. }
  372. public virtual MenuItem Add (string s, EventHandler e)
  373. {
  374. MenuItem item = new MenuItem (s, e);
  375. Add (item);
  376. return item;
  377. }
  378. public virtual MenuItem Add (string s, MenuItem[] items)
  379. {
  380. MenuItem item = new MenuItem (s, items);
  381. Add (item);
  382. return item;
  383. }
  384. public virtual void AddRange (MenuItem[] items)
  385. {
  386. if (items == null)
  387. throw new ArgumentNullException ("items");
  388. foreach (MenuItem mi in items)
  389. Add (mi);
  390. }
  391. public virtual void Clear ()
  392. {
  393. MenuTracker tracker = owner.Tracker;
  394. foreach (MenuItem item in items) {
  395. if (tracker != null)
  396. tracker.RemoveShortcuts (item);
  397. item.parent_menu = null;
  398. }
  399. items.Clear ();
  400. owner.OnMenuChanged (EventArgs.Empty);
  401. }
  402. public bool Contains (MenuItem value)
  403. {
  404. return items.Contains (value);
  405. }
  406. public void CopyTo (Array dest, int index)
  407. {
  408. items.CopyTo (dest, index);
  409. }
  410. public IEnumerator GetEnumerator ()
  411. {
  412. return items.GetEnumerator ();
  413. }
  414. int IList.Add (object value)
  415. {
  416. return Add ((MenuItem)value);
  417. }
  418. bool IList.Contains (object value)
  419. {
  420. return Contains ((MenuItem)value);
  421. }
  422. int IList.IndexOf (object value)
  423. {
  424. return IndexOf ((MenuItem)value);
  425. }
  426. void IList.Insert (int index, object value)
  427. {
  428. Insert (index, (MenuItem) value);
  429. }
  430. void IList.Remove (object value)
  431. {
  432. Remove ((MenuItem) value);
  433. }
  434. public int IndexOf (MenuItem value)
  435. {
  436. return items.IndexOf (value);
  437. }
  438. public virtual void Remove (MenuItem item)
  439. {
  440. RemoveAt (item.Index);
  441. }
  442. public virtual void RemoveAt (int index)
  443. {
  444. if (index < 0 || index >= Count)
  445. throw new ArgumentOutOfRangeException ("Index of out range");
  446. MenuItem item = (MenuItem) items [index];
  447. MenuTracker tracker = owner.Tracker;
  448. if (tracker != null)
  449. tracker.RemoveShortcuts (item);
  450. item.parent_menu = null;
  451. items.RemoveAt (index);
  452. UpdateItemsIndices ();
  453. owner.OnMenuChanged (EventArgs.Empty);
  454. }
  455. #endregion Public Methods
  456. #region Private Methods
  457. private void UpdateItemsIndices ()
  458. {
  459. for (int i = 0; i < Count; i++) // Recalculate indeces
  460. ((MenuItem)items[i]).Index = i;
  461. }
  462. #endregion Private Methods
  463. }
  464. }
  465. }