分享一個很久之前寫的一個Winform換膚組件。 主要利用CBT鉤子,NativeWindow來實現。可實現動態換皮膚插件修改窗體顯示外觀。 我們先定義一個自定義組件 using Skin; using System; using System.Collections.Generic; using ...
分享一個很久之前寫的一個Winform換膚組件。
主要利用CBT鉤子,NativeWindow來實現。可實現動態換皮膚插件修改窗體顯示外觀。
我們先定義一個自定義組件
using Skin; using System; using System.Collections.Generic; using System.ComponentModel; using System.Drawing; using System.Linq; using System.Reflection; using System.Runtime.InteropServices; using System.Text; using System.Windows.Forms; namespace SkinControl { public class LySkinEngine : Component { #region 欄位 public delegate IntPtr HookProc(int nCode, IntPtr wParam, IntPtr lParam); private static HookProc _cbtHook; private static IntPtr Hook; private static string m_SkinName = ""; #endregion #region API /// <summary> /// SetWindowsHookEx /// </summary> /// <param name="idHook"></param> /// <param name="lpfn"></param> /// <param name="hMod"></param> /// <param name="dwThreadId"></param> /// <returns></returns> [DllImport("User32.dll", CharSet = CharSet.Auto)] public static extern IntPtr SetWindowsHookEx(int idHook, HookProc lpfn, int hMod, int dwThreadId); /// <summary> /// CallNextHookEx /// </summary> /// <param name="hhk"></param> /// <param name="nCode"></param> /// <param name="wParam"></param> /// <param name="lParam"></param> /// <returns></returns> [DllImport("User32.dll", CharSet = CharSet.Auto)] public static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, IntPtr lParam); /// <summary> /// UnhookWindowsHookEx /// </summary> /// <param name="hhk"></param> /// <returns></returns> [DllImport("User32.dll", CharSet = CharSet.Auto)] public static extern bool UnhookWindowsHookEx(IntPtr hhk); #endregion #region FnHookProc /// <summary> /// FnHookProc /// </summary> /// <param name="nCode"></param> /// <param name="wParam"></param> /// <param name="lParam"></param> /// <returns></returns> private static IntPtr FnHookProc(int nCode, IntPtr wParam, IntPtr lParam) { switch (nCode) { case 5: Control control = Control.FromHandle(wParam); if (control != null) { Skin.SkinResource tmpSkinClass = GetSkin(); FormBase frmBase = new FormBase(control as Form, tmpSkinClass); } break; default: break; } return CallNextHookEx(Hook, nCode, wParam, lParam); } #endregion #region 動態載入皮膚資源 private static SkinResource GetSkin() { SkinResource tmpResource = new SkinResource(); Assembly ass = Assembly.LoadFrom(AppDomain.CurrentDomain.BaseDirectory + @"\Skin\" + m_SkinName + @".dll"); Type type = ass.GetType("SkinFile.Skin"); //生成實例 Skin1 Object obj = Activator.CreateInstance(type); //標題背景 PropertyInfo pi = type.GetProperty("CaptionBackgroundColor"); tmpResource.CaptionBackgroundColor = (Color)pi.GetValue(obj, null); //標題前景色 PropertyInfo pi1 = type.GetProperty("CaptionColor"); tmpResource.CaptionColor = (Color)pi1.GetValue(obj, null); return tmpResource; } #endregion public void SetSkin(string varSkinName) { m_SkinName = varSkinName; if (Hook == IntPtr.Zero) { _cbtHook = new HookProc(FnHookProc); Hook = SetWindowsHookEx(5, _cbtHook, 0, AppDomain.GetCurrentThreadId()); Application.ApplicationExit += new EventHandler(Application_ApplicationExit); } } /// <summary> /// Application_ApplicationExit /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void Application_ApplicationExit(object sender, EventArgs e) { //Engine.Dispose(false); UnhookWindowsHookEx(Hook); } } }View Code
新增一個皮膚資源類,主要用於存儲皮膚文件中的信息
namespace Skin { public class SkinResource { public Color CaptionColor {get;set;} public Color CaptionBackgroundColor {get;set;} } }
新增一個類,主要實現對窗體的消息接管和繪製
namespace Skin { public partial class FormBase : NativeWindow { Form m_f = null; public FormBase(Form varForm, Skin.SkinResource varSkin) { try { m_f = varForm; AssignHandle(m_f.Handle); m_f.HandleDestroyed += new EventHandler(this.OnHandleDestroyed); CloseButtonImage = Resource1.close_normal; CloseButtonHoverImage = Resource1.close_highlight; CloseButtonPressDownImage = Resource1.close_press; MaximumButtonImage = Resource1.max_normal; MaximumButtonHoverImage = Resource1.max_highlight; MaximumButtonPressDownImage = Resource1.max_press; MaximumNormalButtonImage = Resource1.restore_normal; MaximumNormalButtonHoverImage = Resource1.restore_highlight; MaximumNormalButtonPressDownImage = Resource1.restore_press; MinimumButtonImage = Resource1.min_normal; MinimumButtonHoverImage = Resource1.min_highlight; MinimumButtonPressDownImage = Resource1.min_press; HelpButtonImage = Resource1.skin_normal; HelpButtonHoverImage = Resource1.skin_highlight; HelpButtonPressDownImage = Resource1.skin_press; CaptionColor = varSkin.CaptionColor; CaptionBackgroundColor = varSkin.CaptionBackgroundColor; } catch(Exception ex) { } } #region 欄位 struct _NonClientSizeInfo { public Size CaptionButtonSize; public Size BorderSize; public int CaptionHeight; public Rectangle CaptionRect; public Rectangle Rect; public Rectangle ClientRect; public int Width; public int Height; }; #region 常量 const int WM_NCACTIVATE = 0x86; const int WM_NCPAINT = 0x85; const int WM_NCLBUTTONDOWN = 0xA1; const int WM_NCRBUTTONDOWN = 0x00A4; const int WM_NCRBUTTONUP = 0x00A5; const int WM_NCMOUSEMOVE = 0x00A0; const int WM_NCLBUTTONUP = 0x00A2; const int WM_NCCALCSIZE = 0x0083; const int WM_NCMOUSEHOVER = 0x02A0; const int WM_NCMOUSELEAVE = 0x02A2; const int WM_NCHITTEST = 0x0084; const int WM_NCCREATE = 0x0081; //const int WM_RBUTTONUP = 0x0205; const int WM_LBUTTONDOWN = 0x0201; const int WM_CAPTURECHANGED = 0x0215; const int WM_LBUTTONUP = 0x0202; const int WM_SETCURSOR = 0x0020; const int WM_CLOSE = 0x0010; const int WM_SYSCOMMAND = 0x0112; const int WM_MOUSEMOVE = 0x0200; const int WM_SIZE = 0x0005; const int WM_SIZING = 0x0214; const int WM_GETMINMAXINFO = 0x0024; const int WM_ENTERSIZEMOVE = 0x0231; const int WM_WINDOWPOSCHANGING = 0x0046; // FOR WM_SIZING MSG WPARAM const int WMSZ_BOTTOM = 6; const int WMSZ_BOTTOMLEFT = 7; const int WMSZ_BOTTOMRIGHT = 8; const int WMSZ_LEFT = 1; const int WMSZ_RIGHT = 2; const int WMSZ_TOP = 3; const int WMSZ_TOPLEFT = 4; const int WMSZ_TOPRIGHT = 5; // left mouse button is down. const int MK_LBUTTON = 0x0001; const int SC_CLOSE = 0xF060; const int SC_MAXIMIZE = 0xF030; const int SC_MINIMIZE = 0xF020; const int SC_RESTORE = 0xF120; const int SC_CONTEXTHELP = 0xF180; const int HTCAPTION = 2; const int HTCLOSE = 20; const int HTHELP = 21; const int HTMAXBUTTON = 9; const int HTMINBUTTON = 8; const int HTTOP = 12; const int SM_CYBORDER = 6; const int SM_CXBORDER = 5; const int SM_CYCAPTION = 4; const int CS_DropSHADOW = 0x20000; const int GCL_STYLE = (-26); #endregion #endregion #region windows api [DllImport("User32.dll")] private static extern IntPtr GetWindowDC(IntPtr hwnd); [DllImport("User32.dll")] [return: MarshalAs(UnmanagedType.Bool)] private static extern bool GetWindowRect(IntPtr hwnd, ref Rectangle rect); [DllImport("User32.dll")] private static extern int ReleaseDC(IntPtr hwnd, IntPtr hdc); [DllImport("user32.dll", CharSet = CharSet.Auto)] public static extern int SetClassLong(IntPtr hwnd, int nIndex, int dwNewLong); [DllImport("user32.dll", CharSet = CharSet.Auto)] public static extern int GetClassLong(IntPtr hwnd, int nIndex); #endregion #region 構造函數 internal void OnHandleCreated(object sender, EventArgs e) { AssignHandle(((Form)sender).Handle); } internal void OnHandleDestroyed(object sender, EventArgs e) { ReleaseHandle(); } #endregion #region 屬性 [Category("ControlBox")] [Description("Close button image in control box.")] [DisplayName("CloseButtonImage")] [DesignOnly(true)] public Image CloseButtonImage { get; set; } [Category("ControlBox")] [Description("Close button image pressed down in control box.")] [DisplayName("CloseButtonPressDownImage")] [DesignOnly(true)] public Image CloseButtonPressDownImage { get; set; } [Category("ControlBox")] [Description("Close button image hover in control box.")] [DisplayName("CloseButtonHoverImage")] [DesignOnly(true)] public Image CloseButtonHoverImage { get; set; } [Category("ControlBox")] [Description("Maximum button image in control box.")] [DisplayName("MaximumButtonImage")] [DesignOnly(true)] public Image MaximumButtonImage { get; set; } [Category("ControlBox")] [Description("Maximum button hover image in control box.")] [DisplayName("MaximumButtonHoverImage")] [DesignOnly(true)] public Image MaximumButtonHoverImage { get; set; } [Category("ControlBox")] [Description("Maximum button pressed down image in control box.")] [DisplayName("MaximumButtonPressDownImage")] [DesignOnly(true)] public Image MaximumButtonPressDownImage { get; set; } [Category("ControlBox")] [Description("Maximum Normal button image in control box.")] [DisplayName("MaximumNormalButtonImage")] [DesignOnly(true)] public Image MaximumNormalButtonImage { get; set; } [Category("ControlBox")] [Description("Maximum Normal button hover image in control box.")] [DisplayName("MaximumNormalButtonHoverImage")] [DesignOnly(true)] public Image MaximumNormalButtonHoverImage { get; set; } [Category("ControlBox")] [Description("Maximum Normal button pressed down image in control box.")] [DisplayName("MaximumNormalButtonPressDownImage")] [DesignOnly(true)] public Image MaximumNormalButtonPressDownImage { get; set; } [Category("ControlBox")] [Description("Minimum button image in control box.")] [DisplayName("MinimumButtonImage")] [DesignOnly(true)] public Image MinimumButtonImage { get; set; } [Category("ControlBox")] [Description("Minimum button hover image in control box.")] [DisplayName("MinimumButtonHoverImage")] [DesignOnly(true)] public Image MinimumButtonHoverImage { get; set; } [Category("ControlBox")] [Description("Minimum button pressed down image in control box.")] [DisplayName("MinimumButtonPressDownImage")] [DesignOnly(true)] public Image MinimumButtonPressDownImage { get; set; } [Category("ControlBox")] [Description("Help button image in control box.")] [DisplayName("HelpButtonImage")] [DesignOnly(true)] public Image HelpButtonImage { get; set; } [Category("ControlBox")] [Description("Help button hover image in control box.")] [DisplayName("HelpButtonHoverImage")] [DesignOnly(true)] public Image HelpButtonHoverImage { get; set; } [Category("ControlBox")] [Description("Help button pressed down image in control box.")] [DisplayName("HelpButtonPressDownImage")] [DesignOnly(true)] public Image HelpButtonPressDownImage { get; set; } [Category("CaptionColor")] [Description("The color of caption.")] [DisplayName("CaptionColor")] [DefaultValue(typeof(Color), "Black")] [Browsable(true)] public Color CaptionColor { get; set; } [Category("CaptionColor")] [Description("The color of caption.")] [DisplayName("CaptionBackgroundColor")] [DefaultValue(typeof(Color), "Black")] [Browsable(true)] public Color CaptionBackgroundColor { get; set; } [DefaultValue("")] [Browsable(true)] [Category("ControlBox")] public virtual ContextMenuStrip CaptionContextMenu { get; set; } #endregion #region 方法 private _NonClientSizeInfo GetNonClientInfo(IntPtr hwnd) { _NonClientSizeInfo info = new _NonClientSizeInfo(); info.CaptionButtonSize = SystemInformation.CaptionButtonSize; info.CaptionHeight = SystemInformation.CaptionHeight; switch (m_f.FormBorderStyle) { case System.Windows.Forms.FormBorderStyle.Fixed3D: info.BorderSize = SystemInformation.FixedFrameBorderSize; break; case System.Windows.Forms.FormBorderStyle.FixedDialog: info.BorderSize = SystemInformation.FixedFrameBorderSize; break; case System.Windows.Forms.FormBorderStyle.FixedSingle: info.BorderSize = SystemInformation.FixedFrameBorderSize; break; case System.Windows.Forms.FormBorderStyle.FixedToolWindow: info.BorderSize = SystemInformation.FixedFrameBorderSize; info.CaptionButtonSize = SystemInformation.ToolWindowCaptionButtonSize; info.CaptionHeight = SystemInformation.ToolWindowCaptionHeight; break; case System.Windows.Forms.FormBorderStyle.Sizable: info.BorderSize = SystemInformation.FrameBorderSize; break; case System.Windows.Forms.FormBorderStyle.SizableToolWindow: info.CaptionButtonSize = SystemInformation.ToolWindowCaptionButtonSize; info.BorderSize = SystemInformation.FrameBorderSize; info.CaptionHeight = SystemInformation.ToolWindowCaptionHeight; break; default: info.BorderSize = SystemInformation.BorderSize; break; } Rectangle areatRect = new Rectangle(); GetWindowRect(hwnd, ref areatRect); int width = areatRect.Right - areatRect.Left; int height = areatRect.Bottom - areatRect.Top; info.Width = width; info.Height = height; Point xy = new Point(areatRect.Left, areatRect.Top); xy.Offset(-areatRect.Left, -areatRect.Top); info.CaptionRect = new Rectangle(xy.X, xy.Y + info.BorderSize.Height, width, info.CaptionHeight); info.Rect = new Rectangle(xy.X, xy.Y, width, height); info.ClientRect = new Rectangle(xy.X + info.BorderSize.Width, xy.Y + info.CaptionHeight + info.BorderSize.Height, width - info.BorderSize.Width * 2, height - info.CaptionHeight - info.BorderSize.Height * 2); return info; } private void DrawTitle(Graphics g, _NonClientSizeInfo ncInfo, bool active) { try { int titleX; if (m_f.ShowIcon && m_f.FormBorderStyle != System.Windows.Forms.FormBorderStyle.FixedToolWindow && m_f.FormBorderStyle != System.Windows.Forms.FormBorderStyle.SizableToolWindow) { Size iconSize = SystemInformation.SmallIconSize; g.DrawIcon(m_f.Icon, new Rectangle(new Point(ncInfo.BorderSize.Width+5, ncInfo.BorderSize.Height + (ncInfo.CaptionHeight - iconSize.Height) / 2+5), iconSize)); titleX = ncInfo.BorderSize.Width + iconSize.Width + ncInfo.BorderSize.Width+5; } else { titleX = ncInfo.BorderSize.Width; } SizeF captionTitleSize = g.MeasureString(m_f.Text, SystemFonts.CaptionFont); g.DrawString(m_f.Text, SystemFonts.CaptionFont, new SolidBrush(CaptionColor), new RectangleF(titleX, (ncInfo.BorderSize.Height + ncInfo.CaptionHeight - captionTitleSize.Height) / 2+5, ncInfo.CaptionRect.Width - ncInfo.BorderSize.Width * 2 - SystemInformation.MinimumWindowSize.Width, ncInfo.CaptionRect.Height), StringFormat.GenericTypographic); } catch(Exception ex) { } } private void DrawBorder(Graphics g, _NonClientSizeInfo ncInfo, Brush background, bool active,ref List<Rectangle> varBorders) { int tmpHeight = m_f.Height; int tmpWidth = m_f.Width; if (ncInfo.Rect.Height != tmpHeight) { ncInfo.Rect.Height = tmpHeight + ncInfo.BorderSize.Height; } if (ncInfo.Rect.Width != tmpWidth) { ncInfo.Rect.Width = tmpWidth + ncInfo.BorderSize.Width; } Rectangle borderTop = new Rectangle(ncInfo.Rect.Left, ncInfo.Rect.Top, ncInfo.Rect.Left + ncInfo.Rect.Width, ncInfo.Rect.Top + ncInfo.BorderSize.Height); Rectangle borderLeft = new Rectangle( new Point(ncInfo.Rect.Location.X, ncInfo.Rect.Location.Y + ncInfo.BorderSize.Height), new Size((int)(ncInfo.BorderSize.Width*2), ncInfo.ClientRect.Height + ncInfo.CaptionHeight + ncInfo.BorderSize.Height)); Rectangle borderRight = new Rectangle(ncInfo.Rect.Width- 3*ncInfo.BorderSize.Width, ncInfo.Rect.Top + ncInfo.BorderSize.Height, (int)(ncInfo.BorderSize.Width * 2), ncInfo.ClientRect.Height + ncInfo.CaptionHeight + ncInfo.BorderSize.Height); Rectangle borderBottom = new Rectangle(ncInfo.Rect.Left + ncInfo.BorderSize.Width, ncInfo.Rect.Height - 3*ncInfo.BorderSize.Height, ncInfo.Rect.Width - ncInfo.BorderSize.Width , (int)(ncInfo.BorderSize.Height * 2)); varBorders.Add(borderTop); varBorders.Add(borderLeft); varBorders.Add(borderRight); varBorders.Add(borderBottom); g.FillRectangle(background, borderTop); // left border g.FillRectangle(background, borderLeft); // right border g.FillRectangle(background, borderRight); // bottom border g.FillRectangle(background, borderBottom); } private List<Rectangle> m_Borders = null; private void DrawCaption(IntPtr hwnd, bool active) { m_Borders = new List<Rectangle>(); IntPtr dc; Graphics g; Size iconSize; _NonClientSizeInfo ncInfo; Brush backgroundColor = new SolidBrush(CaptionBackgroundColor); Brush foregroundColor = new SolidBrush(CaptionColor); iconSize = SystemInformation.SmallIconSize; dc = GetWindowDC(hwnd); ncInfo = GetNonClientInfo(hwnd); g = Graphics.FromHdc(dc); Rectangle rc = ncInfo.CaptionRect; rc.Height = (int)(rc.Height + ncInfo.BorderSize.Height); g.FillRectangle(backgroundColor, rc); DrawBorder(g, ncInfo, backgroundColor, active, ref m_Borders); DrawTitle(g, ncInfo, active); DrawControlBox(g, ncInfo, backgroundColor, m_f.ControlBox, m_f.MaximizeBox, m_f.MinimizeBox, m_f.HelpButton); g.Dispose(); ReleaseDC(hwnd, dc); } private void DrawControlBox(Graphics g, _NonClientSizeInfo info, Brush background, bool closeBtn, bool maxBtn, bool minBtn, bool helpBtn) { int tmpHeight = m_f.Height; int tmpWidth = m_f.Width; if (info.CaptionRect.Height > tmpHeight) { info.CaptionRect.Height = tmpHeight; } if (info.CaptionRect.Width > tmpWidth) { info.CaptionRect.Width = tmpWidth; } info.CaptionRect.Height = info.CaptionRect.Height * 2; if (m_f.ControlBox) { int closeBtnPosX = info.CaptionRect.Width - info.BorderSize.Width - info.CaptionButtonSize.Width; int maxBtnPosX = closeBtnPosX - info.CaptionButtonSize.Width; int minBtnPosX = maxBtnPosX - info.CaptionButtonSize.Width; int btnPosY = info.BorderSize.Height + (info.CaptionHeight - info.CaptionButtonSize.Height) / 2; Rectangle btnRect = new Rectangle(new Point(closeBtnPosX, btnPosY), info.CaptionButtonSize); Rectangle maxRect = new Rectangle(new Point(maxBtnPosX, btnPosY), info.CaptionButtonSize); Rectangle minRect = new Rectangle(new Point(minBtnPosX, btnPosY), info.CaptionButtonSize); Brush backgroundColor = new SolidBrush(CaptionBackgroundColor); g.FillRectangle(backgroundColor, btnRect); g.FillRectangle(backgroundColor, maxRect); g.FillRectangle(backgroundColor, minRect); g.DrawImage(CloseButtonImage, btnRect); if (m_f.MaximizeBox || m_f.MinimizeBox) { if (m_f.FormBorderStyle != System.Windows.Forms.FormBorderStyle.FixedToolWindow && m_f.FormBorderStyle != System.Windows.Forms.FormBorderStyle.SizableToolWindow) { if (m_f.WindowState == FormWindowState.Maximized) { g.DrawImage(MaximumNormalButtonImage, maxRect); } else { g.DrawImage(MaximumButtonImage, maxRect); } g.DrawImage(MinimumButtonImage, minRect); } } else if (m_f.HelpButton) { if (m_f.FormBorderStyle != System.Windows.Forms.FormBorderStyle.FixedToolWindow && m_f.FormBorderStyle != System.Windows.Forms.FormBorderStyle.SizableToolWindow) { g.DrawImage(HelpButtonImage, maxRect); } } } } protected virtual void OnCaptionContextMenu(int x, int y) { if (this.CaptionContextMenu != null) this.CaptionContextMenu.Show(x, y); } #endregion #region 消息 private int m_IsDrawButton = 0