我已经使用 Codeproject 中的 BandObjectLib( Extending Explorer with Band Objects using .NET and Windows Forms )实现了一个 DeskBand 任务栏工具栏,修改为支持 IDeskBand2 界面,该界面允许 Windows Vista 中的开始菜单在我的 DeskBand 任务栏工具栏时保持透明度已启用。但是,组合框或文本框中显示的文本会显示与文本原始颜色混合的基础桌面背景颜色。
标签没有这个问题,因为它通常是使用 GDI(+) 绘制的,它忽略了渲染文本(而不是标签的背景)上的 DWMComposition。
我认为这个问题是因为 DWM 在 Vista 上的工作方式涉及某些文本元素,这在以下页面中进行了解释:
在 Aero Glass 上使用 Vista 控件
Windows Vista Aero Pt。1 - 将 Glass 添加到 Windows 窗体应用程序
将 Aero Glass 添加或改装到旧版 Windows 应用程序中
我只在我的 DeskBand 工具栏上使用一个组合框,所以我只需要知道如何强制组合框不使用 DWM 显示,即使 DWM 在系统上启用并且通过 IDeskBand2 接口的实现在 DeskBand 上启用。
更新:我已经对其进行了进一步的研究,在将 Aero Glass 添加或改装到旧版 Windows 应用程序中的 C++ 代码似乎最有可能让这个工作正常工作,因此组合框显示文本不透明。如果有人可以查看仅与组合框有关的代码并帮助我使其适用于 C# 组合框,那将是我的一个月!我已经开始赏金,希望能得到答案。
下面是来自上述项目的 EditProc.cpp 类,它应该可以快速了解我正在寻找的解决方案。要获得完整的图片,您需要查看完整的项目:
/*
*
* $RCSfile: aeroedit.cpp,v $
* $Source: /cvs/common/aeroedit.cpp,v $
* $Author: cvs $
* $Revision: 1.12 $
* $Date: 2007/05/20 10:38:25 $
* $State: Exp $
* Copyright (c) Stefan Kuhr
*/
#include <windows.h>
#include <tchar.h>
#include "safassrt.h"
#include "aaeroint.h"
#include "aerosubc.h"
#include "aeroglss.h"
#include <windowsx.h>
#include <gdiplus.h>
using namespace Gdiplus;
static void UpdateIfSelChanged(HWND hWnd, PAERO_SUBCLASS_WND_DATA pWndData)
{
DWORD dwFirst, dwLast;
SendMessage(hWnd, EM_GETSEL, (WPARAM)&dwFirst, (LPARAM)&dwLast);
if(dwFirst!=pWndData->m_dwSelFirst || dwLast!=pWndData->m_dwSelLast)
{
pWndData->m_dwSelFirst = dwFirst;
pWndData->m_dwSelLast = dwLast;
VERIFY(InvalidateRect(hWnd, NULL, TRUE));
VERIFY(UpdateWindow(hWnd));
}
}
static LRESULT CALLBACK EditProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
PAERO_SUBCLASS_WND_DATA pWndData = (PAERO_SUBCLASS_WND_DATA)GetProp(hWnd, WINDOW_DATA_STRING);
ASSERT(pWndData);
ASSERT(pWndData->m_pDwmApiImpl);
WNDPROC pOldProc = pWndData->m_oldWndProc;
ASSERT(pOldProc);
PAERO_SUBCLASS_WND_DATA pWndDataParent = (PAERO_SUBCLASS_WND_DATA)GetProp(GetParent(hWnd), WINDOW_DATA_STRING);
///
/// if aero glass is turned off and if we are not in destruction code,
/// just call the original wnd proc we had prior to subclassing:
///
if(WM_DESTROY!=uMsg && WM_NCDESTROY!=uMsg && WM_DWMCOMPOSITIONCHANGED!=uMsg && pWndDataParent && !pWndData->m_pDwmApiImpl->IsDwmCompositionEnabled())
return CallWindowProc(pOldProc, hWnd, uMsg, wParam, lParam);
if(pWndData->m_uiRedrawMsg==uMsg && pWndData->m_dwFlags & WD_IN_PAINT_CONTROL)
{
HDC hdc = GetDC(hWnd);
hdc = GetDC(hWnd);
if(hdc)
{
RECT rcClient;
GetClientRect(hWnd, &rcClient);
BP_PAINTPARAMS params = { sizeof(BP_PAINTPARAMS) };
params.dwFlags = 0L;//BPPF_ERASE;
HDC hdcPaint = NULL;
HPAINTBUFFER hBufferedPaint = pWndData->m_pUxTheme->BeginBufferedPaint(hdc, &rcClient, BPBF_TOPDOWNDIB, ¶ms, &hdcPaint);
if (hdcPaint)
{
LONG_PTR dwStyle = GetWindowLongPtr(hWnd, GWL_STYLE);
DWORD_PTR dwSyscolorIdx = (dwStyle&WS_DISABLED || dwStyle&ES_READONLY)?COLOR_3DFACE:COLOR_WINDOW;
VERIFY(FillRect(hdcPaint, &rcClient, (HBRUSH)(dwSyscolorIdx+1)));
SendMessage(hWnd, WM_PRINTCLIENT, (WPARAM) hdcPaint, PRF_CLIENT|PRF_CHECKVISIBLE);
/// Make every pixel opaque
VERIFY(S_OK==pWndData->m_pUxTheme->BufferedPaintMakeOpaque_(hBufferedPaint, &rcClient));
VERIFY(S_OK==pWndData->m_pUxTheme->EndBufferedPaint(hBufferedPaint, TRUE));
}
VERIFY(1==ReleaseDC(hWnd, hdc));
pWndData->m_dwFlags &= ~WD_IN_PAINT_CONTROL;
}
return 1;
}
switch(uMsg)
{
case WM_KEYDOWN:
{
LONG_PTR dwStyle = GetWindowLongPtr(hWnd, GWL_STYLE);
if(dwStyle&WS_VSCROLL || dwStyle&ES_MULTILINE)
{
if(!(pWndData->m_dwFlags&WD_CARET_HIDDEN))
{
HideCaret(hWnd);
pWndData->m_dwFlags|=WD_CARET_HIDDEN;
}
}
}
break;
case WM_KEYUP:
case WM_LBUTTONDOWN:
case WM_LBUTTONUP:
case WM_MOUSELEAVE:
{
LONG_PTR dwStyle = GetWindowLongPtr(hWnd, GWL_STYLE);
if(dwStyle&WS_VSCROLL || dwStyle&ES_MULTILINE)
{
if(pWndData->m_dwFlags&WD_CARET_HIDDEN)
{
ShowCaret(hWnd);
pWndData->m_dwFlags&=~WD_CARET_HIDDEN;
}
UpdateIfSelChanged(hWnd, pWndData);
}
}
break;
case WM_NCPAINT:
{
LRESULT lRes = 0;
lRes = CallWindowProc(pOldProc, hWnd, uMsg, wParam, lParam);
DrawEditBorder(hWnd, pWndData);
return lRes;
}
case WM_NCDESTROY:
case WM_DESTROY:
VERIFY(UnsubclassControl(hWnd, EditProc, pWndData));
break;
}
return CallWindowProc(pOldProc, hWnd, uMsg, wParam, lParam);
}
BOOL AeroSubClassEdit(HWND hwnd)
{
return AeroSubClassControl(hwnd, EditProc, WD_IN_PAINT_CONTROL);
}
谢谢,
John Rennemeyer
MuvEnum, LLC