# HG changeset patch # User StephaneLenclud # Date 1359910910 -3600 # Node ID 090259cfd699035b29948533cdbffdb5ea18cdcc # Parent 7b859a06eecb85dce1b23370f433dfff85e72c6b Adding SoundGraphDisplay and SensorFrontView classes. They were respectively based on SystemTray and SensorNotifyIcon. SoundGraphDisplay is now able to load iMONDisplay.dll providing it lives on your PATH. Adding option to sensor context menu for adding it into FrontView. diff -r 7b859a06eecb -r 090259cfd699 GUI/MainForm.cs --- a/GUI/MainForm.cs Tue Dec 30 22:49:32 2014 +0000 +++ b/GUI/MainForm.cs Sun Feb 03 18:01:50 2013 +0100 @@ -35,7 +35,8 @@ private IDictionary sensorPlotColors = new Dictionary(); private Color[] plotColorPalette; - private SystemTray systemTray; + private SystemTray systemTray; + private SoundGraphDisplay soundGraphDisplay; private StartupManager startupManager = new StartupManager(); private UpdateVisitor updateVisitor = new UpdateVisitor(); private SensorGadget gadget; @@ -129,6 +130,8 @@ systemTray = new SystemTray(computer, settings, unitManager); systemTray.HideShowCommand += hideShowClick; systemTray.ExitCommand += exitClick; + + soundGraphDisplay = new SoundGraphDisplay(computer, settings, unitManager); int p = (int)Environment.OSVersion.Platform; if ((p == 4) || (p == 128)) { // Unix @@ -603,6 +606,7 @@ if (runWebServer.Value) server.Quit(); systemTray.Dispose(); + soundGraphDisplay.Dispose(); } private void aboutMenuItem_Click(object sender, EventArgs e) { @@ -662,6 +666,18 @@ }; treeContextMenu.MenuItems.Add(item); } + { + MenuItem item = new MenuItem("Show in iMON FrontView"); + item.Checked = systemTray.Contains(node.Sensor); + item.Click += delegate(object obj, EventArgs args) + { + if (item.Checked) + soundGraphDisplay.Remove(node.Sensor); + else + soundGraphDisplay.Add(node.Sensor, true); + }; + treeContextMenu.MenuItems.Add(item); + } if (gadget != null) { MenuItem item = new MenuItem("Show in Gadget"); item.Checked = gadget.Contains(node.Sensor); diff -r 7b859a06eecb -r 090259cfd699 GUI/SensorFrontView.cs --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/GUI/SensorFrontView.cs Sun Feb 03 18:01:50 2013 +0100 @@ -0,0 +1,283 @@ +/* + + This Source Code Form is subject to the terms of the Mozilla Public + License, v. 2.0. If a copy of the MPL was not distributed with this + file, You can obtain one at http://mozilla.org/MPL/2.0/. + + Copyright (C) 2009-2012 Michael Möller + +*/ + +using System; +using System.Drawing; +using System.Drawing.Drawing2D; +using System.Drawing.Imaging; +using System.Drawing.Text; +using System.Runtime.InteropServices; +using System.Windows.Forms; +using OpenHardwareMonitor.Hardware; +using OpenHardwareMonitor.Utilities; + +namespace OpenHardwareMonitor.GUI +{ + public class SensorFrontView : IDisposable + { + + private UnitManager unitManager; + + private ISensor sensor; + private Bitmap bitmap; + private Graphics graphics; + private Color color; + private Color darkColor; + private Brush brush; + private Brush darkBrush; + private Pen pen; + private Font font; + private Font smallFont; + + public SensorFrontView(SoundGraphDisplay soundGraphDisplay, ISensor sensor, + bool balloonTip, PersistentSettings settings, UnitManager unitManager) + { + this.unitManager = unitManager; + this.sensor = sensor; + + // get the default dpi to create an icon with the correct size + float dpiX, dpiY; + using (Bitmap b = new Bitmap(1, 1, PixelFormat.Format32bppArgb)) + { + dpiX = b.HorizontalResolution; + dpiY = b.VerticalResolution; + } + + // adjust the size of the icon to current dpi (default is 16x16 at 96 dpi) + int width = (int)Math.Round(16 * dpiX / 96); + int height = (int)Math.Round(16 * dpiY / 96); + + // make sure it does never get smaller than 16x16 + width = width < 16 ? 16 : width; + height = height < 16 ? 16 : height; + + // adjust the font size to the icon size + FontFamily family = SystemFonts.MessageBoxFont.FontFamily; + float baseSize; + switch (family.Name) + { + case "Segoe UI": baseSize = 12; break; + case "Tahoma": baseSize = 11; break; + default: baseSize = 12; break; + } + + this.font = new Font(family, + baseSize * width / 16.0f, GraphicsUnit.Pixel); + this.smallFont = new Font(family, + 0.75f * baseSize * width / 16.0f, GraphicsUnit.Pixel); + + this.bitmap = new Bitmap(width, height, PixelFormat.Format32bppArgb); + this.graphics = Graphics.FromImage(this.bitmap); + + if (Environment.OSVersion.Version.Major > 5) + { + this.graphics.TextRenderingHint = TextRenderingHint.ClearTypeGridFit; + this.graphics.SmoothingMode = SmoothingMode.HighQuality; + } + } + + public ISensor Sensor + { + get { return sensor; } + } + + public Color Color + { + get { return color; } + set + { + this.color = value; + this.darkColor = Color.FromArgb(255, + this.color.R / 3, + this.color.G / 3, + this.color.B / 3); + Brush brush = this.brush; + this.brush = new SolidBrush(this.color); + if (brush != null) + brush.Dispose(); + Brush darkBrush = this.darkBrush; + this.darkBrush = new SolidBrush(this.darkColor); + if (darkBrush != null) + darkBrush.Dispose(); + } + } + + public void Dispose() + { + + if (brush != null) + brush.Dispose(); + if (darkBrush != null) + darkBrush.Dispose(); + pen.Dispose(); + graphics.Dispose(); + bitmap.Dispose(); + font.Dispose(); + smallFont.Dispose(); + } + + private string GetString() + { + if (!sensor.Value.HasValue) + return "-"; + + switch (sensor.SensorType) + { + case SensorType.Voltage: + return string.Format("{0:F1}", sensor.Value); + case SensorType.Clock: + return string.Format("{0:F1}", 1e-3f * sensor.Value); + case SensorType.Load: + return string.Format("{0:F0}", sensor.Value); + case SensorType.Temperature: + if (unitManager.TemperatureUnit == TemperatureUnit.Fahrenheit) + return string.Format("{0:F0}", + UnitManager.CelsiusToFahrenheit(sensor.Value)); + else + return string.Format("{0:F0}", sensor.Value); + case SensorType.Fan: + return string.Format("{0:F1}", 1e-3f * sensor.Value); + case SensorType.Flow: + return string.Format("{0:F1}", 1e-3f * sensor.Value); + case SensorType.Control: + return string.Format("{0:F0}", sensor.Value); + case SensorType.Level: + return string.Format("{0:F0}", sensor.Value); + case SensorType.Power: + return string.Format("{0:F0}", sensor.Value); + case SensorType.Data: + return string.Format("{0:F0}", sensor.Value); + case SensorType.Factor: + return string.Format("{0:F1}", sensor.Value); + } + return "-"; + } + + private Icon CreateTransparentIcon() + { + string text = GetString(); + int count = 0; + for (int i = 0; i < text.Length; i++) + if ((text[i] >= '0' && text[i] <= '9') || text[i] == '-') + count++; + bool small = count > 2; + + graphics.Clear(Color.Black); + TextRenderer.DrawText(graphics, text, small ? smallFont : font, + new Point(-2, small ? 1 : 0), Color.White, Color.Black); + + BitmapData data = bitmap.LockBits( + new Rectangle(0, 0, bitmap.Width, bitmap.Height), + ImageLockMode.ReadOnly, PixelFormat.Format32bppArgb); + + IntPtr Scan0 = data.Scan0; + + int numBytes = bitmap.Width * bitmap.Height * 4; + byte[] bytes = new byte[numBytes]; + Marshal.Copy(Scan0, bytes, 0, numBytes); + bitmap.UnlockBits(data); + + byte red, green, blue; + for (int i = 0; i < bytes.Length; i += 4) + { + blue = bytes[i]; + green = bytes[i + 1]; + red = bytes[i + 2]; + + bytes[i] = color.B; + bytes[i + 1] = color.G; + bytes[i + 2] = color.R; + bytes[i + 3] = (byte)(0.3 * red + 0.59 * green + 0.11 * blue); + } + + return IconFactory.Create(bytes, bitmap.Width, bitmap.Height, + PixelFormat.Format32bppArgb); + } + + private Icon CreatePercentageIcon() + { + try + { + graphics.Clear(Color.Transparent); + } + catch (ArgumentException) + { + graphics.Clear(Color.Black); + } + graphics.FillRectangle(darkBrush, 0.5f, -0.5f, bitmap.Width - 2, bitmap.Height); + float value = sensor.Value.GetValueOrDefault(); + float y = 0.16f * (100 - value); + graphics.FillRectangle(brush, 0.5f, -0.5f + y, bitmap.Width - 2, bitmap.Height - y); + graphics.DrawRectangle(pen, 1, 0, bitmap.Width - 3, bitmap.Height - 1); + + BitmapData data = bitmap.LockBits( + new Rectangle(0, 0, bitmap.Width, bitmap.Height), + ImageLockMode.ReadOnly, PixelFormat.Format32bppArgb); + byte[] bytes = new byte[bitmap.Width * bitmap.Height * 4]; + Marshal.Copy(data.Scan0, bytes, 0, bytes.Length); + bitmap.UnlockBits(data); + + return IconFactory.Create(bytes, bitmap.Width, bitmap.Height, + PixelFormat.Format32bppArgb); + } + + public void Update() + { + + + switch (sensor.SensorType) + { + case SensorType.Load: + case SensorType.Control: + case SensorType.Level: + //notifyIcon.Icon = CreatePercentageIcon(); + break; + default: + //notifyIcon.Icon = CreateTransparentIcon(); + break; + } + + + string format = ""; + switch (sensor.SensorType) + { + case SensorType.Voltage: format = "\n{0}: {1:F2} V"; break; + case SensorType.Clock: format = "\n{0}: {1:F0} MHz"; break; + case SensorType.Load: format = "\n{0}: {1:F1} %"; break; + case SensorType.Temperature: format = "\n{0}: {1:F1} °C"; break; + case SensorType.Fan: format = "\n{0}: {1:F0} RPM"; break; + case SensorType.Flow: format = "\n{0}: {1:F0} L/h"; break; + case SensorType.Control: format = "\n{0}: {1:F1} %"; break; + case SensorType.Level: format = "\n{0}: {1:F1} %"; break; + case SensorType.Power: format = "\n{0}: {1:F0} W"; break; + case SensorType.Data: format = "\n{0}: {1:F0} GB"; break; + case SensorType.Factor: format = "\n{0}: {1:F3} GB"; break; + } + string formattedValue = string.Format(format, sensor.Name, sensor.Value); + + if (sensor.SensorType == SensorType.Temperature && + unitManager.TemperatureUnit == TemperatureUnit.Fahrenheit) + { + format = "\n{0}: {1:F1} °F"; + formattedValue = string.Format(format, sensor.Name, + UnitManager.CelsiusToFahrenheit(sensor.Value)); + } + + string hardwareName = sensor.Hardware.Name; + hardwareName = hardwareName.Substring(0, + Math.Min(63 - formattedValue.Length, hardwareName.Length)); + string text = hardwareName + formattedValue; + if (text.Length > 63) + text = null; + + + } + } +} diff -r 7b859a06eecb -r 090259cfd699 GUI/SoundGraphDisplay.cs --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/GUI/SoundGraphDisplay.cs Sun Feb 03 18:01:50 2013 +0100 @@ -0,0 +1,221 @@ +/* + + This Source Code Form is subject to the terms of the Mozilla Public + License, v. 2.0. If a copy of the MPL was not distributed with this + file, You can obtain one at http://mozilla.org/MPL/2.0/. + + Copyright (C) 2009-2012 Michael Möller + +*/ + +using System; +using System.Collections.Generic; +using System.Drawing; +using System.Text; +using System.Windows.Forms; +using OpenHardwareMonitor.Hardware; +using OpenHardwareMonitor.Utilities; +using System.Runtime.InteropServices; + + + +static class NativeMethods +{ + [System.Flags] + public enum LoadLibraryFlags : uint + { + DONT_RESOLVE_DLL_REFERENCES = 0x00000001, + LOAD_IGNORE_CODE_AUTHZ_LEVEL = 0x00000010, + LOAD_LIBRARY_AS_DATAFILE = 0x00000002, + LOAD_LIBRARY_AS_DATAFILE_EXCLUSIVE = 0x00000040, + LOAD_LIBRARY_AS_IMAGE_RESOURCE = 0x00000020, + LOAD_WITH_ALTERED_SEARCH_PATH = 0x00000008 + } + + + [DllImport("kernel32.dll", SetLastError = true)] + public static extern IntPtr LoadLibrary(string dllToLoad); + + [DllImport("kernel32.dll")] + public static extern IntPtr LoadLibraryEx(string lpFileName, IntPtr hReservedNull, LoadLibraryFlags dwFlags); + + [DllImport("kernel32.dll", SetLastError = true)] + public static extern IntPtr GetProcAddress(IntPtr hModule, string procedureName); + + [DllImport("kernel32.dll", SetLastError = true)] + public static extern bool FreeLibrary(IntPtr hModule); +} + + +namespace OpenHardwareMonitor.GUI +{ + public class SoundGraphDisplay : IDisposable + { + private IComputer computer; + private PersistentSettings settings; + private UnitManager unitManager; + private List list = new List(); + //private bool mainIconEnabled = false; + //private NotifyIconAdv mainIcon; + IntPtr iSoundGraphDll; + + public SoundGraphDisplay(IComputer computer, PersistentSettings settings, + UnitManager unitManager) + { + this.computer = computer; + this.settings = settings; + this.unitManager = unitManager; + computer.HardwareAdded += new HardwareEventHandler(HardwareAdded); + computer.HardwareRemoved += new HardwareEventHandler(HardwareRemoved); + + //Try loading SoundGraph iMON Disaply DLL + iSoundGraphDll = NativeMethods.LoadLibraryEx(@"iMONDisplay.dll", IntPtr.Zero, NativeMethods.LoadLibraryFlags.LOAD_WITH_ALTERED_SEARCH_PATH); + int err=Marshal.GetLastWin32Error(); //If you 193 it means you need build for x86 + if (err == 193) + { + Console.Write(@"iMON: Cannot load x86 DLL from x64 process."); + } + else if (err != 0) + { + Console.Write(@"iMON: Error: %i - Failed to load iMONDisplay.dll", err); + } + else + { + Console.Write(@"iMON: DLL loaded."); + } + + + } + + private void HardwareRemoved(IHardware hardware) + { + hardware.SensorAdded -= new SensorEventHandler(SensorAdded); + hardware.SensorRemoved -= new SensorEventHandler(SensorRemoved); + foreach (ISensor sensor in hardware.Sensors) + SensorRemoved(sensor); + foreach (IHardware subHardware in hardware.SubHardware) + HardwareRemoved(subHardware); + } + + private void HardwareAdded(IHardware hardware) + { + foreach (ISensor sensor in hardware.Sensors) + SensorAdded(sensor); + hardware.SensorAdded += new SensorEventHandler(SensorAdded); + hardware.SensorRemoved += new SensorEventHandler(SensorRemoved); + foreach (IHardware subHardware in hardware.SubHardware) + HardwareAdded(subHardware); + } + + private void SensorAdded(ISensor sensor) + { + if (settings.GetValue(new Identifier(sensor.Identifier, + "tray").ToString(), false)) + Add(sensor, false); + } + + private void SensorRemoved(ISensor sensor) + { + if (Contains(sensor)) + Remove(sensor, false); + } + + public void Dispose() + { + foreach (SensorFrontView icon in list) + icon.Dispose(); + + //Unload our DLL + if (iSoundGraphDll != IntPtr.Zero) + { + bool result = NativeMethods.FreeLibrary(iSoundGraphDll); + } + + } + + public void Redraw() + { + foreach (SensorFrontView icon in list) + icon.Update(); + } + + public bool Contains(ISensor sensor) + { + foreach (SensorFrontView icon in list) + if (icon.Sensor == sensor) + return true; + return false; + } + + public void Add(ISensor sensor, bool balloonTip) + { + if (Contains(sensor)) + { + return; + } + else + { + //SL: + //list.Add(new SensorFrontView(this, sensor, balloonTip, settings, unitManager)); + //UpdateMainIconVisibilty(); + //settings.SetValue(new Identifier(sensor.Identifier, "tray").ToString(), true); + } + } + + public void Remove(ISensor sensor) + { + Remove(sensor, true); + } + + private void Remove(ISensor sensor, bool deleteConfig) + { + if (deleteConfig) + { + settings.Remove( + new Identifier(sensor.Identifier, "tray").ToString()); + settings.Remove( + new Identifier(sensor.Identifier, "traycolor").ToString()); + } + SensorFrontView instance = null; + foreach (SensorFrontView icon in list) + if (icon.Sensor == sensor) + instance = icon; + if (instance != null) + { + list.Remove(instance); + UpdateMainIconVisibilty(); + instance.Dispose(); + } + } + + + + private void UpdateMainIconVisibilty() + { + /* + if (mainIconEnabled) + { + mainIcon.Visible = list.Count == 0; + } + else + { + mainIcon.Visible = false; + } + */ + } + + /* + public bool IsMainIconEnabled + { + get { return mainIconEnabled; } + set + { + if (mainIconEnabled != value) + { + mainIconEnabled = value; + UpdateMainIconVisibilty(); + } + } + }*/ + } +} diff -r 7b859a06eecb -r 090259cfd699 OpenHardwareMonitor.csproj --- a/OpenHardwareMonitor.csproj Tue Dec 30 22:49:32 2014 +0000 +++ b/OpenHardwareMonitor.csproj Sun Feb 03 18:01:50 2013 +0100 @@ -108,9 +108,11 @@ ParameterForm.cs + + Component