是的,这是可能的。但是请注意:这包含相当多的 Win32 互操作(这意味着从托管代码 P/Invokes 到本机 DLL),并且仅适用于某些未记录的 API。虽然,所涉及的唯一未记录的功能是用于获取窗口配色方案(或 DWM 称之为窗口着色颜色),这在另一个问题中有所涉及:
Vista/7: How to get glass color?
在我自己的项目中,我使用了对DwmGetColorizationParameters()的调用:
internal static class NativeMethods
{
[DllImport("dwmapi.dll", EntryPoint="#127")]
internal static extern void DwmGetColorizationParameters(ref DWMCOLORIZATIONPARAMS params);
}
public struct DWMCOLORIZATIONPARAMS
{
public uint ColorizationColor,
ColorizationAfterglow,
ColorizationColorBalance,
ColorizationAfterglowBalance,
ColorizationBlurBalance,
ColorizationGlassReflectionIntensity,
ColorizationOpaqueBlend;
}
我已经对其进行了测试,它非常适用于 Windows 8 及其自动窗口着色功能。正如上面链接中所建议的那样,您可以在注册表中查找颜色值作为 P/Invoke 的替代方法,但我没有测试过该方法,并且如前所述,这些方法没有记录并且不能保证稳定。
一旦您获得绘制渐变画笔的颜色,无论是手动还是由 Windows 自动更改窗口配色方案,画笔都不会更新。幸运的是,Windows 会在发生这种情况时广播WM_DWMCOLORIZATIONCOLORCHANGED window message,因此您只需收听该消息并在发送时更新您的颜色。您可以通过挂钩到窗口过程 (WndProc()) 来做到这一点。
WM_DWMCOLORIZATIONCOLORCHANGED 的值为0x320;您需要在某处将其定义为常量,以便在代码中使用它。
此外,与 WinForms 不同的是,WPF 窗口没有要覆盖的虚拟 WndProc() 方法,因此您必须创建一个虚拟方法并将其挂钩作为其关联窗口句柄 (HWND) 的委托。
从我的这些答案中获取一些示例代码:
我们有:
const int WM_DWMCOLORIZATIONCOLORCHANGED = 0x320;
private IntPtr hwnd;
private HwndSource hsource;
private void Window_SourceInitialized(object sender, EventArgs e)
{
if ((hwnd = new WindowInteropHelper(this).Handle) == IntPtr.Zero)
{
throw new InvalidOperationException("Could not get window handle.");
}
hsource = HwndSource.FromHwnd(hwnd);
hsource.AddHook(WndProc);
}
private static Color GetWindowColorizationColor(bool opaque)
{
var params = NativeMethods.DwmGetColorizationParameters();
return Color.FromArgb(
(byte)(opaque ? 255 : params.ColorizationColor >> 24),
(byte)(params.ColorizationColor >> 16),
(byte)(params.ColorizationColor >> 8),
(byte) params.ColorizationColor
);
}
private IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
{
switch (msg)
{
case WM_DWMCOLORIZATIONCOLORCHANGED:
/*
* Update gradient brushes with new color information from
* NativeMethods.DwmGetColorizationParams() or the registry.
*/
return IntPtr.Zero;
default:
return IntPtr.Zero;
}
}
当 Windows 转换颜色变化时,WM_DWMCOLORIZATIONCOLORCHANGED 会在转换中的每个关键帧处调度,因此在颜色变化期间您会在短时间内收到大量消息。这个是正常的;只需像往常一样更新渐变画笔,您就会注意到,当 Windows 转换窗口颜色方案时,您的渐变也会与其余的窗口框架一起平滑转换。
请记住,您可能需要考虑 DWM 不可用的情况,例如在 Windows XP 上运行时,或者在禁用桌面组合的情况下在 Windows Vista 或更高版本上运行时。您还需要确保不要过度使用它,否则可能会严重影响性能并降低应用速度。