mirror of
https://github.com/seriocomedy/ShiftOS-C-.git
synced 2025-01-24 09:47:34 +00:00
1005 lines
25 KiB
C#
1005 lines
25 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 Novell, Inc. (http://www.novell.com)
|
||
|
//
|
||
|
// Authors:
|
||
|
// Peter Bartok pbartok@novell.com
|
||
|
//
|
||
|
//
|
||
|
|
||
|
using System.Collections;
|
||
|
using System.ComponentModel;
|
||
|
using System.Drawing;
|
||
|
using System.Drawing.Text;
|
||
|
using ShiftUI;
|
||
|
using System;
|
||
|
|
||
|
namespace ShiftUI {
|
||
|
[DefaultEvent ("Popup")]
|
||
|
[ProvideProperty ("ToolTip", typeof(Widget))]
|
||
|
[ToolboxItemFilter("ShiftUI", ToolboxItemFilterType.Allow)]
|
||
|
public class ToolTip : System.ComponentModel.Component, System.ComponentModel.IExtenderProvider {
|
||
|
#region Local variables
|
||
|
internal bool is_active;
|
||
|
internal int automatic_delay;
|
||
|
internal int autopop_delay;
|
||
|
internal int initial_delay;
|
||
|
internal int re_show_delay;
|
||
|
internal bool show_always;
|
||
|
|
||
|
internal Color back_color;
|
||
|
internal Color fore_color;
|
||
|
|
||
|
internal ToolTipWindow tooltip_window; // The actual tooltip window
|
||
|
internal Hashtable tooltip_strings; // List of strings for each Widget, indexed by Widget
|
||
|
internal ArrayList Widgets;
|
||
|
internal Widget active_Widget; // Widget for which the tooltip is currently displayed
|
||
|
internal Widget last_Widget; // last Widget the mouse was in
|
||
|
internal Timer timer; // Used for the various intervals
|
||
|
private Form hooked_form;
|
||
|
|
||
|
private bool isBalloon;
|
||
|
private bool owner_draw;
|
||
|
private bool stripAmpersands;
|
||
|
private ToolTipIcon tool_tip_icon;
|
||
|
private bool useAnimation;
|
||
|
private bool useFading;
|
||
|
private object tag;
|
||
|
|
||
|
#endregion // Local variables
|
||
|
|
||
|
#region ToolTipWindow Class
|
||
|
internal class ToolTipWindow : Widget {
|
||
|
#region ToolTipWindow Class Local Variables
|
||
|
private Widget associated_Widget;
|
||
|
internal Icon icon;
|
||
|
internal string title = String.Empty;
|
||
|
internal Rectangle icon_rect;
|
||
|
internal Rectangle title_rect;
|
||
|
internal Rectangle text_rect;
|
||
|
#endregion // ToolTipWindow Class Local Variables
|
||
|
|
||
|
#region ToolTipWindow Class Constructor
|
||
|
internal ToolTipWindow() {
|
||
|
Visible = false;
|
||
|
Size = new Size(100, 20);
|
||
|
ForeColor = ThemeEngine.Current.ColorInfoText;
|
||
|
BackColor = ThemeEngine.Current.ColorInfo;
|
||
|
|
||
|
VisibleChanged += new EventHandler(ToolTipWindow_VisibleChanged);
|
||
|
|
||
|
// UIA Framework: Used to generate UnPopup
|
||
|
VisibleChanged += new EventHandler (OnUIAToolTip_VisibleChanged);
|
||
|
|
||
|
SetStyle (Widgetstyles.UserPaint | Widgetstyles.AllPaintingInWmPaint, true);
|
||
|
SetStyle (Widgetstyles.ResizeRedraw, true);
|
||
|
if (ThemeEngine.Current.ToolTipTransparentBackground) {
|
||
|
SetStyle (Widgetstyles.SupportsTransparentBackColor, true);
|
||
|
BackColor = Color.Transparent;
|
||
|
} else
|
||
|
SetStyle (Widgetstyles.Opaque, true);
|
||
|
}
|
||
|
|
||
|
#endregion // ToolTipWindow Class Constructor
|
||
|
|
||
|
#region ToolTipWindow Class Protected Instance Methods
|
||
|
protected override void OnCreateWidget() {
|
||
|
base.OnCreateWidget ();
|
||
|
XplatUI.SetTopmost(this.window.Handle, true);
|
||
|
}
|
||
|
|
||
|
protected override CreateParams CreateParams {
|
||
|
get {
|
||
|
CreateParams cp;
|
||
|
|
||
|
cp = base.CreateParams;
|
||
|
|
||
|
cp.Style = (int)WindowStyles.WS_POPUP;
|
||
|
cp.Style |= (int)WindowStyles.WS_CLIPSIBLINGS;
|
||
|
|
||
|
cp.ExStyle = (int)(WindowExStyles.WS_EX_TOOLWINDOW | WindowExStyles.WS_EX_TOPMOST);
|
||
|
|
||
|
return cp;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected override void OnPaint(PaintEventArgs pevent) {
|
||
|
// We don't do double-buffering on purpose:
|
||
|
// 1) we'd have to meddle with is_visible, it destroys the buffers if !visible
|
||
|
// 2) We don't draw much, no need to double buffer
|
||
|
base.OnPaint(pevent);
|
||
|
|
||
|
OnDraw (new DrawToolTipEventArgs (pevent.Graphics, associated_Widget, associated_Widget, ClientRectangle, this.Text, this.BackColor, this.ForeColor, this.Font));
|
||
|
}
|
||
|
|
||
|
protected override void OnTextChanged (EventArgs args)
|
||
|
{
|
||
|
Invalidate ();
|
||
|
base.OnTextChanged (args);
|
||
|
}
|
||
|
|
||
|
protected override void WndProc(ref Message m) {
|
||
|
if (m.Msg == (int)Msg.WM_SETFOCUS) {
|
||
|
if (m.WParam != IntPtr.Zero) {
|
||
|
XplatUI.SetFocus(m.WParam);
|
||
|
}
|
||
|
}
|
||
|
base.WndProc (ref m);
|
||
|
}
|
||
|
|
||
|
|
||
|
#endregion // ToolTipWindow Class Protected Instance Methods
|
||
|
|
||
|
#region ToolTipWindow Class Private Methods
|
||
|
internal virtual void OnDraw (DrawToolTipEventArgs e)
|
||
|
{
|
||
|
DrawToolTipEventHandler eh = (DrawToolTipEventHandler)(Events[DrawEvent]);
|
||
|
if (eh != null)
|
||
|
eh (this, e);
|
||
|
else
|
||
|
ThemeEngine.Current.DrawToolTip (e.Graphics, e.Bounds, this);
|
||
|
}
|
||
|
|
||
|
internal virtual void OnPopup (PopupEventArgs e)
|
||
|
{
|
||
|
PopupEventHandler eh = (PopupEventHandler)(Events[PopupEvent]);
|
||
|
if (eh != null)
|
||
|
eh (this, e);
|
||
|
else
|
||
|
e.ToolTipSize = ThemeEngine.Current.ToolTipSize (this, Text);
|
||
|
}
|
||
|
|
||
|
private void ToolTipWindow_VisibleChanged(object sender, EventArgs e) {
|
||
|
Widget Widget = (Widget)sender;
|
||
|
|
||
|
if (Widget.is_visible) {
|
||
|
XplatUI.SetTopmost(Widget.window.Handle, true);
|
||
|
} else {
|
||
|
XplatUI.SetTopmost(Widget.window.Handle, false);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// UIA Framework
|
||
|
private void OnUIAToolTip_VisibleChanged (object sender, EventArgs e)
|
||
|
{
|
||
|
if (Visible == false)
|
||
|
OnUnPopup (new PopupEventArgs (associated_Widget, associated_Widget, false, Size.Empty));
|
||
|
}
|
||
|
|
||
|
private void OnUnPopup (PopupEventArgs e)
|
||
|
{
|
||
|
PopupEventHandler eh = (PopupEventHandler) (Events [UnPopupEvent]);
|
||
|
if (eh != null)
|
||
|
eh (this, e);
|
||
|
}
|
||
|
|
||
|
|
||
|
#endregion // ToolTipWindow Class Protected Instance Methods
|
||
|
|
||
|
#region Internal Properties
|
||
|
internal override bool ActivateOnShow { get { return false; } }
|
||
|
#endregion
|
||
|
|
||
|
// This Present is used when we are using the expicit Show methods for 2.0.
|
||
|
// It will not reposition the window.
|
||
|
public void PresentModal (Widget Widget, string text)
|
||
|
{
|
||
|
if (IsDisposed)
|
||
|
return;
|
||
|
|
||
|
Size display_size;
|
||
|
XplatUI.GetDisplaySize (out display_size);
|
||
|
|
||
|
associated_Widget = Widget;
|
||
|
|
||
|
Text = text;
|
||
|
|
||
|
PopupEventArgs pea = new PopupEventArgs (Widget, Widget, false, Size.Empty);
|
||
|
OnPopup (pea);
|
||
|
|
||
|
if (pea.Cancel)
|
||
|
return;
|
||
|
|
||
|
Size = pea.ToolTipSize;
|
||
|
|
||
|
Visible = true;
|
||
|
}
|
||
|
|
||
|
public void Present (Widget Widget, string text)
|
||
|
{
|
||
|
if (IsDisposed)
|
||
|
return;
|
||
|
|
||
|
Size display_size;
|
||
|
XplatUI.GetDisplaySize (out display_size);
|
||
|
|
||
|
associated_Widget = Widget;
|
||
|
|
||
|
Text = text;
|
||
|
|
||
|
PopupEventArgs pea = new PopupEventArgs (Widget, Widget, false, Size.Empty);
|
||
|
OnPopup (pea);
|
||
|
|
||
|
if (pea.Cancel)
|
||
|
return;
|
||
|
|
||
|
Size size = pea.ToolTipSize;
|
||
|
|
||
|
Width = size.Width;
|
||
|
Height = size.Height;
|
||
|
|
||
|
int cursor_w, cursor_h, hot_x, hot_y;
|
||
|
XplatUI.GetCursorInfo (Widget.Cursor.Handle, out cursor_w, out cursor_h, out hot_x, out hot_y);
|
||
|
Point loc = Widget.MousePosition;
|
||
|
loc.Y += (cursor_h - hot_y);
|
||
|
|
||
|
if ((loc.X + Width) > display_size.Width)
|
||
|
loc.X = display_size.Width - Width;
|
||
|
|
||
|
if ((loc.Y + Height) > display_size.Height)
|
||
|
loc.Y = Widget.MousePosition.Y - Height - hot_y;
|
||
|
|
||
|
Location = loc;
|
||
|
Visible = true;
|
||
|
BringToFront ();
|
||
|
}
|
||
|
|
||
|
|
||
|
#region Internal Events
|
||
|
static object DrawEvent = new object ();
|
||
|
static object PopupEvent = new object ();
|
||
|
|
||
|
// UIA Framework
|
||
|
static object UnPopupEvent = new object ();
|
||
|
|
||
|
public event DrawToolTipEventHandler Draw {
|
||
|
add { Events.AddHandler (DrawEvent, value); }
|
||
|
remove { Events.RemoveHandler (DrawEvent, value); }
|
||
|
}
|
||
|
|
||
|
public event PopupEventHandler Popup {
|
||
|
add { Events.AddHandler (PopupEvent, value); }
|
||
|
remove { Events.RemoveHandler (PopupEvent, value); }
|
||
|
}
|
||
|
|
||
|
internal event PopupEventHandler UnPopup {
|
||
|
add { Events.AddHandler (UnPopupEvent, value); }
|
||
|
remove { Events.RemoveHandler (UnPopupEvent, value); }
|
||
|
}
|
||
|
#endregion
|
||
|
}
|
||
|
#endregion // ToolTipWindow Class
|
||
|
|
||
|
#region Public Constructors & Destructors
|
||
|
public ToolTip() {
|
||
|
|
||
|
// Defaults from MS
|
||
|
is_active = true;
|
||
|
automatic_delay = 500;
|
||
|
autopop_delay = 5000;
|
||
|
initial_delay = 500;
|
||
|
re_show_delay = 100;
|
||
|
show_always = false;
|
||
|
back_color = SystemColors.Info;
|
||
|
fore_color = SystemColors.InfoText;
|
||
|
|
||
|
isBalloon = false;
|
||
|
stripAmpersands = false;
|
||
|
useAnimation = true;
|
||
|
useFading = true;
|
||
|
tooltip_strings = new Hashtable(5);
|
||
|
Widgets = new ArrayList(5);
|
||
|
|
||
|
tooltip_window = new ToolTipWindow();
|
||
|
tooltip_window.MouseLeave += new EventHandler(Widget_MouseLeave);
|
||
|
tooltip_window.Draw += new DrawToolTipEventHandler (tooltip_window_Draw);
|
||
|
tooltip_window.Popup += new PopupEventHandler (tooltip_window_Popup);
|
||
|
|
||
|
// UIA Framework: Static event handlers
|
||
|
tooltip_window.UnPopup += delegate (object sender, PopupEventArgs args) {
|
||
|
OnUnPopup (args);
|
||
|
};
|
||
|
UnPopup += new PopupEventHandler (OnUIAUnPopup);
|
||
|
|
||
|
timer = new Timer();
|
||
|
timer.Enabled = false;
|
||
|
timer.Tick +=new EventHandler(timer_Tick);
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
#region UIA Framework: Events, Delegates and Methods
|
||
|
// NOTE:
|
||
|
// We are using Reflection to add/remove internal events.
|
||
|
// Class ToolTipListener uses the events.
|
||
|
//
|
||
|
// - UIAUnPopup. Event used to generate ChildRemoved in ToolTip
|
||
|
// - UIAToolTipHookUp. Event used to keep track of associated Widgets
|
||
|
// - UIAToolTipUnhookUp. Event used to remove track of associated Widgets
|
||
|
static object UnPopupEvent = new object ();
|
||
|
|
||
|
internal event PopupEventHandler UnPopup {
|
||
|
add { Events.AddHandler (UnPopupEvent, value); }
|
||
|
remove { Events.RemoveHandler (UnPopupEvent, value); }
|
||
|
}
|
||
|
|
||
|
internal static event PopupEventHandler UIAUnPopup;
|
||
|
internal static event WidgetEventHandler UIAToolTipHookUp;
|
||
|
internal static event WidgetEventHandler UIAToolTipUnhookUp;
|
||
|
|
||
|
internal Rectangle UIAToolTipRectangle {
|
||
|
get { return tooltip_window.Bounds; }
|
||
|
}
|
||
|
|
||
|
internal static void OnUIAUnPopup (object sender, PopupEventArgs args)
|
||
|
{
|
||
|
if (UIAUnPopup != null)
|
||
|
UIAUnPopup (sender, args);
|
||
|
}
|
||
|
|
||
|
internal static void OnUIAToolTipHookUp (object sender, WidgetEventArgs args)
|
||
|
{
|
||
|
if (UIAToolTipHookUp != null)
|
||
|
UIAToolTipHookUp (sender, args);
|
||
|
}
|
||
|
|
||
|
internal static void OnUIAToolTipUnhookUp (object sender, WidgetEventArgs args)
|
||
|
{
|
||
|
if (UIAToolTipUnhookUp != null)
|
||
|
UIAToolTipUnhookUp (sender, args);
|
||
|
}
|
||
|
|
||
|
#endregion
|
||
|
|
||
|
public ToolTip(System.ComponentModel.IContainer cont) : this() {
|
||
|
cont.Add (this);
|
||
|
}
|
||
|
|
||
|
~ToolTip() {
|
||
|
}
|
||
|
#endregion // Public Constructors & Destructors
|
||
|
|
||
|
#region Public Instance Properties
|
||
|
[DefaultValue (true)]
|
||
|
public bool Active {
|
||
|
get {
|
||
|
return is_active;
|
||
|
}
|
||
|
|
||
|
set {
|
||
|
if (is_active != value) {
|
||
|
is_active = value;
|
||
|
|
||
|
if (tooltip_window.Visible) {
|
||
|
tooltip_window.Visible = false;
|
||
|
active_Widget = null;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[DefaultValue (500)]
|
||
|
[RefreshProperties (RefreshProperties.All)]
|
||
|
public int AutomaticDelay {
|
||
|
get {
|
||
|
return automatic_delay;
|
||
|
}
|
||
|
|
||
|
set {
|
||
|
if (automatic_delay != value) {
|
||
|
automatic_delay = value;
|
||
|
autopop_delay = automatic_delay * 10;
|
||
|
initial_delay = automatic_delay;
|
||
|
re_show_delay = automatic_delay / 5;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[RefreshProperties (RefreshProperties.All)]
|
||
|
public int AutoPopDelay {
|
||
|
get {
|
||
|
return autopop_delay;
|
||
|
}
|
||
|
|
||
|
set {
|
||
|
if (autopop_delay != value) {
|
||
|
autopop_delay = value;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[DefaultValue ("Color [Info]")]
|
||
|
public Color BackColor {
|
||
|
get { return this.back_color; }
|
||
|
set { this.back_color = value; tooltip_window.BackColor = value; }
|
||
|
}
|
||
|
|
||
|
[DefaultValue ("Color [InfoText]")]
|
||
|
public Color ForeColor
|
||
|
{
|
||
|
get { return this.fore_color; }
|
||
|
set { this.fore_color = value; tooltip_window.ForeColor = value; }
|
||
|
}
|
||
|
|
||
|
[RefreshProperties (RefreshProperties.All)]
|
||
|
public int InitialDelay {
|
||
|
get {
|
||
|
return initial_delay;
|
||
|
}
|
||
|
|
||
|
set {
|
||
|
if (initial_delay != value) {
|
||
|
initial_delay = value;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[DefaultValue (false)]
|
||
|
public bool OwnerDraw {
|
||
|
get { return this.owner_draw; }
|
||
|
set { this.owner_draw = value; }
|
||
|
}
|
||
|
|
||
|
[RefreshProperties (RefreshProperties.All)]
|
||
|
public int ReshowDelay {
|
||
|
get {
|
||
|
return re_show_delay;
|
||
|
}
|
||
|
|
||
|
set {
|
||
|
if (re_show_delay != value) {
|
||
|
re_show_delay = value;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[DefaultValue (false)]
|
||
|
public bool ShowAlways {
|
||
|
get {
|
||
|
return show_always;
|
||
|
}
|
||
|
|
||
|
set {
|
||
|
if (show_always != value) {
|
||
|
show_always = value;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
[DefaultValue (false)]
|
||
|
public bool IsBalloon {
|
||
|
get { return isBalloon; }
|
||
|
set { isBalloon = value; }
|
||
|
}
|
||
|
|
||
|
[Browsable (true)]
|
||
|
[DefaultValue (false)]
|
||
|
public bool StripAmpersands {
|
||
|
get { return stripAmpersands; }
|
||
|
set { stripAmpersands = value; }
|
||
|
}
|
||
|
|
||
|
[Localizable (false)]
|
||
|
[Bindable (true)]
|
||
|
[TypeConverter (typeof (StringConverter))]
|
||
|
[DefaultValue (null)]
|
||
|
public object Tag {
|
||
|
get { return tag; }
|
||
|
set { tag = value; }
|
||
|
}
|
||
|
|
||
|
[DefaultValue (ToolTipIcon.None)]
|
||
|
public ToolTipIcon ToolTipIcon {
|
||
|
get { return this.tool_tip_icon; }
|
||
|
set {
|
||
|
switch (value) {
|
||
|
case ToolTipIcon.None:
|
||
|
tooltip_window.icon = null;
|
||
|
break;
|
||
|
case ToolTipIcon.Error:
|
||
|
tooltip_window.icon = SystemIcons.Error;
|
||
|
break;
|
||
|
case ToolTipIcon.Warning:
|
||
|
tooltip_window.icon = SystemIcons.Warning;
|
||
|
break;
|
||
|
case ToolTipIcon.Info:
|
||
|
tooltip_window.icon = SystemIcons.Information;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
tool_tip_icon = value;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[DefaultValue ("")]
|
||
|
public string ToolTipTitle {
|
||
|
get { return tooltip_window.title; }
|
||
|
set {
|
||
|
if (value == null)
|
||
|
value = String.Empty;
|
||
|
|
||
|
tooltip_window.title = value;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[Browsable (true)]
|
||
|
[DefaultValue (true)]
|
||
|
public bool UseAnimation {
|
||
|
get { return useAnimation; }
|
||
|
set { useAnimation = value; }
|
||
|
}
|
||
|
|
||
|
[Browsable (true)]
|
||
|
[DefaultValue (true)]
|
||
|
public bool UseFading {
|
||
|
get { return useFading; }
|
||
|
set { useFading = value; }
|
||
|
}
|
||
|
|
||
|
#endregion // Public Instance Properties
|
||
|
|
||
|
#region Protected Properties
|
||
|
protected virtual CreateParams CreateParams
|
||
|
{
|
||
|
get
|
||
|
{
|
||
|
CreateParams cp = new CreateParams ();
|
||
|
|
||
|
cp.Style = 2;
|
||
|
|
||
|
return cp;
|
||
|
}
|
||
|
}
|
||
|
#endregion
|
||
|
|
||
|
#region Public Instance Methods
|
||
|
public bool CanExtend(object target) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
[Localizable (true)]
|
||
|
[DefaultValue ("")]
|
||
|
public string GetToolTip (Widget Widget)
|
||
|
{
|
||
|
string tooltip = (string)tooltip_strings[Widget];
|
||
|
if (tooltip == null)
|
||
|
return "";
|
||
|
return tooltip;
|
||
|
}
|
||
|
|
||
|
public void RemoveAll() {
|
||
|
tooltip_strings.Clear();
|
||
|
//UIA Framework: ToolTip isn't associated anymore
|
||
|
foreach (Widget Widget in Widgets)
|
||
|
OnUIAToolTipUnhookUp (this, new WidgetEventArgs (Widget));
|
||
|
|
||
|
Widgets.Clear();
|
||
|
}
|
||
|
|
||
|
public void SetToolTip(Widget Widget, string caption) {
|
||
|
// UIA Framework
|
||
|
OnUIAToolTipHookUp (this, new WidgetEventArgs (Widget));
|
||
|
tooltip_strings[Widget] = caption;
|
||
|
|
||
|
// no need for duplicates
|
||
|
if (!Widgets.Contains(Widget)) {
|
||
|
Widget.MouseEnter += new EventHandler(Widget_MouseEnter);
|
||
|
Widget.MouseMove += new MouseEventHandler(Widget_MouseMove);
|
||
|
Widget.MouseLeave += new EventHandler(Widget_MouseLeave);
|
||
|
Widget.MouseDown += new MouseEventHandler (Widget_MouseDown);
|
||
|
Widgets.Add(Widget);
|
||
|
}
|
||
|
|
||
|
// if SetToolTip is called from a Widget and the mouse is currently over that Widget,
|
||
|
// make sure that tooltip_window.Text gets updated if it's being shown,
|
||
|
// or show the tooltip for it if is not
|
||
|
if (active_Widget == Widget && caption != null && state == TipState.Show) {
|
||
|
Size size = ThemeEngine.Current.ToolTipSize(tooltip_window, caption);
|
||
|
tooltip_window.Width = size.Width;
|
||
|
tooltip_window.Height = size.Height;
|
||
|
tooltip_window.Text = caption;
|
||
|
timer.Stop ();
|
||
|
timer.Start ();
|
||
|
} else if (Widget.IsHandleCreated && MouseInWidget (Widget, false))
|
||
|
ShowTooltip (Widget);
|
||
|
}
|
||
|
|
||
|
public override string ToString() {
|
||
|
return base.ToString() + " InitialDelay: " + initial_delay + ", ShowAlways: " + show_always;
|
||
|
}
|
||
|
|
||
|
public void Show (string text, IWin32Window window)
|
||
|
{
|
||
|
Show (text, window, 0);
|
||
|
}
|
||
|
|
||
|
public void Show (string text, IWin32Window window, int duration)
|
||
|
{
|
||
|
if (window == null)
|
||
|
throw new ArgumentNullException ("window");
|
||
|
if (duration < 0)
|
||
|
throw new ArgumentOutOfRangeException ("duration", "duration cannot be less than zero");
|
||
|
|
||
|
if (!Active)
|
||
|
return;
|
||
|
|
||
|
timer.Stop ();
|
||
|
|
||
|
Widget c = (Widget)window;
|
||
|
|
||
|
XplatUI.SetOwner (tooltip_window.Handle, c.TopLevelWidget.Handle);
|
||
|
|
||
|
// If the mouse is in the requested window, use that position
|
||
|
// Else, center in the requested window
|
||
|
if (c.ClientRectangle.Contains (c.PointToClient (Widget.MousePosition))) {
|
||
|
tooltip_window.Location = Widget.MousePosition;
|
||
|
tooltip_strings[c] = text;
|
||
|
HookupWidgetEvents (c);
|
||
|
}
|
||
|
else
|
||
|
tooltip_window.Location = c.PointToScreen (new Point (c.Width / 2, c.Height / 2));
|
||
|
|
||
|
// We need to hide our tooltip if the form loses focus, is closed, or is minimized
|
||
|
HookupFormEvents ((Form)c.TopLevelWidget);
|
||
|
|
||
|
tooltip_window.PresentModal ((Widget)window, text);
|
||
|
|
||
|
state = TipState.Show;
|
||
|
|
||
|
if (duration > 0) {
|
||
|
timer.Interval = duration;
|
||
|
timer.Start ();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void Show (string text, IWin32Window window, Point point)
|
||
|
{
|
||
|
Show (text, window, point, 0);
|
||
|
}
|
||
|
|
||
|
public void Show (string text, IWin32Window window, int x, int y)
|
||
|
{
|
||
|
Show (text, window, new Point (x, y), 0);
|
||
|
}
|
||
|
|
||
|
public void Show (string text, IWin32Window window, Point point, int duration)
|
||
|
{
|
||
|
if (window == null)
|
||
|
throw new ArgumentNullException ("window");
|
||
|
if (duration < 0)
|
||
|
throw new ArgumentOutOfRangeException ("duration", "duration cannot be less than zero");
|
||
|
|
||
|
if (!Active)
|
||
|
return;
|
||
|
|
||
|
timer.Stop ();
|
||
|
|
||
|
Widget c = (Widget)window;
|
||
|
|
||
|
Point display_point = c.PointToScreen (Point.Empty);
|
||
|
display_point.X += point.X;
|
||
|
display_point.Y += point.Y;
|
||
|
|
||
|
XplatUI.SetOwner (tooltip_window.Handle, c.TopLevelWidget.Handle);
|
||
|
|
||
|
// We need to hide our tooltip if the form loses focus, is closed, or is minimized
|
||
|
HookupFormEvents ((Form)c.TopLevelWidget);
|
||
|
|
||
|
tooltip_window.Location = display_point;
|
||
|
tooltip_window.PresentModal ((Widget)window, text);
|
||
|
|
||
|
state = TipState.Show;
|
||
|
|
||
|
if (duration > 0) {
|
||
|
timer.Interval = duration;
|
||
|
timer.Start ();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void Show (string text, IWin32Window window, int x, int y, int duration)
|
||
|
{
|
||
|
Show (text, window, new Point (x, y), duration);
|
||
|
}
|
||
|
|
||
|
public void Hide (IWin32Window win)
|
||
|
{
|
||
|
timer.Stop ();
|
||
|
state = TipState.Initial;
|
||
|
|
||
|
UnhookFormEvents ();
|
||
|
tooltip_window.Visible = false;
|
||
|
}
|
||
|
#endregion // Public Instance Methods
|
||
|
|
||
|
#region Protected Instance Methods
|
||
|
protected override void Dispose(bool disposing) {
|
||
|
// call the base impl first to avoid conflicts with any parent's events
|
||
|
base.Dispose (disposing);
|
||
|
|
||
|
if (disposing) {
|
||
|
// Mop up the mess; or should we wait for the GC to kick in?
|
||
|
timer.Stop();
|
||
|
timer.Dispose();
|
||
|
|
||
|
// Not sure if we should clean up tooltip_window
|
||
|
tooltip_window.Dispose();
|
||
|
|
||
|
tooltip_strings.Clear();
|
||
|
|
||
|
//UIA Framework: ToolTip isn't associated anymore
|
||
|
foreach (Widget Widget in Widgets)
|
||
|
OnUIAToolTipUnhookUp (this, new WidgetEventArgs (Widget));
|
||
|
Widgets.Clear();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected void StopTimer ()
|
||
|
{
|
||
|
timer.Stop ();
|
||
|
}
|
||
|
#endregion // Protected Instance Methods
|
||
|
|
||
|
internal enum TipState {
|
||
|
Initial,
|
||
|
Show,
|
||
|
Down
|
||
|
}
|
||
|
|
||
|
TipState state = TipState.Initial;
|
||
|
|
||
|
#region Private Methods
|
||
|
|
||
|
private void HookupFormEvents (Form form)
|
||
|
{
|
||
|
hooked_form = form;
|
||
|
|
||
|
form.Deactivate += new EventHandler (Form_Deactivate);
|
||
|
form.Closed += new EventHandler (Form_Closed);
|
||
|
form.Resize += new EventHandler (Form_Resize);
|
||
|
}
|
||
|
|
||
|
private void HookupWidgetEvents (Widget widget)
|
||
|
{
|
||
|
if (!Widgets.Contains (widget)) {
|
||
|
widget.MouseEnter += new EventHandler (Widget_MouseEnter);
|
||
|
widget.MouseMove += new MouseEventHandler (Widget_MouseMove);
|
||
|
widget.MouseLeave += new EventHandler (Widget_MouseLeave);
|
||
|
widget.MouseDown += new MouseEventHandler (Widget_MouseDown);
|
||
|
Widgets.Add (widget);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void UnhookWidgetEvents (Widget widget)
|
||
|
{
|
||
|
widget.MouseEnter -= new EventHandler (Widget_MouseEnter);
|
||
|
widget.MouseMove -= new MouseEventHandler (Widget_MouseMove);
|
||
|
widget.MouseLeave -= new EventHandler (Widget_MouseLeave);
|
||
|
widget.MouseDown -= new MouseEventHandler (Widget_MouseDown);
|
||
|
}
|
||
|
private void UnhookFormEvents ()
|
||
|
{
|
||
|
if (hooked_form == null)
|
||
|
return;
|
||
|
|
||
|
hooked_form.Deactivate -= new EventHandler (Form_Deactivate);
|
||
|
hooked_form.Closed -= new EventHandler (Form_Closed);
|
||
|
hooked_form.Resize -= new EventHandler (Form_Resize);
|
||
|
|
||
|
hooked_form = null;
|
||
|
}
|
||
|
|
||
|
|
||
|
private void Form_Resize (object sender, EventArgs e)
|
||
|
{
|
||
|
Form f = (Form)sender;
|
||
|
|
||
|
if (f.WindowState == FormWindowState.Minimized)
|
||
|
tooltip_window.Visible = false;
|
||
|
}
|
||
|
|
||
|
private void Form_Closed (object sender, EventArgs e)
|
||
|
{
|
||
|
tooltip_window.Visible = false;
|
||
|
}
|
||
|
|
||
|
private void Form_Deactivate (object sender, EventArgs e)
|
||
|
{
|
||
|
tooltip_window.Visible = false;
|
||
|
}
|
||
|
|
||
|
internal void Present (Widget Widget, string text)
|
||
|
{
|
||
|
tooltip_window.Present (Widget, text);
|
||
|
}
|
||
|
|
||
|
private void Widget_MouseEnter (object sender, EventArgs e)
|
||
|
{
|
||
|
ShowTooltip (sender as Widget);
|
||
|
}
|
||
|
|
||
|
private void ShowTooltip (Widget Widget)
|
||
|
{
|
||
|
last_Widget = Widget;
|
||
|
|
||
|
// Whatever we're displaying right now, we don't want it anymore
|
||
|
tooltip_window.Visible = false;
|
||
|
timer.Stop();
|
||
|
state = TipState.Initial;
|
||
|
|
||
|
if (!is_active)
|
||
|
return;
|
||
|
|
||
|
// ShowAlways Widgets whether the Widgets in non-active forms
|
||
|
// can display its tooltips, even if they are not current active Widget.
|
||
|
if (!show_always && Widget.FindForm () != Form.ActiveForm)
|
||
|
return;
|
||
|
|
||
|
string text = (string)tooltip_strings[Widget];
|
||
|
if (text != null && text.Length > 0) {
|
||
|
if (active_Widget == null) {
|
||
|
timer.Interval = Math.Max (initial_delay, 1);
|
||
|
} else {
|
||
|
timer.Interval = Math.Max (re_show_delay, 1);
|
||
|
}
|
||
|
|
||
|
active_Widget = Widget;
|
||
|
timer.Start ();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void timer_Tick(object sender, EventArgs e) {
|
||
|
timer.Stop();
|
||
|
|
||
|
switch (state) {
|
||
|
case TipState.Initial:
|
||
|
if (active_Widget == null)
|
||
|
return;
|
||
|
tooltip_window.Present (active_Widget, (string)tooltip_strings[active_Widget]);
|
||
|
state = TipState.Show;
|
||
|
timer.Interval = autopop_delay;
|
||
|
timer.Start();
|
||
|
break;
|
||
|
|
||
|
case TipState.Show:
|
||
|
tooltip_window.Visible = false;
|
||
|
state = TipState.Down;
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
throw new Exception ("Timer shouldn't be running in state: " + state);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void tooltip_window_Popup (object sender, PopupEventArgs e)
|
||
|
{
|
||
|
e.ToolTipSize = ThemeEngine.Current.ToolTipSize (tooltip_window, tooltip_window.Text);
|
||
|
OnPopup (e);
|
||
|
}
|
||
|
|
||
|
private void tooltip_window_Draw (object sender, DrawToolTipEventArgs e)
|
||
|
{
|
||
|
if (OwnerDraw)
|
||
|
OnDraw (e);
|
||
|
else
|
||
|
ThemeEngine.Current.DrawToolTip (e.Graphics, e.Bounds, tooltip_window);
|
||
|
}
|
||
|
|
||
|
private bool MouseInWidget (Widget Widget, bool fuzzy) {
|
||
|
Point m;
|
||
|
Point c;
|
||
|
Size cw;
|
||
|
|
||
|
if (Widget == null) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
m = Widget.MousePosition;
|
||
|
c = new Point(Widget.Bounds.X, Widget.Bounds.Y);
|
||
|
if (Widget.Parent != null) {
|
||
|
c = Widget.Parent.PointToScreen(c);
|
||
|
}
|
||
|
cw = Widget.ClientSize;
|
||
|
|
||
|
|
||
|
Rectangle rect = new Rectangle (c, cw);
|
||
|
|
||
|
//
|
||
|
// We won't get mouse move events on all platforms with the exact same
|
||
|
// frequency, so cheat a bit.
|
||
|
if (fuzzy)
|
||
|
rect.Inflate (2, 2);
|
||
|
|
||
|
return rect.Contains (m);
|
||
|
}
|
||
|
|
||
|
private void Widget_MouseLeave(object sender, EventArgs e)
|
||
|
{
|
||
|
timer.Stop ();
|
||
|
|
||
|
active_Widget = null;
|
||
|
tooltip_window.Visible = false;
|
||
|
|
||
|
if (last_Widget == sender)
|
||
|
last_Widget = null;
|
||
|
}
|
||
|
|
||
|
|
||
|
void Widget_MouseDown (object sender, MouseEventArgs e)
|
||
|
{
|
||
|
timer.Stop();
|
||
|
|
||
|
active_Widget = null;
|
||
|
tooltip_window.Visible = false;
|
||
|
|
||
|
if (last_Widget == sender)
|
||
|
last_Widget = null;
|
||
|
}
|
||
|
|
||
|
private void Widget_MouseMove(object sender, MouseEventArgs e) {
|
||
|
if (state != TipState.Down) {
|
||
|
timer.Stop();
|
||
|
timer.Start();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal void OnDraw (DrawToolTipEventArgs e)
|
||
|
{
|
||
|
DrawToolTipEventHandler eh = (DrawToolTipEventHandler)(Events[DrawEvent]);
|
||
|
if (eh != null)
|
||
|
eh (this, e);
|
||
|
}
|
||
|
|
||
|
internal void OnPopup (PopupEventArgs e)
|
||
|
{
|
||
|
PopupEventHandler eh = (PopupEventHandler) (Events [PopupEvent]);
|
||
|
if (eh != null)
|
||
|
eh (this, e);
|
||
|
}
|
||
|
|
||
|
internal void OnUnPopup (PopupEventArgs e)
|
||
|
{
|
||
|
PopupEventHandler eh = (PopupEventHandler) (Events [UnPopupEvent]);
|
||
|
if (eh != null)
|
||
|
eh (this, e);
|
||
|
}
|
||
|
|
||
|
internal bool Visible {
|
||
|
get { return tooltip_window.Visible; }
|
||
|
}
|
||
|
#endregion // Private Methods
|
||
|
|
||
|
#region Events
|
||
|
static object PopupEvent = new object ();
|
||
|
static object DrawEvent = new object ();
|
||
|
|
||
|
public event PopupEventHandler Popup {
|
||
|
add { Events.AddHandler (PopupEvent, value); }
|
||
|
remove { Events.RemoveHandler (PopupEvent, value); }
|
||
|
}
|
||
|
|
||
|
public event DrawToolTipEventHandler Draw {
|
||
|
add { Events.AddHandler (DrawEvent, value); }
|
||
|
remove { Events.RemoveHandler (DrawEvent, value); }
|
||
|
}
|
||
|
#endregion
|
||
|
}
|
||
|
}
|