获取Windows 8自动颜色主题的活动颜色

use*_*880 21 .net c# wpf winapi windows-8

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

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

有没有办法获得当前/实际的配色方案并被告知C#的变化?

Bol*_*ock 15

是的,这是可能的.但是要注意:这包含了相当多的Win32互操作(这意味着P /从托管代码调用本机DLL),并且只能用于某些未记录的API.虽然,所涉及的唯一未记录的功能是获取窗口颜色方案(或DWM称之为窗口着色颜色),这在另一个问题中有所涉及:

Vista/7:如何获得玻璃颜色?

在我自己的项目中,我使用了一个调用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;
}
Run Code Online (Sandbox Code Playgroud)

我已经对它进行了测试,它在Windows 8及其自动窗口着色功能方面效果很好.正如上面的链接所示,您可以在注册表中查找颜色值作为P/Invoke的替代方法,但我没有测试过该方法,并且如上所述,这些方法没有记录,并且不能保证稳定.

一旦获得用于绘制渐变画笔的颜色,当窗口颜色方案发生更改时,无论是手动操作还是Windows自动更新,画笔都不会更新.值得庆幸的是,Windows 会在发生这种情况时广播WM_DWMCOLORIZATIONCOLORCHANGED窗口消息,因此您只需要监听该消息并在发送时更新颜色.你可以通过挂钩窗口过程(WndProc())来做到这一点.

价值WM_DWMCOLORIZATIONCOLORCHANGED0x320; 你需要将它定义为某个常量,以便在代码中使用它.

此外,与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;
    }
}
Run Code Online (Sandbox Code Playgroud)

当Windows转换颜色更改时,WM_DWMCOLORIZATIONCOLORCHANGED将在转换中的每个关键帧处调度,因此在颜色更改期间,您将在短时间内收到大量消息.这个是正常的; 只需像往常一样更新渐变画笔,您会注意到当Windows转换窗口颜色方案时,渐变也会与其他窗口框架一起平滑过渡.

请记住,您可能需要考虑DWM不可用的情况,例如在Windows XP上运行时,或者在禁用桌面组合的Windows Vista或更高版本上运行时.您还需要确保不要过度使用此功能,否则可能会导致严重的性能损失并降低应用速度.

  • 这是我发布的第一个Windows 8答案... Windows 8.不太破旧...... (2认同)
  • 我没有特别测试过此代码,但是Windows Vista中未公开的功能`DwmGetColorizationParameters`仍可在Windows 10上正常工作,以检索强调色/颜色。但是,您可以在Windows 8上调用更多有趣的未公开文档的API:https://code.msdn.microsoft.com/windowsdesktop/How-to-get-Accent-brush-6652403f (2认同)

Bil*_*ees 14

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

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

<Grid Background="{x:Static SystemParameters.WindowGlassBrush}">
Run Code Online (Sandbox Code Playgroud)

但是,使用此分配时,当Windows更改其颜色时,背景颜色不会自动更新.不幸的是,SystemParameters并没有提供WindowGlassBrushKeyWindowGlassColorKey性能与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();
        }
    }
}
Run Code Online (Sandbox Code Playgroud)