在Windows 8中,我将颜色方案设置为自动,并将我的壁纸配置为在x分钟后更改。配色方案根据活动壁纸而变化。
我正在开发一个WPF应用程序,并希望在Windows更改颜色方案以匹配当前壁纸时更改渐变。
有没有办法获得当前/实际的配色方案并被告知C#的变化?
答案 0 :(得分:14)
是的,这是可能的。但是请注意:这包含了相当多的Win32互操作(这意味着P /从托管代码调用本机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或更高版本上运行时。您还需要确保不要过度使用此功能,否则可能会导致严重的性能损失并降低应用速度。
答案 1 :(得分:12)
这可以在没有P / Invokes的.NET 4.5及更高版本中完成。 SystemParameters类现在具有静态WindowGlassBrush和WindowGlassColor属性以及StaticPropertyChanged事件。
从XAML,您可以绑定到WindowGlassBrush属性,如:
<Grid Background="{x:Static SystemParameters.WindowGlassBrush}">
但是,使用此分配时,当Windows更改其颜色时,背景颜色不会自动更新。不幸的是,SystemParameters 不提供 WindowGlassBrushKey 或 WindowGlassColorKey 属性以用作DynamicResource的ResourceKeys,因此获取更改通知需要代码来处理StaticPropertyChanged事件
public partial class MainWindow : Window
{
public MainWindow()
{
this.InitializeComponent();
SystemParameters.StaticPropertyChanged += this.SystemParameters_StaticPropertyChanged;
// Call this if you haven't set Background in XAML.
this.SetBackgroundColor();
}
protected override void OnClosed(EventArgs e)
{
SystemParameters.StaticPropertyChanged -= this.SystemParameters_StaticPropertyChanged;
base.OnClosed(e);
}
private void SetBackgroundColor()
{
this.Background = SystemParameters.WindowGlassBrush;
}
private void SystemParameters_StaticPropertyChanged(object sender, System.ComponentModel.PropertyChangedEventArgs e)
{
if (e.PropertyName == "WindowGlassBrush")
{
this.SetBackgroundColor();
}
}
}