ShiftOS-C-/source/ShiftUI/Menu/Menu.cs
MichaelTheShifter d40fed5ce2 Move ShiftUI source code to ShiftOS
This'll be a lot easier to work on.
2016-07-20 09:40:36 -04:00

621 lines
14 KiB
C#

// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//
// Copyright (c) 2004-2005 Novell, Inc.
//
// Authors:
// Jordi Mas i Hernandez, jordi@ximian.com
//
// TODO:
// - FindMenuItem
// - MdiListItem
//
using System.Collections;
using System.ComponentModel;
using System.ComponentModel.Design;
using System.Reflection;
using System.Runtime.InteropServices;
using System.Collections.Generic;
using System.Drawing;
using System;
namespace ShiftUI
{
[ToolboxItemFilter("ShiftUI", ToolboxItemFilterType.Allow)]
[ListBindable(false)]
public abstract class Menu : Component
{
internal MenuItemCollection menu_items;
internal IntPtr menu_handle = IntPtr.Zero;
internal Menu parent_menu = null;
System.Drawing.Rectangle rect = new Rectangle ();
// UIA Framework Note: Used to keep track of expanded menus
internal Widget Wnd;
internal MenuTracker tracker;
private string control_name;
private object control_tag;
public const int FindHandle = 0;
public const int FindShortcut = 1;
protected Menu (MenuItem[] items)
{
menu_items = new MenuItemCollection (this);
if (items != null)
menu_items.AddRange (items);
}
#region Public Properties
[BrowsableAttribute(false)]
//[EditorBrowsableAttribute(EditorBrowsableState.Advanced)]
//[DesignerSerializationVisibilityAttribute(DesignerSerializationVisibility.Hidden)]
public IntPtr Handle {
get { return menu_handle; }
}
internal virtual void OnMenuChanged (EventArgs e)
{
EventHandler eh = (EventHandler)(Events [MenuChangedEvent]);
if (eh != null)
eh (this, e);
}
[BrowsableAttribute(false)]
//[DesignerSerializationVisibilityAttribute(DesignerSerializationVisibility.Hidden)]
public virtual bool IsParent {
get {
if (menu_items != null && menu_items.Count > 0)
return true;
else
return false;
}
}
[BrowsableAttribute(false)]
//[DesignerSerializationVisibilityAttribute(DesignerSerializationVisibility.Hidden)]
public MenuItem MdiListItem {
get {
throw new NotImplementedException ();
}
}
[BrowsableAttribute(false)]
//[DesignerSerializationVisibilityAttribute(DesignerSerializationVisibility.Content)]
[MergableProperty(false)]
public MenuItemCollection MenuItems {
get { return menu_items; }
}
[BrowsableAttribute(false)]
//[DesignerSerializationVisibilityAttribute(DesignerSerializationVisibility.Hidden)]
public string Name {
get { return control_name; }
set { control_name = value; }
}
[Localizable(false)]
[Bindable(true)]
[TypeConverter(typeof(StringConverter))]
[DefaultValue(null)]
[MWFCategory("Data")]
public object Tag {
get { return control_tag; }
set { control_tag = value; }
}
#endregion Public Properties
#region Private Properties
// UIA Framework Note: Used to obtain menu bounds
internal Rectangle Rect {
get { return rect; }
}
internal MenuItem SelectedItem {
get {
foreach (MenuItem item in MenuItems)
if (item.Selected)
return item;
return null;
}
}
internal int Height {
get { return rect.Height; }
set { rect.Height = value; }
}
internal int Width {
get { return rect.Width; }
set { rect.Width = value; }
}
internal int X {
get { return rect.X; }
set { rect.X = value; }
}
internal int Y {
get { return rect.Y; }
set { rect.Y = value; }
}
internal MenuTracker Tracker {
get {
Menu top = this;
while (top.parent_menu != null)
top = top.parent_menu;
return top.tracker;
}
}
#endregion Private Properties
#region Public Methods
protected void CloneMenu (Menu menuSrc)
{
Dispose (true);
menu_items = new MenuItemCollection (this);
for (int i = 0; i < menuSrc.MenuItems.Count ; i++)
menu_items.Add (menuSrc.MenuItems [i].CloneMenu ());
}
protected virtual IntPtr CreateMenuHandle ()
{
return IntPtr.Zero;
}
protected override void Dispose (bool disposing)
{
if (disposing) {
if (menu_items != null) {
// MenuItem.Dispose removes the item from the list
while (menu_items.Count > 0) {
menu_items [0].Dispose ();
}
}
if (menu_handle != IntPtr.Zero) {
menu_handle = IntPtr.Zero;
}
}
}
// From Microsoft documentation is impossible to guess that
// this method is supossed to do
//
// update: according to MS documentation, first parameter is on of this
// constant values FindHandle or FindShortcut, value depends from what
// you what to search, by shortcut or handle. FindHandle and FindShortcut
// is a constant fields and was defined for this class.
public MenuItem FindMenuItem (int type, IntPtr value)
{
return null;
}
protected int FindMergePosition (int mergeOrder)
{
int cnt = MenuItems.Count, cur, pos;
for (pos = 0; pos < cnt; ) {
cur = (pos + cnt) /2;
if (MenuItems[cur].MergeOrder > mergeOrder) {
cnt = cur;
} else {
pos = cur +1;
}
}
return pos;
}
public MainMenu GetMainMenu ()
{
for (Menu item = this; item != null; item = item.parent_menu) {
if (item is MainMenu) {
return (MainMenu) item;
}
}
return null;
}
internal virtual void InvalidateItem (MenuItem item)
{
if (Wnd != null)
Wnd.Invalidate (item.bounds);
}
public virtual void MergeMenu (Menu menuSrc)
{
if (menuSrc == this)
throw new ArgumentException ("The menu cannot be merged with itself");
if (menuSrc == null)
return;
for (int i = 0; i < menuSrc.MenuItems.Count; i++) {
MenuItem sourceitem = menuSrc.MenuItems[i];
switch (sourceitem.MergeType) {
case MenuMerge.Remove: // Item not included
break;
case MenuMerge.Add:
{
int pos = FindMergePosition (sourceitem.MergeOrder);
MenuItems.Add (pos, sourceitem.CloneMenu ());
break;
}
case MenuMerge.Replace:
case MenuMerge.MergeItems:
{
for (int pos = FindMergePosition (sourceitem.MergeOrder-1); pos <= MenuItems.Count; pos++) {
if ((pos >= MenuItems.Count) || (MenuItems[pos].MergeOrder != sourceitem.MergeOrder)) {
MenuItems.Add (pos, sourceitem.CloneMenu ());
break;
}
MenuItem mergeitem = MenuItems[pos];
if (mergeitem.MergeType != MenuMerge.Add) {
if ((sourceitem.MergeType == MenuMerge.MergeItems) && (mergeitem.MergeType == MenuMerge.MergeItems)) {
mergeitem.MergeMenu (sourceitem);
} else {
MenuItems.Remove (sourceitem);
MenuItems.Add (pos, sourceitem.CloneMenu ());
}
break;
}
}
break;
}
default:
break;
}
}
}
protected internal virtual bool ProcessCmdKey (ref Message msg, Keys keyData)
{
if (tracker == null)
return false;
return tracker.ProcessKeys (ref msg, keyData);
}
public override string ToString ()
{
return base.ToString () + ", Items.Count: " + MenuItems.Count;
}
#endregion Public Methods
static object MenuChangedEvent = new object ();
// UIA Framework Note: Used to track changes in MenuItemCollection
internal event EventHandler MenuChanged {
add { Events.AddHandler (MenuChangedEvent, value); }
remove { Events.RemoveHandler (MenuChangedEvent, value); }
}
[ListBindable(false)]
public class MenuItemCollection : IList, ICollection, IEnumerable
{
private Menu owner;
private ArrayList items = new ArrayList ();
public MenuItemCollection (Menu owner)
{
this.owner = owner;
}
#region Public Properties
public int Count {
get { return items.Count;}
}
public bool IsReadOnly {
get { return false; }
}
bool ICollection.IsSynchronized {
get { return false;}
}
object ICollection.SyncRoot {
get { return this;}
}
bool IList.IsFixedSize {
get { return false;}
}
public virtual MenuItem this [int index] {
get {
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException ("Index of out range");
return (MenuItem) items[index];
}
}
public virtual MenuItem this [string key] {
get {
if (string.IsNullOrEmpty (key))
return null;
foreach (MenuItem m in items)
if (string.Compare (m.Name, key, true) == 0)
return m;
return null;
}
}
object IList.this[int index] {
get { return items[index]; }
set { throw new NotSupportedException (); }
}
#endregion Public Properties
#region Public Methods
public virtual int Add (MenuItem item)
{
if (item.Parent != null)
item.Parent.MenuItems.Remove (item);
items.Add (item);
item.Index = items.Count - 1;
UpdateItem (item);
owner.OnMenuChanged (EventArgs.Empty);
if (owner.parent_menu != null)
owner.parent_menu.OnMenuChanged (EventArgs.Empty);
return items.Count - 1;
}
internal void AddNoEvents (MenuItem mi)
{
if (mi.Parent != null)
mi.Parent.MenuItems.Remove (mi);
items.Add (mi);
mi.Index = items.Count - 1;
mi.parent_menu = owner;
}
public virtual MenuItem Add (string caption)
{
MenuItem item = new MenuItem (caption);
Add (item);
return item;
}
public virtual int Add (int index, MenuItem item)
{
if (index < 0 || index > Count)
throw new ArgumentOutOfRangeException ("Index of out range");
ArrayList new_items = new ArrayList (Count + 1);
for (int i = 0; i < index; i++)
new_items.Add (items[i]);
new_items.Add (item);
for (int i = index; i < Count; i++)
new_items.Add (items[i]);
items = new_items;
UpdateItemsIndices ();
UpdateItem (item);
return index;
}
private void UpdateItem (MenuItem mi)
{
mi.parent_menu = owner;
owner.OnMenuChanged (EventArgs.Empty);
if (owner.parent_menu != null)
owner.parent_menu.OnMenuChanged (EventArgs.Empty);
if (owner.Tracker != null)
owner.Tracker.AddShortcuts (mi);
}
internal void Insert (int index, MenuItem mi)
{
if (index < 0 || index > Count)
throw new ArgumentOutOfRangeException ("Index of out range");
items.Insert (index, mi);
UpdateItemsIndices ();
UpdateItem (mi);
}
public virtual MenuItem Add (string caption, EventHandler onClick)
{
MenuItem item = new MenuItem (caption, onClick);
Add (item);
return item;
}
public virtual MenuItem Add (string caption, MenuItem[] items)
{
MenuItem item = new MenuItem (caption, items);
Add (item);
return item;
}
public virtual void AddRange (MenuItem[] items)
{
if (items == null)
throw new ArgumentNullException ("items");
foreach (MenuItem mi in items)
Add (mi);
}
public virtual void Clear ()
{
MenuTracker tracker = owner.Tracker;
foreach (MenuItem item in items) {
if (tracker != null)
tracker.RemoveShortcuts (item);
item.parent_menu = null;
}
items.Clear ();
owner.OnMenuChanged (EventArgs.Empty);
}
public bool Contains (MenuItem value)
{
return items.Contains (value);
}
public virtual bool ContainsKey (string key)
{
return !(this[key] == null);
}
public void CopyTo (Array dest, int index)
{
items.CopyTo (dest, index);
}
public MenuItem[] Find (string key, bool searchAllChildren)
{
if (string.IsNullOrEmpty (key))
throw new ArgumentNullException ("key");
List<MenuItem> list = new List<MenuItem> ();
foreach (MenuItem m in items)
if (string.Compare (m.Name, key, true) == 0)
list.Add (m);
if (searchAllChildren)
foreach (MenuItem m in items)
list.AddRange (m.MenuItems.Find (key, true));
return list.ToArray ();
}
public IEnumerator GetEnumerator ()
{
return items.GetEnumerator ();
}
int IList.Add (object value)
{
return Add ((MenuItem)value);
}
bool IList.Contains (object value)
{
return Contains ((MenuItem)value);
}
int IList.IndexOf (object value)
{
return IndexOf ((MenuItem)value);
}
void IList.Insert (int index, object value)
{
Insert (index, (MenuItem) value);
}
void IList.Remove (object value)
{
Remove ((MenuItem) value);
}
public int IndexOf (MenuItem value)
{
return items.IndexOf (value);
}
public virtual int IndexOfKey (string key)
{
if (string.IsNullOrEmpty (key))
return -1;
return IndexOf (this[key]);
}
public virtual void Remove (MenuItem item)
{
RemoveAt (item.Index);
}
public virtual void RemoveAt (int index)
{
if (index < 0 || index >= Count)
throw new ArgumentOutOfRangeException ("Index of out range");
MenuItem item = (MenuItem) items [index];
MenuTracker tracker = owner.Tracker;
if (tracker != null)
tracker.RemoveShortcuts (item);
item.parent_menu = null;
items.RemoveAt (index);
UpdateItemsIndices ();
owner.OnMenuChanged (EventArgs.Empty);
}
public virtual void RemoveByKey (string key)
{
Remove (this[key]);
}
#endregion Public Methods
#region Private Methods
private void UpdateItemsIndices ()
{
for (int i = 0; i < Count; i++) // Recalculate indeces
((MenuItem)items[i]).Index = i;
}
#endregion Private Methods
}
}
}