【问题标题】:Get the active color of Windows 8 automatic color theme获取 Windows 8 自动颜色主题的活动颜色
【发布时间】:2012-11-19 14:29:45
【问题描述】:

在 Windows 8 中,我已将配色方案设置为自动,并将我的壁纸配置为在 x 分钟后更改。配色方案根据活动壁纸而变化。

我正在开发 WPF 应用程序,并希望在 Windows 更改配色方案以匹配当前壁纸时更改我的渐变。

有没有办法获取当前/实际的配色方案并在 C# 中收到更改通知?

【问题讨论】:

  • 这包含相当多的互操作性,并且仅适用于某些未记录的 API。事实上,我已经用我自己的 Windows Vista/7 项目完成了这项工作,并在 Windows 8 上成功测试了它。我可以尝试在此基础上给出一个答案,但我需要一段时间。另见:Vista/7: How to get glass color?
  • 非常感谢您提供的信息。我使用了注册表技巧,效果很好。
  • 这里有一个更好且受支持(即不依赖未记录的 API)的答案:stackoverflow.com/questions/50840395/…

标签: c# .net wpf winapi windows-8


【解决方案1】:

是的,这是可能的。但是请注意:这包含相当多的 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 或更高版本上运行时。您还需要确保不要过度使用它,否则可能会严重影响性能并降低应用速度。

【讨论】:

  • 这是我从... Windows 8 发布的第一个 Windows 8 答案。不太破旧...
  • 我没有特别测试过这段代码,但是来自 Windows Vista 的未记录函数 DwmGetColorizationParameters 在 Windows 10 上仍然可以正常工作,以检索重音/着色颜色。但是,您可以在 Windows 8 上调用更多有趣的未记录 API:code.msdn.microsoft.com/windowsdesktop/…
【解决方案2】:

这可以在 .NET 4.5 及更高版本中完成,无需 P/Invokes。 SystemParameters 类现在具有静态 WindowGlassBrushWindowGlassColor 属性以及 StaticPropertyChanged 事件。

从 XAML,您可以绑定到 WindowGlassBrush 属性,例如:

<Grid Background="{x:Static SystemParameters.WindowGlassBrush}">

但是,通过此分配,当 Windows 更改其颜色时,背景颜色不会自动更新。不幸的是,SystemParameters 提供 WindowGlassBrushKeyWindowGlassColorKey 属性以用作带有 DynamicResource 的 ResourceKey,因此获取更改通知需要后面的代码来处理 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();
        }
    }
}

【讨论】:

  • .NET 4.5 框架中的 SystemParameters.WindowGlassColor 属性只是包装了 DwmGetColorizationColor
  • 我在模板中使用这个属性,有没有办法更新模板以用新颜色替换旧颜色?
  • 遗憾的是,SystemParameters.WindowGlassBrush 不会返回正确的强调色,至少在我的机器上是不同的。我使用此代码:gist.github.com/paulcbetts/3c6aedc9f0cd39a77c37
  • 正如@RobertMuehsig 建议的那样,如果启用了“从我的背景中自动选择强调色”,这将无法获得正确的颜色
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-02-19
  • 2017-08-12
相关资源
最近更新 更多