- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我已经使用来自 Codeproject 的 BandObjectLib ( Extending Explorer with Band Objects using .NET and Windows Forms ) 实现了 DeskBand 任务栏工具栏,修改后支持 IDeskBand2 界面,它允许 Windows Vista 中的“开始”菜单在启用我的 DeskBand 任务栏工具栏时保持透明。但是,组合框或文本框中显示的文本会显示底层桌面背景的颜色与文本的原始颜色混合。
标签没有这个问题,因为它通常是使用 GDI(+) 绘制的,它会忽略渲染文本(而不是标签的背景)上的 DWMComposition。
我认为问题是因为 DWM 在 Vista 上关于某些文本元素的工作方式,这在以下页面中有解释:
Using Vista Controls on Aero Glass
Windows Vista Aero Pt. 1 - Adding Glass to a Windows Forms Application
Adding or Retrofitting Aero Glass into Legacy Windows Applications
我只在我的 DeskBand 工具栏上使用组合框,所以我只需要知道如何强制组合框不使用 DWM 显示,即使 DWM 在系统上启用并且通过 IDeskBand2 的实现在 DeskBand 上启用界面。
更新:我进一步研究了它,C++ 代码位于 Adding or Retrofitting Aero Glass into Legacy Windows Applications似乎最有可能让这个工作成为可能,因此组合框显示文本不透明。如果有人可以查看仅与组合框有关的代码并帮助我让它为 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);
}
谢谢,
约翰·伦尼迈尔
MuvEnum, LLC
最佳答案
这太痛苦了,我不知道为什么 Microsoft 没有让 WinForms 控件与 DWMManager 一起工作。 TextBox 很容易做到,您可以在其上方重新绘制位图。 DropDown 比较棘手,因为它内部有一个本地控件“编辑”框。由于它不是 .Net 控件(编辑部分),因此您无法使用 .Net 类轻松重绘它(我仍在尝试弄清楚如何执行此操作)。
不过,我已经弄清楚如何让 DropDown 在玻璃上呈现,但前提是文本框部分被禁用(通过更改样式来完成)。不太理想。我的在 VB.Net 中,我仍在努力。然而,我受到这个项目的启发,这个人用 C# 完成了它,可能对你有很大帮助:
http://dwmwinform.codeplex.com/
附带说明一下,WPF 的所有控件都支持 Aero Glass 效果。它的功能更强大,但使用起来也更耗时(IMO ......而且,如果你正在 retrofit WinForms 应用程序,WPF 对你没有任何好处)。我只是倾向于喜欢 WinForms,因为我编写业务应用程序并且不太关心或没有时间编写动画(WPF 很酷,不要我错了,我更喜欢 WinForms)。
这是我继承 ComboBox(在 Vb.Net 中)的起点。这个网站没有正确发布整个类(class),所以我只包括类(class)的内部:
''' <summary>
''' Enum of Windows messages that will trigger the redraw of the control
''' </summary>
''' <remarks></remarks>
Public Enum WindowsMessage
WM_CHAR = &H100
WM_KEYDOWN = &H102
WM_MOUSEMOVE = &H200
WM_PAINT = 15
WM_PRINT = &H314
End Enum
''' <summary>
''' Constructor
''' </summary>
''' <remarks></remarks>
Sub New()
Me.DropDownStyle = ComboBoxStyle.DropDownList
End Sub
''' <summary>
''' Processing of incoming messages. We're going to get a bitmap of the control and then
''' redraw it onto the form when a few specified windows messages come through.
''' </summary>
''' <param name="m"></param>
Protected Overrides Sub WndProc(ByRef m As System.Windows.Forms.Message)
MyBase.WndProc(m)
Select Case m.Msg
Case WindowsMessage.WM_PAINT, WindowsMessage.WM_CHAR, WindowsMessage.WM_KEYDOWN, _
WindowsMessage.WM_MOUSEMOVE, WindowsMessage.WM_PRINT
RedrawControlAsBitmap(Me.Handle)
End Select
End Sub
''' <summary>
''' Redraws a given control as a bitmap ontop of itself.
''' </summary>
''' <param name="hwnd"></param>
''' <remarks></remarks>
Public Sub RedrawControlAsBitmap(ByVal hwnd As IntPtr)
Dim c As Control = Control.FromHandle(hwnd)
If c IsNot Nothing Then
Using bm As New Bitmap(c.Width, c.Height)
c.DrawToBitmap(bm, c.ClientRectangle)
Using g As Graphics = c.CreateGraphics
g.DrawImage(bm, New Point(0, 0))
End Using
End Using
End If
c = Nothing
End Sub
关于toolbar - 如何在 DeskBand 任务栏工具栏上强制使用 C# 呈现 ComboBox 的 GDI+? (透明度问题),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/870369/
我正在尝试使用以下代码以编程方式将 DeskBand 对象添加到系统托盘: [ComImport, Guid("6D67E846-5B9C-4db8-9CBC-DDE12F4254F1"), Inte
我正在尝试在 Windows 8.1 中实现 Deskband。我以某种方式让用户界面和其他一切正常工作。这听起来可能很愚蠢,但我不知道如何更改出现在任务栏右键单击菜单上的 Toolbars 子菜单选
我有一个用 C++ 编写的 Deskband DLL,它是通过调用 regsvr32 mydeskband.dll 注册的。 Deskband 有一个用 C# 编写的配置应用程序,我可以使用桌面应用程
我想创建一个 DeskBand 来在我的 Windows 任务栏上显示一些信息,但我正在努力实现我需要的 C++ 功能。我接触 C++ 已经有 10-15 年了。在过去的 7 年里,我一直完全在 .N
我正在寻找一个完整的示例,用于在 win7 64 位的任务栏上创建桌面工具栏。 在 C++ 中。谢谢 最佳答案 Here这是您可以使用的一个示例。 关于c++ - 在 c++ 中寻找 win7 64
如何在 Windows 7 中将自定义工具栏放在任务面板之前的任务栏上?有一个名为 Pokki 的流行应用程序可以做到这一点。 我想知道我该怎么做? Edit1:我需要一种 100% 的方式来访问 W
我已经使用来自 Codeproject 的 BandObjectLib ( Extending Explorer with Band Objects using .NET and Windows Fo
我是一名优秀的程序员,十分优秀!