ShiftOS-C-/source/ShiftUI/Internal/InternalWindowManager.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

1211 lines
29 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) 2005 Novell, Inc. (http://www.novell.com)
//
// Authors:
// Jackson Harper (jackson@ximian.com)
//
//
using System;
using System.Drawing;
using System.Runtime.InteropServices;
namespace ShiftUI {
internal abstract class InternalWindowManager {
private TitleButtons title_buttons;
internal Form form;
// moving windows
internal Point start;
internal State state;
protected Point clicked_point;
private FormPos sizing_edge;
internal Rectangle virtual_position;
private Rectangle normal_bounds;
private Rectangle iconic_bounds;
public enum State {
Idle,
Moving,
Sizing,
}
[Flags]
public enum FormPos {
None,
TitleBar = 1,
Top = 2,
Left = 4,
Right = 8,
Bottom = 16,
TopLeft = Top | Left,
TopRight = Top | Right,
BottomLeft = Bottom | Left,
BottomRight = Bottom | Right,
AnyEdge = Top | Left | Right | Bottom,
}
public InternalWindowManager (Form form)
{
this.form = form;
form.SizeChanged += new EventHandler (FormSizeChangedHandler);
title_buttons = new TitleButtons (form);
ThemeEngine.Current.ManagedWindowSetButtonLocations (this);
}
public Form Form {
get { return form; }
}
public int IconWidth {
get { return TitleBarHeight - 5; }
}
public TitleButtons TitleButtons {
get {
return title_buttons;
}
}
internal Rectangle NormalBounds {
get {
return normal_bounds;
}
set {
normal_bounds = value;
}
}
internal Size IconicSize {
get {
return SystemInformation.MinimizedWindowSize;
}
}
internal Rectangle IconicBounds {
get {
if (iconic_bounds == Rectangle.Empty)
return Rectangle.Empty;
Rectangle result = iconic_bounds;
result.Y = Form.Parent.ClientRectangle.Bottom - iconic_bounds.Y;
return result;
}
set {
iconic_bounds = value;
iconic_bounds.Y = Form.Parent.ClientRectangle.Bottom - iconic_bounds.Y;
}
}
internal virtual Rectangle MaximizedBounds {
get {
return Form.Parent.ClientRectangle;
}
}
public virtual void UpdateWindowState (FormWindowState old_window_state, FormWindowState new_window_state, bool force)
{
if (old_window_state == FormWindowState.Normal) {
NormalBounds = form.Bounds;
} else if (old_window_state == FormWindowState.Minimized) {
IconicBounds = form.Bounds;
}
switch (new_window_state) {
case FormWindowState.Minimized:
if (IconicBounds == Rectangle.Empty) {
Size size = IconicSize;
Point location = new Point (0, Form.Parent.ClientSize.Height - size.Height);
IconicBounds = new Rectangle (location, size);
}
form.Bounds = IconicBounds;
break;
case FormWindowState.Maximized:
form.Bounds = MaximizedBounds;
break;
case FormWindowState.Normal:
form.Bounds = NormalBounds;
break;
}
UpdateWindowDecorations (new_window_state);
form.ResetCursor ();
}
public virtual void UpdateWindowDecorations (FormWindowState window_state)
{
ThemeEngine.Current.ManagedWindowSetButtonLocations (this);
if (form.IsHandleCreated)
XplatUI.RequestNCRecalc (form.Handle);
}
public virtual bool WndProc (ref Message m)
{
#if debug
Console.WriteLine(DateTime.Now.ToLongTimeString () + " " + this.GetType () .Name + " (Handle={0},Text={1}) received message {2}", form.IsHandleCreated ? form.Handle : IntPtr.Zero, form.Text, m.ToString ());
#endif
switch ((Msg)m.Msg) {
// The mouse handling messages are actually
// not WM_NC* messages except for the first button and NCMOVEs
// down because we capture on the form
case Msg.WM_MOUSEMOVE:
return HandleMouseMove (form, ref m);
case Msg.WM_LBUTTONUP:
HandleLButtonUp (ref m);
break;
case Msg.WM_RBUTTONDOWN:
return HandleRButtonDown (ref m);
case Msg.WM_LBUTTONDOWN:
return HandleLButtonDown (ref m);
case Msg.WM_LBUTTONDBLCLK:
return HandleLButtonDblClick (ref m);
case Msg.WM_PARENTNOTIFY:
if (Widget.LowOrder(m.WParam.ToInt32()) == (int) Msg.WM_LBUTTONDOWN)
Activate ();
break;
case Msg.WM_NCHITTEST:
return HandleNCHitTest (ref m);
// Return true from these guys, otherwise win32 will mess up z-order
case Msg.WM_NCLBUTTONUP:
HandleNCLButtonUp (ref m);
return true;
case Msg.WM_NCLBUTTONDOWN:
HandleNCLButtonDown (ref m);
return true;
case Msg.WM_NCMOUSEMOVE:
HandleNCMouseMove (ref m);
return true;
case Msg.WM_NCLBUTTONDBLCLK:
HandleNCLButtonDblClick (ref m);
break;
case Msg.WM_NCMOUSELEAVE:
HandleNCMouseLeave (ref m);
break;
case Msg.WM_MOUSELEAVE:
HandleMouseLeave (ref m);
break;
case Msg.WM_NCCALCSIZE:
return HandleNCCalcSize (ref m);
case Msg.WM_NCPAINT:
return HandleNCPaint (ref m);
}
return false;
}
protected virtual bool HandleNCPaint (ref Message m)
{
PaintEventArgs pe = XplatUI.PaintEventStart (ref m, form.Handle, false);
Rectangle clip;
if (form.ActiveMenu != null) {
Point pnt;
pnt = GetMenuOrigin ();
// The entire menu has to be in the clip rectangle because the
// control buttons are right-aligned and otherwise they would
// stay painted when the window gets resized.
clip = new Rectangle (pnt.X, pnt.Y, form.ClientSize.Width, 0);
clip = Rectangle.Union (clip, pe.ClipRectangle);
pe.SetClip (clip);
pe.Graphics.SetClip (clip);
form.ActiveMenu.Draw (pe, new Rectangle (pnt.X, pnt.Y, form.ClientSize.Width, 0));
}
if (HasBorders || IsMinimized && !(Form.IsMdiChild && IsMaximized)) {
// clip region is not correct on win32.
// use the entire form's area.
clip = new Rectangle (0, 0, form.Width, form.Height);
ThemeEngine.Current.DrawManagedWindowDecorations (pe.Graphics, clip, this);
}
XplatUI.PaintEventEnd (ref m, form.Handle, false);
return true;
}
protected virtual bool HandleNCCalcSize (ref Message m)
{
XplatUIWin32.NCCALCSIZE_PARAMS ncp;
XplatUIWin32.RECT rect;
if (m.WParam == (IntPtr)1) {
ncp = (XplatUIWin32.NCCALCSIZE_PARAMS)Marshal.PtrToStructure (m.LParam,
typeof (XplatUIWin32.NCCALCSIZE_PARAMS));
ncp.rgrc1 = NCCalcSize (ncp.rgrc1);
Marshal.StructureToPtr (ncp, m.LParam, true);
} else {
rect = (XplatUIWin32.RECT) Marshal.PtrToStructure (m.LParam, typeof (XplatUIWin32.RECT));
rect = NCCalcSize (rect);
Marshal.StructureToPtr (rect, m.LParam, true);
}
return true;
}
protected virtual XplatUIWin32.RECT NCCalcSize (XplatUIWin32.RECT proposed_window_rect)
{
int bw = ThemeEngine.Current.ManagedWindowBorderWidth (this);
if (HasBorders) {
proposed_window_rect.top += TitleBarHeight + bw;
proposed_window_rect.bottom -= bw;
proposed_window_rect.left += bw;
proposed_window_rect.right -= bw;
}
if (XplatUI.RequiresPositiveClientAreaSize) {
// This is necessary for Linux, can't handle <= 0-sized
// client areas correctly.
if (proposed_window_rect.right <= proposed_window_rect.left) {
proposed_window_rect.right += proposed_window_rect.left - proposed_window_rect.right + 1;
}
if (proposed_window_rect.top >= proposed_window_rect.bottom) {
proposed_window_rect.bottom += proposed_window_rect.top - proposed_window_rect.bottom + 1;
}
}
return proposed_window_rect;
}
protected virtual bool HandleNCHitTest (ref Message m)
{
int x = Widget.LowOrder ((int)m.LParam.ToInt32 ());
int y = Widget.HighOrder ((int)m.LParam.ToInt32 ());
NCPointToClient (ref x, ref y);
FormPos pos = FormPosForCoords (x, y);
if (pos == FormPos.TitleBar) {
m.Result = new IntPtr ((int)HitTest.HTCAPTION);
return true;
}
if (!IsSizable)
return false;
switch (pos) {
case FormPos.Top:
m.Result = new IntPtr ((int)HitTest.HTTOP);
break;
case FormPos.Left:
m.Result = new IntPtr ((int)HitTest.HTLEFT);
break;
case FormPos.Right:
m.Result = new IntPtr ((int)HitTest.HTRIGHT);
break;
case FormPos.Bottom:
m.Result = new IntPtr ((int)HitTest.HTBOTTOM);
break;
case FormPos.TopLeft:
m.Result = new IntPtr ((int)HitTest.HTTOPLEFT);
break;
case FormPos.TopRight:
m.Result = new IntPtr ((int)HitTest.HTTOPRIGHT);
break;
case FormPos.BottomLeft:
m.Result = new IntPtr ((int)HitTest.HTBOTTOMLEFT);
break;
case FormPos.BottomRight:
m.Result = new IntPtr ((int)HitTest.HTBOTTOMRIGHT);
break;
default:
// We return false so that DefWndProc handles things
return false;
}
return true;
}
public virtual void UpdateBorderStyle (FormBorderStyle border_style)
{
if (form.IsHandleCreated) {
XplatUI.SetBorderStyle (form.Handle, border_style);
}
if (ShouldRemoveWindowManager (border_style)) {
form.RemoveWindowManager ();
return;
}
ThemeEngine.Current.ManagedWindowSetButtonLocations (this);
}
public virtual void SetWindowState (FormWindowState old_state, FormWindowState window_state)
{
UpdateWindowState (old_state, window_state, false);
}
public virtual FormWindowState GetWindowState ()
{
return form.window_state;
}
public virtual void PointToClient (ref int x, ref int y)
{
// toolwindows stay in screencoords we just have to make sure
// they obey the working area
Rectangle working = SystemInformation.WorkingArea;
if (x > working.Right)
x = working.Right;
if (x < working.Left)
x = working.Left;
if (y < working.Top)
y = working.Top;
if (y > working.Bottom)
y = working.Bottom;
}
public virtual void PointToScreen (ref int x, ref int y)
{
XplatUI.ClientToScreen (form.Handle, ref x, ref y);
}
protected virtual bool ShouldRemoveWindowManager (FormBorderStyle style)
{
return style != FormBorderStyle.FixedToolWindow && style != FormBorderStyle.SizableToolWindow;
}
public bool IconRectangleContains (int x, int y)
{
if (!ShowIcon)
return false;
Rectangle icon = ThemeEngine.Current.ManagedWindowGetTitleBarIconArea (this);
return icon.Contains (x, y);
}
public bool ShowIcon {
get {
if (!Form.ShowIcon)
return false;
if (!HasBorders)
return false;
if (IsMinimized)
return true;
if (IsToolWindow || Form.FormBorderStyle == FormBorderStyle.FixedDialog)
return false;
return true;
}
}
protected virtual void Activate ()
{
form.Invalidate (true);
form.Update ();
}
public virtual bool IsActive {
get {
return true;
}
}
private void FormSizeChangedHandler (object sender, EventArgs e)
{
if (form.IsHandleCreated) {
ThemeEngine.Current.ManagedWindowSetButtonLocations (this);
XplatUI.InvalidateNC (form.Handle);
}
}
protected virtual bool HandleRButtonDown (ref Message m)
{
Activate ();
return false;
}
protected virtual bool HandleLButtonDown (ref Message m)
{
Activate ();
return false;
}
protected virtual bool HandleLButtonDblClick(ref Message m)
{
return false;
}
protected virtual bool HandleNCMouseLeave (ref Message m)
{
int x = Widget.LowOrder ((int)m.LParam.ToInt32 ());
int y = Widget.HighOrder ((int)m.LParam.ToInt32 ());
NCPointToClient (ref x, ref y);
FormPos pos = FormPosForCoords (x, y);
if (pos != FormPos.TitleBar) {
HandleTitleBarLeave (x, y);
return true;
}
return true;
}
protected virtual bool HandleNCMouseMove (ref Message m)
{
int x = Widget.LowOrder((int)m.LParam.ToInt32( ));
int y = Widget.HighOrder((int)m.LParam.ToInt32( ));
NCPointToClient (ref x, ref y);
FormPos pos = FormPosForCoords (x, y);
if (pos == FormPos.TitleBar) {
HandleTitleBarMouseMove (x, y);
return true;
}
if (form.ActiveMenu != null && XplatUI.IsEnabled (form.Handle)) {
MouseEventArgs mea = new MouseEventArgs (Form.FromParamToMouseButtons (m.WParam.ToInt32 ()), form.mouse_clicks, x, y, 0);
form.ActiveMenu.OnMouseMove (form, mea);
}
return true;
}
protected virtual bool HandleNCLButtonDown (ref Message m)
{
Activate ();
start = Cursor.Position;
virtual_position = form.Bounds;
int x = Widget.LowOrder ((int) m.LParam.ToInt32 ());
int y = Widget.HighOrder ((int) m.LParam.ToInt32 ());
// Need to adjust because we are in NC land
NCPointToClient (ref x, ref y);
FormPos pos = FormPosForCoords (x, y);
if (form.ActiveMenu != null && XplatUI.IsEnabled (form.Handle)) {
MouseEventArgs mea = new MouseEventArgs (Form.FromParamToMouseButtons (m.WParam.ToInt32 ()), form.mouse_clicks, x, y - TitleBarHeight, 0);
form.ActiveMenu.OnMouseDown (form, mea);
}
if (pos == FormPos.TitleBar) {
HandleTitleBarDown (x, y);
return true;
}
if (IsSizable) {
if ((pos & FormPos.AnyEdge) == 0)
return false;
virtual_position = form.Bounds;
state = State.Sizing;
sizing_edge = pos;
form.Capture = true;
return true;
}
return false;
}
protected virtual void HandleNCLButtonDblClick (ref Message m)
{
int x = Widget.LowOrder ((int)m.LParam.ToInt32 ());
int y = Widget.HighOrder ((int)m.LParam.ToInt32 ());
// Need to adjust because we are in NC land
NCPointToClient (ref x, ref y);
FormPos pos = FormPosForCoords (x, y);
if (pos == FormPos.TitleBar || pos == FormPos.Top)
HandleTitleBarDoubleClick (x, y);
}
protected virtual void HandleTitleBarDoubleClick (int x, int y)
{
}
protected virtual void HandleTitleBarLeave (int x, int y)
{
title_buttons.MouseLeave (x, y);
}
protected virtual void HandleTitleBarMouseMove (int x, int y)
{
if (title_buttons.MouseMove (x, y))
XplatUI.InvalidateNC (form.Handle);
}
protected virtual void HandleTitleBarUp (int x, int y)
{
title_buttons.MouseUp (x, y);
return;
}
protected virtual void HandleTitleBarDown (int x, int y)
{
title_buttons.MouseDown (x, y);
if (!TitleButtons.AnyPushedTitleButtons && !IsMaximized) {
state = State.Moving;
clicked_point = new Point (x, y);
if (form.Parent != null) {
form.CaptureWithConfine (form.Parent);
} else {
form.Capture = true;
}
}
XplatUI.InvalidateNC (form.Handle);
}
private bool HandleMouseMove (Form form, ref Message m)
{
switch (state) {
case State.Moving:
HandleWindowMove (m);
return true;
case State.Sizing:
HandleSizing (m);
return true;
}
return false;
}
private void HandleMouseLeave (ref Message m)
{
form.ResetCursor ();
}
protected virtual void HandleWindowMove (Message m)
{
Point move = MouseMove (Cursor.Position);
UpdateVP (virtual_position.X + move.X, virtual_position.Y + move.Y,
virtual_position.Width, virtual_position.Height);
}
private void HandleSizing (Message m)
{
Rectangle pos = virtual_position;
int mw;
int mh;
if (IsToolWindow) {
int border_width = BorderWidth;
mw = 2 * (border_width + Theme.ManagedWindowSpacingAfterLastTitleButton) + ThemeEngine.Current.ManagedWindowButtonSize (this).Width;
mh = 2 * border_width + TitleBarHeight;
} else {
Size minimum_size = SystemInformation.MinWindowTrackSize;
mw = minimum_size.Width;
mh = minimum_size.Height;
}
int x = Cursor.Position.X;
int y = Cursor.Position.Y;
PointToClient (ref x, ref y);
if ((sizing_edge & FormPos.Top) != 0) {
if (pos.Bottom - y < mh)
y = pos.Bottom - mh;
pos.Height = pos.Bottom - y;
pos.Y = y;
} else if ((sizing_edge & FormPos.Bottom) != 0) {
int height = y - pos.Top;
if (height <= mh)
height = mh;
pos.Height = height;
}
if ((sizing_edge & FormPos.Left) != 0) {
if (pos.Right - x < mw)
x = pos.Right - mw;
pos.Width = pos.Right - x;
pos.X = x;
} else if ((sizing_edge & FormPos.Right) != 0) {
int width = x - form.Left;
if (width <= mw)
width = mw;
pos.Width = width;
}
UpdateVP (pos);
}
public bool IsMaximized {
get { return GetWindowState () == FormWindowState.Maximized; }
}
public bool IsMinimized {
get { return GetWindowState () == FormWindowState.Minimized; }
}
public bool IsSizable {
get {
switch (form.FormBorderStyle) {
case FormBorderStyle.Sizable:
case FormBorderStyle.SizableToolWindow:
return (form.window_state != FormWindowState.Minimized);
default:
return false;
}
}
}
public bool HasBorders {
get {
return form.FormBorderStyle != FormBorderStyle.None;
}
}
public bool IsToolWindow {
get {
if (form.FormBorderStyle == FormBorderStyle.SizableToolWindow ||
form.FormBorderStyle == FormBorderStyle.FixedToolWindow ||
form.GetCreateParams().IsSet (WindowExStyles.WS_EX_TOOLWINDOW))
return true;
return false;
}
}
public int TitleBarHeight {
get {
return ThemeEngine.Current.ManagedWindowTitleBarHeight (this);
}
}
public int BorderWidth {
get {
return ThemeEngine.Current.ManagedWindowBorderWidth (this);
}
}
public virtual int MenuHeight {
get {
return (form.Menu != null ? ThemeEngine.Current.MenuHeight : 0);
}
}
protected void UpdateVP (Rectangle r)
{
UpdateVP (r.X, r.Y, r.Width, r.Height);
}
protected void UpdateVP (Point loc, int w, int h)
{
UpdateVP (loc.X, loc.Y, w, h);
}
protected void UpdateVP (int x, int y, int w, int h)
{
virtual_position.X = x;
virtual_position.Y = y;
virtual_position.Width = w;
virtual_position.Height = h;
DrawVirtualPosition (virtual_position);
}
protected virtual void HandleLButtonUp (ref Message m)
{
if (state == State.Idle)
return;
ClearVirtualPosition ();
form.Capture = false;
if (state == State.Moving && form.Location != virtual_position.Location)
form.Location = virtual_position.Location;
else if (state == State.Sizing && form.Bounds != virtual_position)
form.Bounds = virtual_position;
state = State.Idle;
OnWindowFinishedMoving ();
}
private bool HandleNCLButtonUp (ref Message m)
{
if (form.Capture) {
ClearVirtualPosition ();
form.Capture = false;
state = State.Idle;
if (form.MdiContainer != null)
form.MdiContainer.SizeScrollBars();
}
int x = Widget.LowOrder ((int) m.LParam.ToInt32 ());
int y = Widget.HighOrder ((int) m.LParam.ToInt32 ());
NCPointToClient (ref x, ref y);
FormPos pos = FormPosForCoords (x, y);
if (pos == FormPos.TitleBar) {
HandleTitleBarUp (x, y);
return true;
}
return true;
}
protected void DrawTitleButton (Graphics dc, TitleButton button, Rectangle clip)
{
if (!button.Rectangle.IntersectsWith (clip))
return;
ThemeEngine.Current.ManagedWindowDrawMenuButton (dc, button, clip, this);
}
public virtual void DrawMaximizedButtons (object sender, PaintEventArgs pe)
{
}
protected Point MouseMove (Point pos)
{
return new Point (pos.X - start.X, pos.Y - start.Y);
}
protected virtual void DrawVirtualPosition (Rectangle virtual_position)
{
form.Bounds = virtual_position;
start = Cursor.Position;
}
protected virtual void ClearVirtualPosition ()
{
}
protected virtual void OnWindowFinishedMoving ()
{
}
protected virtual void NCPointToClient(ref int x, ref int y) {
form.PointToClient(ref x, ref y);
NCClientToNC (ref x, ref y);
}
protected virtual void NCClientToNC (ref int x, ref int y) {
y += TitleBarHeight;
y += BorderWidth;
y += MenuHeight;
}
internal Point GetMenuOrigin ()
{
return new Point (BorderWidth, BorderWidth + TitleBarHeight);
}
protected FormPos FormPosForCoords (int x, int y)
{
int bw = BorderWidth;
if (y < TitleBarHeight + bw) {
// Console.WriteLine ("A");
if (y > bw && x > bw &&
x < form.Width - bw)
return FormPos.TitleBar;
if (x < bw || (x < 20 && y < bw))
return FormPos.TopLeft;
if (x > form.Width - bw ||
(x > form.Width - 20 && y < bw))
return FormPos.TopRight;
if (y < bw)
return FormPos.Top;
} else if (y > form.Height - 20) {
// Console.WriteLine ("B");
if (x < bw ||
(x < 20 && y > form.Height - bw))
return FormPos.BottomLeft;
if (x > form.Width - (bw * 2) ||
(x > form.Width - 20 &&
y > form.Height - bw))
return FormPos.BottomRight;
if (y > form.Height - (bw * 2))
return FormPos.Bottom;
} else if (x < bw) {
// Console.WriteLine ("C");
return FormPos.Left;
} else if (x > form.Width - (bw * 2)) {
// Console.WriteLine ("D");
return FormPos.Right;
} else {
// Console.WriteLine ("E {0}", form.Width - bw);
}
return FormPos.None;
}
}
internal class TitleButton
{
public Rectangle Rectangle;
public ButtonState State;
public CaptionButton Caption;
private EventHandler Clicked;
public bool Visible;
bool entered;
public TitleButton (CaptionButton caption, EventHandler clicked)
{
Caption = caption;
Clicked = clicked;
}
public void OnClick ()
{
if (Clicked != null) {
Clicked (this, EventArgs.Empty);
}
}
public bool Entered {
get { return entered; }
set { entered = value; }
}
}
internal class TitleButtons : System.Collections.IEnumerable
{
public TitleButton MinimizeButton;
public TitleButton MaximizeButton;
public TitleButton RestoreButton;
public TitleButton CloseButton;
public TitleButton HelpButton;
public TitleButton [] AllButtons;
public bool Visible;
private ToolTip.ToolTipWindow tooltip;
private Timer tooltip_timer;
private TitleButton tooltip_hovered_button;
private TitleButton tooltip_hidden_button;
private const int tooltip_hide_interval = 3000;
private const int tooltip_show_interval = 1000;
private Form form;
public TitleButtons (Form frm)
{
this.form = frm;
this.Visible = true;
MinimizeButton = new TitleButton (CaptionButton.Minimize, new EventHandler (ClickHandler));
MaximizeButton = new TitleButton (CaptionButton.Maximize, new EventHandler (ClickHandler));
RestoreButton = new TitleButton (CaptionButton.Restore, new EventHandler (ClickHandler));
CloseButton = new TitleButton (CaptionButton.Close, new EventHandler (ClickHandler));
HelpButton = new TitleButton (CaptionButton.Help, new EventHandler (ClickHandler));
AllButtons = new TitleButton [] { MinimizeButton, MaximizeButton, RestoreButton, CloseButton, HelpButton };
}
private void ClickHandler (object sender, EventArgs e)
{
if (!Visible) {
return;
}
TitleButton button = (TitleButton) sender;
switch (button.Caption) {
case CaptionButton.Close:
form.Close ();
break;
case CaptionButton.Help:
Console.WriteLine ("Help not implemented.");
break;
case CaptionButton.Maximize:
form.WindowState = FormWindowState.Maximized;
break;
case CaptionButton.Minimize:
form.WindowState = FormWindowState.Minimized;
break;
case CaptionButton.Restore:
form.WindowState = FormWindowState.Normal;
break;
}
}
public TitleButton FindButton (int x, int y)
{
if (!Visible) {
return null;
}
foreach (TitleButton button in AllButtons) {
if (button.Visible && button.Rectangle.Contains (x, y)) {
return button;
}
}
return null;
}
public bool AnyPushedTitleButtons {
get {
if (!Visible) {
return false;
}
foreach (TitleButton button in AllButtons) {
if (button.Visible && button.State == ButtonState.Pushed) {
return true;
}
}
return false;
}
}
#region IEnumerable Members
public System.Collections.IEnumerator GetEnumerator ()
{
return AllButtons.GetEnumerator ();
}
#endregion
#region ToolTip helpers
// Called from MouseMove if mouse is over a button
public void ToolTipStart (TitleButton button)
{
tooltip_hovered_button = button;
if (tooltip_hovered_button == tooltip_hidden_button)
return;
tooltip_hidden_button = null;
if (tooltip != null && tooltip.Visible)
ToolTipShow (true);
if (tooltip_timer == null) {
tooltip_timer = new Timer ();
tooltip_timer.Tick += new EventHandler (ToolTipTimerTick);
}
tooltip_timer.Interval = tooltip_show_interval;
tooltip_timer.Start ();
tooltip_hovered_button = button;
}
public void ToolTipTimerTick (object sender, EventArgs e)
{
if (tooltip_timer.Interval == tooltip_hide_interval) {
tooltip_hidden_button = tooltip_hovered_button;
ToolTipHide (false);
} else {
ToolTipShow (false);
}
}
// Called from timer (with only_refresh = false)
// Called from ToolTipStart if tooltip is already shown (with only_refresh = true)
public void ToolTipShow (bool only_refresh)
{
if (!form.Visible)
return;
string text = String.Format (tooltip_hovered_button.Caption.ToString ());
tooltip_timer.Interval = tooltip_hide_interval;
tooltip_timer.Enabled = true;
if (only_refresh && (tooltip == null || !tooltip.Visible)) {
return;
}
if (tooltip == null)
tooltip = new ToolTip.ToolTipWindow ();
else if (tooltip.Text == text && tooltip.Visible)
return;
else if (tooltip.Visible)
tooltip.Visible = false;
if (form.WindowState == FormWindowState.Maximized && form.MdiParent != null)
tooltip.Present (form.MdiParent, text);
else
tooltip.Present (form, text);
}
// Called from MouseLeave (with reset_hidden_button = true)
// Called from MouseDown (with reset_hidden_button = false)
// Called from MouseMove if mouse isn't over any button (with reset_hidden_button = false)
// Called from Timer if hiding (with reset_hidden_button = false)
public void ToolTipHide (bool reset_hidden_button)
{
if (tooltip_timer != null)
tooltip_timer.Enabled = false;
if (tooltip != null && tooltip.Visible)
tooltip.Visible = false;
if (reset_hidden_button)
tooltip_hidden_button = null;
}
#endregion
public bool MouseMove (int x, int y)
{
if (!Visible) {
return false;
}
bool any_change = false;
bool any_pushed_buttons = AnyPushedTitleButtons;
bool any_tooltip = false;
TitleButton over_button = FindButton (x, y);
foreach (TitleButton button in this) {
if (button == null)
continue;
if (button.State == ButtonState.Inactive)
continue;
if (button == over_button) {
if (any_pushed_buttons) {
any_change |= button.State != ButtonState.Pushed;
button.State = ButtonState.Pushed;
}
ToolTipStart (button);
any_tooltip = true;
if (!button.Entered) {
button.Entered = true;
if (ThemeEngine.Current.ManagedWindowTitleButtonHasHotElementStyle (button, form))
any_change = true;
}
} else {
if (any_pushed_buttons) {
any_change |= button.State != ButtonState.Normal;
button.State = ButtonState.Normal;
}
if (button.Entered) {
button.Entered = false;
if (ThemeEngine.Current.ManagedWindowTitleButtonHasHotElementStyle (button, form))
any_change = true;
}
}
}
if (!any_tooltip)
ToolTipHide (false);
return any_change;
}
public void MouseDown (int x, int y)
{
if (!Visible) {
return;
}
ToolTipHide (false);
foreach (TitleButton button in this) {
if (button != null && button.State != ButtonState.Inactive) {
button.State = ButtonState.Normal;
}
}
TitleButton clicked_button = FindButton (x, y);
if (clicked_button != null && clicked_button.State != ButtonState.Inactive) {
clicked_button.State = ButtonState.Pushed;
}
}
public void MouseUp (int x, int y)
{
if (!Visible) {
return;
}
TitleButton clicked_button = FindButton (x, y);
if (clicked_button != null && clicked_button.State != ButtonState.Inactive) {
clicked_button.OnClick ();
}
foreach (TitleButton button in this) {
if (button == null || button.State == ButtonState.Inactive)
continue;
button.State = ButtonState.Normal;
}
if (clicked_button == CloseButton && !form.closing)
XplatUI.InvalidateNC (form.Handle);
ToolTipHide (true);
}
internal void MouseLeave (int x, int y)
{
if (!Visible) {
return;
}
foreach (TitleButton button in this) {
if (button == null || button.State == ButtonState.Inactive)
continue;
button.State = ButtonState.Normal;
}
ToolTipHide (true);
}
}
}