Added a wrapper for the NotifyIconAdv to use the normal NotifyIcon class on Linux systems and the (fixed) custom implementation on Windows systems.
1.1 --- a/GUI/NotifyIconAdv.cs Mon Jul 23 21:54:35 2012 +0000
1.2 +++ b/GUI/NotifyIconAdv.cs Tue Jul 24 16:04:30 2012 +0000
1.3 @@ -16,115 +16,298 @@
1.4 using System.Windows.Forms;
1.5
1.6 namespace OpenHardwareMonitor.GUI {
1.7 - public class NotifyIconAdv : Component {
1.8
1.9 - private static int nextId = 0;
1.10 + public class NotifyIconAdv : IDisposable {
1.11
1.12 - private object syncObj = new object();
1.13 - private Icon icon;
1.14 - private string text = "";
1.15 - private int id;
1.16 - private bool created;
1.17 - private NotifyIconNativeWindow window;
1.18 - private bool doubleClickDown;
1.19 - private bool visible;
1.20 - private MethodInfo commandDispatch;
1.21 + private NotifyIcon genericNotifyIcon;
1.22 + private NotifyIconWindowsImplementation windowsNotifyIcon;
1.23
1.24 - public event EventHandler BalloonTipClicked;
1.25 - public event EventHandler BalloonTipClosed;
1.26 - public event EventHandler BalloonTipShown;
1.27 - public event EventHandler Click;
1.28 - public event EventHandler DoubleClick;
1.29 - public event MouseEventHandler MouseClick;
1.30 - public event MouseEventHandler MouseDoubleClick;
1.31 - public event MouseEventHandler MouseDown;
1.32 - public event MouseEventHandler MouseMove;
1.33 - public event MouseEventHandler MouseUp;
1.34 + public NotifyIconAdv() {
1.35 + int p = (int)Environment.OSVersion.Platform;
1.36 + if ((p == 4) || (p == 128)) { // Unix
1.37 + genericNotifyIcon = new NotifyIcon();
1.38 + } else { // Windows
1.39 + windowsNotifyIcon = new NotifyIconWindowsImplementation();
1.40 + }
1.41 + }
1.42
1.43 - public string BalloonTipText { get; set; }
1.44 - public ToolTipIcon BalloonTipIcon { get; set; }
1.45 - public string BalloonTipTitle { get; set; }
1.46 - public ContextMenu ContextMenu { get; set; }
1.47 - public ContextMenuStrip ContextMenuStrip { get; set; }
1.48 + public event EventHandler BalloonTipClicked {
1.49 + add {
1.50 + if (genericNotifyIcon != null)
1.51 + genericNotifyIcon.BalloonTipClicked += value;
1.52 + else
1.53 + windowsNotifyIcon.BalloonTipClicked += value;
1.54 + }
1.55 + remove {
1.56 + if (genericNotifyIcon != null)
1.57 + genericNotifyIcon.BalloonTipClicked -= value;
1.58 + else
1.59 + windowsNotifyIcon.BalloonTipClicked -= value;
1.60 + }
1.61 + }
1.62 +
1.63 + public event EventHandler BalloonTipClosed {
1.64 + add {
1.65 + if (genericNotifyIcon != null)
1.66 + genericNotifyIcon.BalloonTipClosed += value;
1.67 + else
1.68 + windowsNotifyIcon.BalloonTipClosed += value;
1.69 + }
1.70 + remove {
1.71 + if (genericNotifyIcon != null)
1.72 + genericNotifyIcon.BalloonTipClosed -= value;
1.73 + else
1.74 + windowsNotifyIcon.BalloonTipClosed -= value;
1.75 + }
1.76 + }
1.77 +
1.78 + public event EventHandler BalloonTipShown {
1.79 + add {
1.80 + if (genericNotifyIcon != null)
1.81 + genericNotifyIcon.BalloonTipShown += value;
1.82 + else
1.83 + windowsNotifyIcon.BalloonTipShown += value;
1.84 + }
1.85 + remove {
1.86 + if (genericNotifyIcon != null)
1.87 + genericNotifyIcon.BalloonTipShown -= value;
1.88 + else
1.89 + windowsNotifyIcon.BalloonTipShown -= value;
1.90 + }
1.91 + }
1.92 +
1.93 + public event EventHandler Click {
1.94 + add {
1.95 + if (genericNotifyIcon != null)
1.96 + genericNotifyIcon.Click += value;
1.97 + else
1.98 + windowsNotifyIcon.Click += value;
1.99 + }
1.100 + remove {
1.101 + if (genericNotifyIcon != null)
1.102 + genericNotifyIcon.Click -= value;
1.103 + else
1.104 + windowsNotifyIcon.Click -= value;
1.105 + }
1.106 + }
1.107 +
1.108 + public event EventHandler DoubleClick {
1.109 + add {
1.110 + if (genericNotifyIcon != null)
1.111 + genericNotifyIcon.DoubleClick += value;
1.112 + else
1.113 + windowsNotifyIcon.DoubleClick += value;
1.114 + }
1.115 + remove {
1.116 + if (genericNotifyIcon != null)
1.117 + genericNotifyIcon.DoubleClick -= value;
1.118 + else
1.119 + windowsNotifyIcon.DoubleClick -= value;
1.120 + }
1.121 + }
1.122 +
1.123 + public event MouseEventHandler MouseClick {
1.124 + add {
1.125 + if (genericNotifyIcon != null)
1.126 + genericNotifyIcon.MouseClick += value;
1.127 + else
1.128 + windowsNotifyIcon.MouseClick += value;
1.129 + }
1.130 + remove {
1.131 + if (genericNotifyIcon != null)
1.132 + genericNotifyIcon.MouseClick -= value;
1.133 + else
1.134 + windowsNotifyIcon.MouseClick -= value;
1.135 + }
1.136 + }
1.137 +
1.138 + public event MouseEventHandler MouseDoubleClick {
1.139 + add {
1.140 + if (genericNotifyIcon != null)
1.141 + genericNotifyIcon.MouseDoubleClick += value;
1.142 + else
1.143 + windowsNotifyIcon.MouseDoubleClick += value;
1.144 + }
1.145 + remove {
1.146 + if (genericNotifyIcon != null)
1.147 + genericNotifyIcon.MouseDoubleClick -= value;
1.148 + else
1.149 + windowsNotifyIcon.MouseDoubleClick -= value;
1.150 + }
1.151 + }
1.152 +
1.153 + public event MouseEventHandler MouseDown {
1.154 + add {
1.155 + if (genericNotifyIcon != null)
1.156 + genericNotifyIcon.MouseDown += value;
1.157 + else
1.158 + windowsNotifyIcon.MouseDown += value;
1.159 + }
1.160 + remove {
1.161 + if (genericNotifyIcon != null)
1.162 + genericNotifyIcon.MouseDown -= value;
1.163 + else
1.164 + windowsNotifyIcon.MouseDown -= value;
1.165 + }
1.166 + }
1.167 +
1.168 + public event MouseEventHandler MouseMove {
1.169 + add {
1.170 + if (genericNotifyIcon != null)
1.171 + genericNotifyIcon.MouseMove += value;
1.172 + else
1.173 + windowsNotifyIcon.MouseMove += value;
1.174 + }
1.175 + remove {
1.176 + if (genericNotifyIcon != null)
1.177 + genericNotifyIcon.MouseMove -= value;
1.178 + else
1.179 + windowsNotifyIcon.MouseMove -= value;
1.180 + }
1.181 + }
1.182 +
1.183 + public event MouseEventHandler MouseUp {
1.184 + add {
1.185 + if (genericNotifyIcon != null)
1.186 + genericNotifyIcon.MouseUp += value;
1.187 + else
1.188 + windowsNotifyIcon.MouseUp += value;
1.189 + }
1.190 + remove {
1.191 + if (genericNotifyIcon != null)
1.192 + genericNotifyIcon.MouseUp -= value;
1.193 + else
1.194 + windowsNotifyIcon.MouseUp -= value;
1.195 + }
1.196 + }
1.197 +
1.198 + public string BalloonTipText {
1.199 + get {
1.200 + if (genericNotifyIcon != null)
1.201 + return genericNotifyIcon.BalloonTipText;
1.202 + else
1.203 + return windowsNotifyIcon.BalloonTipText;
1.204 + }
1.205 + set {
1.206 + if (genericNotifyIcon != null)
1.207 + genericNotifyIcon.BalloonTipText = value;
1.208 + else
1.209 + windowsNotifyIcon.BalloonTipText = value;
1.210 + }
1.211 + }
1.212 +
1.213 + public ToolTipIcon BalloonTipIcon {
1.214 + get {
1.215 + if (genericNotifyIcon != null)
1.216 + return genericNotifyIcon.BalloonTipIcon;
1.217 + else
1.218 + return windowsNotifyIcon.BalloonTipIcon;
1.219 + }
1.220 + set {
1.221 + if (genericNotifyIcon != null)
1.222 + genericNotifyIcon.BalloonTipIcon = value;
1.223 + else
1.224 + windowsNotifyIcon.BalloonTipIcon = value;
1.225 + }
1.226 + }
1.227 +
1.228 + public string BalloonTipTitle {
1.229 + get {
1.230 + if (genericNotifyIcon != null)
1.231 + return genericNotifyIcon.BalloonTipTitle;
1.232 + else
1.233 + return windowsNotifyIcon.BalloonTipTitle;
1.234 + }
1.235 + set {
1.236 + if (genericNotifyIcon != null)
1.237 + genericNotifyIcon.BalloonTipTitle = value;
1.238 + else
1.239 + windowsNotifyIcon.BalloonTipTitle = value;
1.240 + }
1.241 + }
1.242 +
1.243 + public ContextMenu ContextMenu {
1.244 + get {
1.245 + if (genericNotifyIcon != null)
1.246 + return genericNotifyIcon.ContextMenu;
1.247 + else
1.248 + return windowsNotifyIcon.ContextMenu;
1.249 + }
1.250 + set {
1.251 + if (genericNotifyIcon != null)
1.252 + genericNotifyIcon.ContextMenu = value;
1.253 + else
1.254 + windowsNotifyIcon.ContextMenu = value;
1.255 + }
1.256 + }
1.257 +
1.258 + public ContextMenuStrip ContextMenuStrip {
1.259 + get {
1.260 + if (genericNotifyIcon != null)
1.261 + return genericNotifyIcon.ContextMenuStrip;
1.262 + else
1.263 + return windowsNotifyIcon.ContextMenuStrip;
1.264 + }
1.265 + set {
1.266 + if (genericNotifyIcon != null)
1.267 + genericNotifyIcon.ContextMenuStrip = value;
1.268 + else
1.269 + windowsNotifyIcon.ContextMenuStrip = value;
1.270 + }
1.271 + }
1.272 +
1.273 public object Tag { get; set; }
1.274
1.275 public Icon Icon {
1.276 get {
1.277 - return icon;
1.278 + if (genericNotifyIcon != null)
1.279 + return genericNotifyIcon.Icon;
1.280 + else
1.281 + return windowsNotifyIcon.Icon;
1.282 }
1.283 set {
1.284 - if (icon != value) {
1.285 - icon = value;
1.286 - UpdateNotifyIcon(visible);
1.287 - }
1.288 + if (genericNotifyIcon != null)
1.289 + genericNotifyIcon.Icon = value;
1.290 + else
1.291 + windowsNotifyIcon.Icon = value;
1.292 }
1.293 }
1.294
1.295 public string Text {
1.296 get {
1.297 - return text;
1.298 + if (genericNotifyIcon != null)
1.299 + return genericNotifyIcon.Text;
1.300 + else
1.301 + return windowsNotifyIcon.Text;
1.302 }
1.303 set {
1.304 - if (value == null)
1.305 - value = "";
1.306 -
1.307 - if (value.Length > 63)
1.308 - throw new ArgumentOutOfRangeException();
1.309 -
1.310 - if (!value.Equals(text)) {
1.311 - text = value;
1.312 -
1.313 - if (visible)
1.314 - UpdateNotifyIcon(visible);
1.315 - }
1.316 + if (genericNotifyIcon != null)
1.317 + genericNotifyIcon.Text = value;
1.318 + else
1.319 + windowsNotifyIcon.Text = value;
1.320 }
1.321 }
1.322
1.323 public bool Visible {
1.324 get {
1.325 - return visible;
1.326 + if (genericNotifyIcon != null)
1.327 + return genericNotifyIcon.Visible;
1.328 + else
1.329 + return windowsNotifyIcon.Visible;
1.330 }
1.331 set {
1.332 - if (visible != value) {
1.333 - visible = value;
1.334 - UpdateNotifyIcon(visible);
1.335 - }
1.336 + if (genericNotifyIcon != null)
1.337 + genericNotifyIcon.Visible = value;
1.338 + else
1.339 + windowsNotifyIcon.Visible = value;
1.340 }
1.341 }
1.342 -
1.343 - public NotifyIconAdv() {
1.344 - BalloonTipText = "";
1.345 - BalloonTipTitle = "";
1.346
1.347 - commandDispatch = typeof(Form).Assembly.
1.348 - GetType("System.Windows.Forms.Command").GetMethod("DispatchID",
1.349 - BindingFlags.Static | BindingFlags.NonPublic | BindingFlags.Public,
1.350 - null, new Type[] { typeof(int) }, null);
1.351 -
1.352 - id = ++NotifyIconAdv.nextId;
1.353 - window = new NotifyIconNativeWindow(this);
1.354 - UpdateNotifyIcon(visible);
1.355 - }
1.356 -
1.357 - protected override void Dispose(bool disposing) {
1.358 - if (disposing) {
1.359 - if (window != null) {
1.360 - icon = null;
1.361 - text = "";
1.362 - UpdateNotifyIcon(false);
1.363 - window.DestroyHandle();
1.364 - window = null;
1.365 - ContextMenu = null;
1.366 - ContextMenuStrip = null;
1.367 - }
1.368 - } else {
1.369 - if (window != null && window.Handle != IntPtr.Zero) {
1.370 - NativeMethods.PostMessage(
1.371 - new HandleRef(window, window.Handle), WM_CLOSE, 0, 0);
1.372 - window.ReleaseHandle();
1.373 - }
1.374 - }
1.375 - base.Dispose(disposing);
1.376 + public void Dispose() {
1.377 + if (genericNotifyIcon != null)
1.378 + genericNotifyIcon.Dispose();
1.379 + else
1.380 + windowsNotifyIcon.Dispose();
1.381 }
1.382
1.383 public void ShowBalloonTip(int timeout) {
1.384 @@ -132,371 +315,493 @@
1.385 }
1.386
1.387 public void ShowBalloonTip(int timeout, string tipTitle, string tipText,
1.388 - ToolTipIcon tipIcon)
1.389 - {
1.390 - if (timeout < 0)
1.391 - throw new ArgumentOutOfRangeException("timeout");
1.392 + ToolTipIcon tipIcon) {
1.393 + if (genericNotifyIcon != null)
1.394 + genericNotifyIcon.ShowBalloonTip(timeout, tipTitle, tipText, tipIcon);
1.395 + else
1.396 + windowsNotifyIcon.ShowBalloonTip(timeout, tipTitle, tipText, tipIcon);
1.397 + }
1.398 +
1.399 + private class NotifyIconWindowsImplementation : Component {
1.400
1.401 - if (string.IsNullOrEmpty(tipText))
1.402 - throw new ArgumentException("tipText");
1.403 + private static int nextId = 0;
1.404
1.405 - if (DesignMode)
1.406 - return;
1.407 + private object syncObj = new object();
1.408 + private Icon icon;
1.409 + private string text = "";
1.410 + private int id;
1.411 + private bool created;
1.412 + private NotifyIconNativeWindow window;
1.413 + private bool doubleClickDown;
1.414 + private bool visible;
1.415 + private MethodInfo commandDispatch;
1.416
1.417 - if (created) {
1.418 - NativeMethods.NotifyIconData data = new NativeMethods.NotifyIconData();
1.419 - if (window.Handle == IntPtr.Zero)
1.420 - window.CreateHandle(new CreateParams());
1.421 + public event EventHandler BalloonTipClicked;
1.422 + public event EventHandler BalloonTipClosed;
1.423 + public event EventHandler BalloonTipShown;
1.424 + public event EventHandler Click;
1.425 + public event EventHandler DoubleClick;
1.426 + public event MouseEventHandler MouseClick;
1.427 + public event MouseEventHandler MouseDoubleClick;
1.428 + public event MouseEventHandler MouseDown;
1.429 + public event MouseEventHandler MouseMove;
1.430 + public event MouseEventHandler MouseUp;
1.431
1.432 - data.Window = window.Handle;
1.433 - data.ID = id;
1.434 - data.Flags = NativeMethods.NotifyIconDataFlags.Info;
1.435 - data.TimeoutOrVersion = timeout;
1.436 - data.InfoTitle = tipTitle;
1.437 - data.Info = tipText;
1.438 - data.InfoFlags = (int)tipIcon;
1.439 + public string BalloonTipText { get; set; }
1.440 + public ToolTipIcon BalloonTipIcon { get; set; }
1.441 + public string BalloonTipTitle { get; set; }
1.442 + public ContextMenu ContextMenu { get; set; }
1.443 + public ContextMenuStrip ContextMenuStrip { get; set; }
1.444 + public object Tag { get; set; }
1.445
1.446 - NativeMethods.Shell_NotifyIcon(
1.447 - NativeMethods.NotifyIconMessage.Modify, data);
1.448 - }
1.449 - }
1.450 -
1.451 - private void ShowContextMenu() {
1.452 - if (ContextMenu == null && ContextMenuStrip == null)
1.453 - return;
1.454 -
1.455 - NativeMethods.Point p = new NativeMethods.Point();
1.456 - NativeMethods.GetCursorPos(ref p);
1.457 - NativeMethods.SetForegroundWindow(
1.458 - new HandleRef(window, window.Handle));
1.459 -
1.460 - if (ContextMenu != null) {
1.461 - ContextMenu.GetType().InvokeMember("OnPopup",
1.462 - BindingFlags.NonPublic | BindingFlags.InvokeMethod |
1.463 - BindingFlags.Instance, null, ContextMenu,
1.464 - new Object[] { System.EventArgs.Empty });
1.465 -
1.466 - NativeMethods.TrackPopupMenuEx(
1.467 - new HandleRef(ContextMenu, ContextMenu.Handle), 72,
1.468 - p.x, p.y, new HandleRef(window, window.Handle),
1.469 - IntPtr.Zero);
1.470 -
1.471 - NativeMethods.PostMessage(
1.472 - new HandleRef(window, window.Handle), WM_NULL, 0, 0);
1.473 - return;
1.474 - }
1.475 -
1.476 - if (ContextMenuStrip != null)
1.477 - ContextMenuStrip.GetType().InvokeMember("ShowInTaskbar",
1.478 - BindingFlags.NonPublic | BindingFlags.InvokeMethod |
1.479 - BindingFlags.Instance, null, ContextMenuStrip,
1.480 - new Object[] { p.x, p.y });
1.481 - }
1.482 -
1.483 - private void UpdateNotifyIcon(bool showNotifyIcon) {
1.484 - if (DesignMode)
1.485 - return;
1.486 -
1.487 - lock (syncObj) {
1.488 - window.LockReference(showNotifyIcon);
1.489 -
1.490 - NativeMethods.NotifyIconData data = new NativeMethods.NotifyIconData();
1.491 - data.CallbackMessage = WM_TRAYMOUSEMESSAGE;
1.492 - data.Flags = NativeMethods.NotifyIconDataFlags.Message;
1.493 -
1.494 - if (showNotifyIcon && window.Handle == IntPtr.Zero)
1.495 - window.CreateHandle(new CreateParams());
1.496 -
1.497 - data.Window = window.Handle;
1.498 - data.ID = id;
1.499 -
1.500 - if (icon != null) {
1.501 - data.Flags |= NativeMethods.NotifyIconDataFlags.Icon;
1.502 - data.Icon = icon.Handle;
1.503 + public Icon Icon {
1.504 + get {
1.505 + return icon;
1.506 }
1.507 -
1.508 - data.Flags |= NativeMethods.NotifyIconDataFlags.Tip;
1.509 - data.Tip = text;
1.510 -
1.511 - if (showNotifyIcon && icon != null) {
1.512 - if (!created) {
1.513 - int i = 0;
1.514 - do {
1.515 - created = NativeMethods.Shell_NotifyIcon(
1.516 - NativeMethods.NotifyIconMessage.Add, data);
1.517 - if (!created) {
1.518 - System.Threading.Thread.Sleep(200);
1.519 - i++;
1.520 - }
1.521 - } while (!created && i < 40);
1.522 - } else {
1.523 - NativeMethods.Shell_NotifyIcon(
1.524 - NativeMethods.NotifyIconMessage.Modify, data);
1.525 - }
1.526 - } else {
1.527 - if (created) {
1.528 - int i = 0;
1.529 - bool deleted = false;
1.530 - do {
1.531 - deleted = NativeMethods.Shell_NotifyIcon(
1.532 - NativeMethods.NotifyIconMessage.Delete, data);
1.533 - if (!deleted) {
1.534 - System.Threading.Thread.Sleep(200);
1.535 - i++;
1.536 - }
1.537 - } while (!deleted && i < 40);
1.538 - created = false;
1.539 + set {
1.540 + if (icon != value) {
1.541 + icon = value;
1.542 + UpdateNotifyIcon(visible);
1.543 }
1.544 }
1.545 }
1.546 - }
1.547
1.548 - private void ProcessMouseDown(ref Message message, MouseButtons button,
1.549 - bool doubleClick)
1.550 - {
1.551 - if (doubleClick) {
1.552 - if (DoubleClick != null)
1.553 - DoubleClick(this, new MouseEventArgs(button, 2, 0, 0, 0));
1.554 + public string Text {
1.555 + get {
1.556 + return text;
1.557 + }
1.558 + set {
1.559 + if (value == null)
1.560 + value = "";
1.561
1.562 - if (MouseDoubleClick != null)
1.563 - MouseDoubleClick(this, new MouseEventArgs(button, 2, 0, 0, 0));
1.564 + if (value.Length > 63)
1.565 + throw new ArgumentOutOfRangeException();
1.566
1.567 - doubleClickDown = true;
1.568 + if (!value.Equals(text)) {
1.569 + text = value;
1.570 +
1.571 + if (visible)
1.572 + UpdateNotifyIcon(visible);
1.573 + }
1.574 + }
1.575 }
1.576
1.577 - if (MouseDown != null)
1.578 - MouseDown(this,
1.579 - new MouseEventArgs(button, doubleClick ? 2 : 1, 0, 0, 0));
1.580 - }
1.581 -
1.582 - private void ProcessMouseUp(ref Message message, MouseButtons button) {
1.583 - if (MouseUp != null)
1.584 - MouseUp(this, new MouseEventArgs(button, 0, 0, 0, 0));
1.585 -
1.586 - if (!doubleClickDown) {
1.587 - if (Click != null)
1.588 - Click(this, new MouseEventArgs(button, 0, 0, 0, 0));
1.589 -
1.590 - if (MouseClick != null)
1.591 - MouseClick(this, new MouseEventArgs(button, 0, 0, 0, 0));
1.592 - }
1.593 - doubleClickDown = false;
1.594 - }
1.595 -
1.596 - private void ProcessInitMenuPopup(ref Message message) {
1.597 - if (ContextMenu != null &&
1.598 - (bool)ContextMenu.GetType().InvokeMember("ProcessInitMenuPopup",
1.599 - BindingFlags.NonPublic | BindingFlags.InvokeMethod |
1.600 - BindingFlags.Instance, null, ContextMenu,
1.601 - new Object[] { message.WParam })) {
1.602 - return;
1.603 - }
1.604 - window.DefWndProc(ref message);
1.605 - }
1.606 -
1.607 - private void WndProc(ref Message message) {
1.608 - switch (message.Msg) {
1.609 - case WM_DESTROY:
1.610 - UpdateNotifyIcon(false);
1.611 - return;
1.612 - case WM_COMMAND:
1.613 - if (message.LParam != IntPtr.Zero) {
1.614 - window.DefWndProc(ref message);
1.615 - return;
1.616 + public bool Visible {
1.617 + get {
1.618 + return visible;
1.619 + }
1.620 + set {
1.621 + if (visible != value) {
1.622 + visible = value;
1.623 + UpdateNotifyIcon(visible);
1.624 }
1.625 - commandDispatch.Invoke(null, new object[] {
1.626 - message.WParam.ToInt32() & 0xFFFF });
1.627 - return;
1.628 - case WM_INITMENUPOPUP:
1.629 - ProcessInitMenuPopup(ref message);
1.630 - return;
1.631 - case WM_TRAYMOUSEMESSAGE:
1.632 - switch ((int)message.LParam) {
1.633 - case WM_MOUSEMOVE:
1.634 - if (MouseMove != null)
1.635 - MouseMove(this,
1.636 - new MouseEventArgs(Control.MouseButtons, 0, 0, 0, 0));
1.637 - return;
1.638 - case WM_LBUTTONDOWN:
1.639 - ProcessMouseDown(ref message, MouseButtons.Left, false);
1.640 - return;
1.641 - case WM_LBUTTONUP:
1.642 - ProcessMouseUp(ref message, MouseButtons.Left);
1.643 - return;
1.644 - case WM_LBUTTONDBLCLK:
1.645 - ProcessMouseDown(ref message, MouseButtons.Left, true);
1.646 - return;
1.647 - case WM_RBUTTONDOWN:
1.648 - ProcessMouseDown(ref message, MouseButtons.Right, false);
1.649 - return;
1.650 - case WM_RBUTTONUP:
1.651 - if (ContextMenu != null || ContextMenuStrip != null)
1.652 - ShowContextMenu();
1.653 - ProcessMouseUp(ref message, MouseButtons.Right);
1.654 - return;
1.655 - case WM_RBUTTONDBLCLK:
1.656 - ProcessMouseDown(ref message, MouseButtons.Right, true);
1.657 - return;
1.658 - case WM_MBUTTONDOWN:
1.659 - ProcessMouseDown(ref message, MouseButtons.Middle, false);
1.660 - return;
1.661 - case WM_MBUTTONUP:
1.662 - ProcessMouseUp(ref message, MouseButtons.Middle);
1.663 - return;
1.664 - case WM_MBUTTONDBLCLK:
1.665 - ProcessMouseDown(ref message, MouseButtons.Middle, true);
1.666 - return;
1.667 - case NIN_BALLOONSHOW:
1.668 - if (BalloonTipShown != null)
1.669 - BalloonTipShown(this, EventArgs.Empty);
1.670 - return;
1.671 - case NIN_BALLOONHIDE:
1.672 - case NIN_BALLOONTIMEOUT:
1.673 - if (BalloonTipClosed != null)
1.674 - BalloonTipClosed(this, EventArgs.Empty);
1.675 - return;
1.676 - case NIN_BALLOONUSERCLICK:
1.677 - if (BalloonTipClicked != null)
1.678 - BalloonTipClicked(this, EventArgs.Empty);
1.679 - return;
1.680 - default:
1.681 - return;
1.682 - }
1.683 + }
1.684 }
1.685
1.686 - if (message.Msg == NotifyIconAdv.WM_TASKBARCREATED) {
1.687 - lock (syncObj) {
1.688 - created = false;
1.689 - }
1.690 + public NotifyIconWindowsImplementation() {
1.691 + BalloonTipText = "";
1.692 + BalloonTipTitle = "";
1.693 +
1.694 + commandDispatch = typeof(Form).Assembly.
1.695 + GetType("System.Windows.Forms.Command").GetMethod("DispatchID",
1.696 + BindingFlags.Static | BindingFlags.NonPublic | BindingFlags.Public,
1.697 + null, new Type[] { typeof(int) }, null);
1.698 +
1.699 + id = ++NotifyIconWindowsImplementation.nextId;
1.700 + window = new NotifyIconNativeWindow(this);
1.701 UpdateNotifyIcon(visible);
1.702 }
1.703
1.704 - window.DefWndProc(ref message);
1.705 - }
1.706 -
1.707 - private class NotifyIconNativeWindow : NativeWindow {
1.708 - private NotifyIconAdv reference;
1.709 - private GCHandle referenceHandle;
1.710 -
1.711 - internal NotifyIconNativeWindow(NotifyIconAdv component) {
1.712 - this.reference = component;
1.713 + protected override void Dispose(bool disposing) {
1.714 + if (disposing) {
1.715 + if (window != null) {
1.716 + icon = null;
1.717 + text = "";
1.718 + UpdateNotifyIcon(false);
1.719 + window.DestroyHandle();
1.720 + window = null;
1.721 + ContextMenu = null;
1.722 + ContextMenuStrip = null;
1.723 + }
1.724 + } else {
1.725 + if (window != null && window.Handle != IntPtr.Zero) {
1.726 + NativeMethods.PostMessage(
1.727 + new HandleRef(window, window.Handle), WM_CLOSE, 0, 0);
1.728 + window.ReleaseHandle();
1.729 + }
1.730 + }
1.731 + base.Dispose(disposing);
1.732 }
1.733
1.734 - ~NotifyIconNativeWindow() {
1.735 - if (base.Handle != IntPtr.Zero)
1.736 - NativeMethods.PostMessage(
1.737 - new HandleRef(this, base.Handle), WM_CLOSE, 0, 0);
1.738 + public void ShowBalloonTip(int timeout) {
1.739 + ShowBalloonTip(timeout, BalloonTipTitle, BalloonTipText, BalloonTipIcon);
1.740 }
1.741
1.742 - public void LockReference(bool locked) {
1.743 - if (locked) {
1.744 - if (!referenceHandle.IsAllocated) {
1.745 - referenceHandle = GCHandle.Alloc(reference, GCHandleType.Normal);
1.746 - return;
1.747 - }
1.748 - } else {
1.749 - if (referenceHandle.IsAllocated)
1.750 - referenceHandle.Free();
1.751 + public void ShowBalloonTip(int timeout, string tipTitle, string tipText,
1.752 + ToolTipIcon tipIcon) {
1.753 + if (timeout < 0)
1.754 + throw new ArgumentOutOfRangeException("timeout");
1.755 +
1.756 + if (string.IsNullOrEmpty(tipText))
1.757 + throw new ArgumentException("tipText");
1.758 +
1.759 + if (DesignMode)
1.760 + return;
1.761 +
1.762 + if (created) {
1.763 + NativeMethods.NotifyIconData data = new NativeMethods.NotifyIconData();
1.764 + if (window.Handle == IntPtr.Zero)
1.765 + window.CreateHandle(new CreateParams());
1.766 +
1.767 + data.Window = window.Handle;
1.768 + data.ID = id;
1.769 + data.Flags = NativeMethods.NotifyIconDataFlags.Info;
1.770 + data.TimeoutOrVersion = timeout;
1.771 + data.InfoTitle = tipTitle;
1.772 + data.Info = tipText;
1.773 + data.InfoFlags = (int)tipIcon;
1.774 +
1.775 + NativeMethods.Shell_NotifyIcon(
1.776 + NativeMethods.NotifyIconMessage.Modify, data);
1.777 }
1.778 }
1.779
1.780 - protected override void OnThreadException(Exception e) {
1.781 - Application.OnThreadException(e);
1.782 + private void ShowContextMenu() {
1.783 + if (ContextMenu == null && ContextMenuStrip == null)
1.784 + return;
1.785 +
1.786 + NativeMethods.Point p = new NativeMethods.Point();
1.787 + NativeMethods.GetCursorPos(ref p);
1.788 + NativeMethods.SetForegroundWindow(
1.789 + new HandleRef(window, window.Handle));
1.790 +
1.791 + if (ContextMenu != null) {
1.792 + ContextMenu.GetType().InvokeMember("OnPopup",
1.793 + BindingFlags.NonPublic | BindingFlags.InvokeMethod |
1.794 + BindingFlags.Instance, null, ContextMenu,
1.795 + new Object[] { System.EventArgs.Empty });
1.796 +
1.797 + NativeMethods.TrackPopupMenuEx(
1.798 + new HandleRef(ContextMenu, ContextMenu.Handle), 72,
1.799 + p.x, p.y, new HandleRef(window, window.Handle),
1.800 + IntPtr.Zero);
1.801 +
1.802 + NativeMethods.PostMessage(
1.803 + new HandleRef(window, window.Handle), WM_NULL, 0, 0);
1.804 + return;
1.805 + }
1.806 +
1.807 + if (ContextMenuStrip != null)
1.808 + ContextMenuStrip.GetType().InvokeMember("ShowInTaskbar",
1.809 + BindingFlags.NonPublic | BindingFlags.InvokeMethod |
1.810 + BindingFlags.Instance, null, ContextMenuStrip,
1.811 + new Object[] { p.x, p.y });
1.812 }
1.813
1.814 - protected override void WndProc(ref Message m) {
1.815 - reference.WndProc(ref m);
1.816 - }
1.817 - }
1.818 + private void UpdateNotifyIcon(bool showNotifyIcon) {
1.819 + if (DesignMode)
1.820 + return;
1.821
1.822 - private const int WM_NULL = 0x00;
1.823 - private const int WM_DESTROY = 0x02;
1.824 - private const int WM_CLOSE = 0x10;
1.825 - private const int WM_COMMAND = 0x111;
1.826 - private const int WM_INITMENUPOPUP = 0x117;
1.827 - private const int WM_MOUSEMOVE = 0x200;
1.828 - private const int WM_LBUTTONDOWN = 0x201;
1.829 - private const int WM_LBUTTONUP = 0x202;
1.830 - private const int WM_LBUTTONDBLCLK = 0x203;
1.831 - private const int WM_RBUTTONDOWN = 0x204;
1.832 - private const int WM_RBUTTONUP = 0x205;
1.833 - private const int WM_RBUTTONDBLCLK = 0x206;
1.834 - private const int WM_MBUTTONDOWN = 0x207;
1.835 - private const int WM_MBUTTONUP = 0x208;
1.836 - private const int WM_MBUTTONDBLCLK = 0x209;
1.837 - private const int WM_TRAYMOUSEMESSAGE = 0x800;
1.838 + lock (syncObj) {
1.839 + window.LockReference(showNotifyIcon);
1.840
1.841 - private const int NIN_BALLOONSHOW = 0x402;
1.842 - private const int NIN_BALLOONHIDE = 0x403;
1.843 - private const int NIN_BALLOONTIMEOUT = 0x404;
1.844 - private const int NIN_BALLOONUSERCLICK = 0x405;
1.845 + NativeMethods.NotifyIconData data = new NativeMethods.NotifyIconData();
1.846 + data.CallbackMessage = WM_TRAYMOUSEMESSAGE;
1.847 + data.Flags = NativeMethods.NotifyIconDataFlags.Message;
1.848
1.849 - private static int WM_TASKBARCREATED =
1.850 - NativeMethods.RegisterWindowMessage("TaskbarCreated");
1.851 + if (showNotifyIcon && window.Handle == IntPtr.Zero)
1.852 + window.CreateHandle(new CreateParams());
1.853
1.854 - private static class NativeMethods {
1.855 - [DllImport("user32.dll", CharSet = CharSet.Auto)]
1.856 - public static extern IntPtr PostMessage(HandleRef hwnd, int msg,
1.857 - int wparam, int lparam);
1.858 + data.Window = window.Handle;
1.859 + data.ID = id;
1.860
1.861 - [DllImport("user32.dll", CharSet = CharSet.Auto)]
1.862 - public static extern int RegisterWindowMessage(string msg);
1.863 + if (icon != null) {
1.864 + data.Flags |= NativeMethods.NotifyIconDataFlags.Icon;
1.865 + data.Icon = icon.Handle;
1.866 + }
1.867
1.868 - [Flags]
1.869 - public enum NotifyIconDataFlags : int {
1.870 - Message = 0x1,
1.871 - Icon = 0x2,
1.872 - Tip = 0x4,
1.873 - State = 0x8,
1.874 - Info = 0x10
1.875 + data.Flags |= NativeMethods.NotifyIconDataFlags.Tip;
1.876 + data.Tip = text;
1.877 +
1.878 + if (showNotifyIcon && icon != null) {
1.879 + if (!created) {
1.880 + int i = 0;
1.881 + do {
1.882 + created = NativeMethods.Shell_NotifyIcon(
1.883 + NativeMethods.NotifyIconMessage.Add, data);
1.884 + if (!created) {
1.885 + System.Threading.Thread.Sleep(200);
1.886 + i++;
1.887 + }
1.888 + } while (!created && i < 40);
1.889 + } else {
1.890 + NativeMethods.Shell_NotifyIcon(
1.891 + NativeMethods.NotifyIconMessage.Modify, data);
1.892 + }
1.893 + } else {
1.894 + if (created) {
1.895 + int i = 0;
1.896 + bool deleted = false;
1.897 + do {
1.898 + deleted = NativeMethods.Shell_NotifyIcon(
1.899 + NativeMethods.NotifyIconMessage.Delete, data);
1.900 + if (!deleted) {
1.901 + System.Threading.Thread.Sleep(200);
1.902 + i++;
1.903 + }
1.904 + } while (!deleted && i < 40);
1.905 + created = false;
1.906 + }
1.907 + }
1.908 + }
1.909 }
1.910
1.911 - [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)]
1.912 - public class NotifyIconData {
1.913 - private int Size = Marshal.SizeOf(typeof(NotifyIconData));
1.914 - public IntPtr Window;
1.915 - public int ID;
1.916 - public NotifyIconDataFlags Flags;
1.917 - public int CallbackMessage;
1.918 - public IntPtr Icon;
1.919 - [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 128)]
1.920 - public string Tip;
1.921 - public int State;
1.922 - public int StateMask;
1.923 - [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
1.924 - public string Info;
1.925 - public int TimeoutOrVersion;
1.926 - [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 64)]
1.927 - public string InfoTitle;
1.928 - public int InfoFlags;
1.929 + private void ProcessMouseDown(ref Message message, MouseButtons button,
1.930 + bool doubleClick) {
1.931 + if (doubleClick) {
1.932 + if (DoubleClick != null)
1.933 + DoubleClick(this, new MouseEventArgs(button, 2, 0, 0, 0));
1.934 +
1.935 + if (MouseDoubleClick != null)
1.936 + MouseDoubleClick(this, new MouseEventArgs(button, 2, 0, 0, 0));
1.937 +
1.938 + doubleClickDown = true;
1.939 + }
1.940 +
1.941 + if (MouseDown != null)
1.942 + MouseDown(this,
1.943 + new MouseEventArgs(button, doubleClick ? 2 : 1, 0, 0, 0));
1.944 }
1.945
1.946 - public enum NotifyIconMessage : int {
1.947 - Add = 0x0,
1.948 - Modify = 0x1,
1.949 - Delete = 0x2
1.950 + private void ProcessMouseUp(ref Message message, MouseButtons button) {
1.951 + if (MouseUp != null)
1.952 + MouseUp(this, new MouseEventArgs(button, 0, 0, 0, 0));
1.953 +
1.954 + if (!doubleClickDown) {
1.955 + if (Click != null)
1.956 + Click(this, new MouseEventArgs(button, 0, 0, 0, 0));
1.957 +
1.958 + if (MouseClick != null)
1.959 + MouseClick(this, new MouseEventArgs(button, 0, 0, 0, 0));
1.960 + }
1.961 + doubleClickDown = false;
1.962 }
1.963
1.964 - [DllImport("shell32.dll", CharSet = CharSet.Auto)]
1.965 - [return: MarshalAs(UnmanagedType.Bool)]
1.966 - public static extern bool Shell_NotifyIcon(NotifyIconMessage message,
1.967 - NotifyIconData pnid);
1.968 -
1.969 - [DllImport("user32.dll", CharSet = CharSet.Auto, ExactSpelling = true)]
1.970 - public static extern bool TrackPopupMenuEx(HandleRef hmenu, int fuFlags,
1.971 - int x, int y, HandleRef hwnd, IntPtr tpm);
1.972 -
1.973 - [StructLayout(LayoutKind.Sequential)]
1.974 - public struct Point {
1.975 - public int x;
1.976 - public int y;
1.977 + private void ProcessInitMenuPopup(ref Message message) {
1.978 + if (ContextMenu != null &&
1.979 + (bool)ContextMenu.GetType().InvokeMember("ProcessInitMenuPopup",
1.980 + BindingFlags.NonPublic | BindingFlags.InvokeMethod |
1.981 + BindingFlags.Instance, null, ContextMenu,
1.982 + new Object[] { message.WParam })) {
1.983 + return;
1.984 + }
1.985 + window.DefWndProc(ref message);
1.986 }
1.987
1.988 - [DllImport("user32.dll", CharSet = CharSet.Auto, ExactSpelling = true)]
1.989 - public static extern bool GetCursorPos(ref Point point);
1.990 + private void WndProc(ref Message message) {
1.991 + switch (message.Msg) {
1.992 + case WM_DESTROY:
1.993 + UpdateNotifyIcon(false);
1.994 + return;
1.995 + case WM_COMMAND:
1.996 + if (message.LParam != IntPtr.Zero) {
1.997 + window.DefWndProc(ref message);
1.998 + return;
1.999 + }
1.1000 + commandDispatch.Invoke(null, new object[] {
1.1001 + message.WParam.ToInt32() & 0xFFFF });
1.1002 + return;
1.1003 + case WM_INITMENUPOPUP:
1.1004 + ProcessInitMenuPopup(ref message);
1.1005 + return;
1.1006 + case WM_TRAYMOUSEMESSAGE:
1.1007 + switch ((int)message.LParam) {
1.1008 + case WM_MOUSEMOVE:
1.1009 + if (MouseMove != null)
1.1010 + MouseMove(this,
1.1011 + new MouseEventArgs(Control.MouseButtons, 0, 0, 0, 0));
1.1012 + return;
1.1013 + case WM_LBUTTONDOWN:
1.1014 + ProcessMouseDown(ref message, MouseButtons.Left, false);
1.1015 + return;
1.1016 + case WM_LBUTTONUP:
1.1017 + ProcessMouseUp(ref message, MouseButtons.Left);
1.1018 + return;
1.1019 + case WM_LBUTTONDBLCLK:
1.1020 + ProcessMouseDown(ref message, MouseButtons.Left, true);
1.1021 + return;
1.1022 + case WM_RBUTTONDOWN:
1.1023 + ProcessMouseDown(ref message, MouseButtons.Right, false);
1.1024 + return;
1.1025 + case WM_RBUTTONUP:
1.1026 + if (ContextMenu != null || ContextMenuStrip != null)
1.1027 + ShowContextMenu();
1.1028 + ProcessMouseUp(ref message, MouseButtons.Right);
1.1029 + return;
1.1030 + case WM_RBUTTONDBLCLK:
1.1031 + ProcessMouseDown(ref message, MouseButtons.Right, true);
1.1032 + return;
1.1033 + case WM_MBUTTONDOWN:
1.1034 + ProcessMouseDown(ref message, MouseButtons.Middle, false);
1.1035 + return;
1.1036 + case WM_MBUTTONUP:
1.1037 + ProcessMouseUp(ref message, MouseButtons.Middle);
1.1038 + return;
1.1039 + case WM_MBUTTONDBLCLK:
1.1040 + ProcessMouseDown(ref message, MouseButtons.Middle, true);
1.1041 + return;
1.1042 + case NIN_BALLOONSHOW:
1.1043 + if (BalloonTipShown != null)
1.1044 + BalloonTipShown(this, EventArgs.Empty);
1.1045 + return;
1.1046 + case NIN_BALLOONHIDE:
1.1047 + case NIN_BALLOONTIMEOUT:
1.1048 + if (BalloonTipClosed != null)
1.1049 + BalloonTipClosed(this, EventArgs.Empty);
1.1050 + return;
1.1051 + case NIN_BALLOONUSERCLICK:
1.1052 + if (BalloonTipClicked != null)
1.1053 + BalloonTipClicked(this, EventArgs.Empty);
1.1054 + return;
1.1055 + default:
1.1056 + return;
1.1057 + }
1.1058 + }
1.1059
1.1060 - [DllImport("user32.dll", CharSet = CharSet.Auto, ExactSpelling = true)]
1.1061 - public static extern bool SetForegroundWindow(HandleRef hWnd);
1.1062 + if (message.Msg == NotifyIconWindowsImplementation.WM_TASKBARCREATED) {
1.1063 + lock (syncObj) {
1.1064 + created = false;
1.1065 + }
1.1066 + UpdateNotifyIcon(visible);
1.1067 + }
1.1068 +
1.1069 + window.DefWndProc(ref message);
1.1070 + }
1.1071 +
1.1072 + private class NotifyIconNativeWindow : NativeWindow {
1.1073 + private NotifyIconWindowsImplementation reference;
1.1074 + private GCHandle referenceHandle;
1.1075 +
1.1076 + internal NotifyIconNativeWindow(NotifyIconWindowsImplementation component) {
1.1077 + this.reference = component;
1.1078 + }
1.1079 +
1.1080 + ~NotifyIconNativeWindow() {
1.1081 + if (base.Handle != IntPtr.Zero)
1.1082 + NativeMethods.PostMessage(
1.1083 + new HandleRef(this, base.Handle), WM_CLOSE, 0, 0);
1.1084 + }
1.1085 +
1.1086 + public void LockReference(bool locked) {
1.1087 + if (locked) {
1.1088 + if (!referenceHandle.IsAllocated) {
1.1089 + referenceHandle = GCHandle.Alloc(reference, GCHandleType.Normal);
1.1090 + return;
1.1091 + }
1.1092 + } else {
1.1093 + if (referenceHandle.IsAllocated)
1.1094 + referenceHandle.Free();
1.1095 + }
1.1096 + }
1.1097 +
1.1098 + protected override void OnThreadException(Exception e) {
1.1099 + Application.OnThreadException(e);
1.1100 + }
1.1101 +
1.1102 + protected override void WndProc(ref Message m) {
1.1103 + reference.WndProc(ref m);
1.1104 + }
1.1105 + }
1.1106 +
1.1107 + private const int WM_NULL = 0x00;
1.1108 + private const int WM_DESTROY = 0x02;
1.1109 + private const int WM_CLOSE = 0x10;
1.1110 + private const int WM_COMMAND = 0x111;
1.1111 + private const int WM_INITMENUPOPUP = 0x117;
1.1112 + private const int WM_MOUSEMOVE = 0x200;
1.1113 + private const int WM_LBUTTONDOWN = 0x201;
1.1114 + private const int WM_LBUTTONUP = 0x202;
1.1115 + private const int WM_LBUTTONDBLCLK = 0x203;
1.1116 + private const int WM_RBUTTONDOWN = 0x204;
1.1117 + private const int WM_RBUTTONUP = 0x205;
1.1118 + private const int WM_RBUTTONDBLCLK = 0x206;
1.1119 + private const int WM_MBUTTONDOWN = 0x207;
1.1120 + private const int WM_MBUTTONUP = 0x208;
1.1121 + private const int WM_MBUTTONDBLCLK = 0x209;
1.1122 + private const int WM_TRAYMOUSEMESSAGE = 0x800;
1.1123 +
1.1124 + private const int NIN_BALLOONSHOW = 0x402;
1.1125 + private const int NIN_BALLOONHIDE = 0x403;
1.1126 + private const int NIN_BALLOONTIMEOUT = 0x404;
1.1127 + private const int NIN_BALLOONUSERCLICK = 0x405;
1.1128 +
1.1129 + private static int WM_TASKBARCREATED =
1.1130 + NativeMethods.RegisterWindowMessage("TaskbarCreated");
1.1131 +
1.1132 + private static class NativeMethods {
1.1133 + [DllImport("user32.dll", CharSet = CharSet.Auto)]
1.1134 + public static extern IntPtr PostMessage(HandleRef hwnd, int msg,
1.1135 + int wparam, int lparam);
1.1136 +
1.1137 + [DllImport("user32.dll", CharSet = CharSet.Auto)]
1.1138 + public static extern int RegisterWindowMessage(string msg);
1.1139 +
1.1140 + [Flags]
1.1141 + public enum NotifyIconDataFlags : int {
1.1142 + Message = 0x1,
1.1143 + Icon = 0x2,
1.1144 + Tip = 0x4,
1.1145 + State = 0x8,
1.1146 + Info = 0x10
1.1147 + }
1.1148 +
1.1149 + [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)]
1.1150 + public class NotifyIconData {
1.1151 + private int Size = Marshal.SizeOf(typeof(NotifyIconData));
1.1152 + public IntPtr Window;
1.1153 + public int ID;
1.1154 + public NotifyIconDataFlags Flags;
1.1155 + public int CallbackMessage;
1.1156 + public IntPtr Icon;
1.1157 + [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 128)]
1.1158 + public string Tip;
1.1159 + public int State;
1.1160 + public int StateMask;
1.1161 + [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
1.1162 + public string Info;
1.1163 + public int TimeoutOrVersion;
1.1164 + [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 64)]
1.1165 + public string InfoTitle;
1.1166 + public int InfoFlags;
1.1167 + }
1.1168 +
1.1169 + public enum NotifyIconMessage : int {
1.1170 + Add = 0x0,
1.1171 + Modify = 0x1,
1.1172 + Delete = 0x2
1.1173 + }
1.1174 +
1.1175 + [DllImport("shell32.dll", CharSet = CharSet.Auto)]
1.1176 + [return: MarshalAs(UnmanagedType.Bool)]
1.1177 + public static extern bool Shell_NotifyIcon(NotifyIconMessage message,
1.1178 + NotifyIconData pnid);
1.1179 +
1.1180 + [DllImport("user32.dll", CharSet = CharSet.Auto, ExactSpelling = true)]
1.1181 + public static extern bool TrackPopupMenuEx(HandleRef hmenu, int fuFlags,
1.1182 + int x, int y, HandleRef hwnd, IntPtr tpm);
1.1183 +
1.1184 + [StructLayout(LayoutKind.Sequential)]
1.1185 + public struct Point {
1.1186 + public int x;
1.1187 + public int y;
1.1188 + }
1.1189 +
1.1190 + [DllImport("user32.dll", CharSet = CharSet.Auto, ExactSpelling = true)]
1.1191 + public static extern bool GetCursorPos(ref Point point);
1.1192 +
1.1193 + [DllImport("user32.dll", CharSet = CharSet.Auto, ExactSpelling = true)]
1.1194 + public static extern bool SetForegroundWindow(HandleRef hWnd);
1.1195 + }
1.1196 }
1.1197 }
1.1198 }
2.1 --- a/OpenHardwareMonitor.csproj Mon Jul 23 21:54:35 2012 +0000
2.2 +++ b/OpenHardwareMonitor.csproj Tue Jul 24 16:04:30 2012 +0000
2.3 @@ -73,9 +73,7 @@
2.4 <Compile Include="GUI\GadgetWindow.cs" />
2.5 <Compile Include="GUI\Gadget.cs" />
2.6 <Compile Include="GUI\HardwareTypeImage.cs" />
2.7 - <Compile Include="GUI\NotifyIconAdv.cs">
2.8 - <SubType>Component</SubType>
2.9 - </Compile>
2.10 + <Compile Include="GUI\NotifyIconAdv.cs" />
2.11 <Compile Include="GUI\PlotPanel.cs">
2.12 <SubType>UserControl</SubType>
2.13 </Compile>