jjzjj

c++ - 渐变父窗口上的背景组合框?

coder 2024-06-20 原文

我有带渐变背景的窗口。组合框具有自己的背景画笔。
如何删除组合框中的白角?如何更改画笔或其他方式。
在图片的白色角上标有红色框。



我将组合框创建为:

DWORD dwStyle = WS_CHILD | CBS_DROPDOWNLIST;
if (m_bVisible) dwStyle |= WS_VISIBLE;
m_hWnd = CreateWindow(WC_COMBOBOX, NULL, dwStyle,
    m_posX, m_posY, m_width, m_height, m_hParent, (HMENU)m_id, m_hInstance, NULL);

我尝试用消息WM_CTLCOLOREDIT更改背景画笔,但没有效果:
case WM_CTLCOLOREDIT:
    if ((HWND)lParam == m_hSrcListBox)
    {
        return (LRESULT)m_hBrush;
    }
break;

===已解决。工作版本===

第一种方式。

在父级WndProc中:
case WM_CTLCOLORSTATIC:
    if ((HWND)lParam == m_hSrcListBox)
    {
        return (LRESULT)m_pSrcListBox->GetHbrush();
    }
break;

在我的课上:
//
// CListBox::GetHbrush().
//
// Get brush.
//
HBRUSH CListBox::GetHbrush()
{
    if (!m_hBrush)
    {
        m_hBrush = CreateTransparentBackgroundBrush(m_hParent, m_hWnd);
    }
    return m_hBrush;
}

创建透明背景:
//
// CListBox::CreateTransparentBackgroundBrush().
//
// Create transparent background for element.
//
HBRUSH CListBox::CreateTransparentBackgroundBrush(HWND parent, HWND client)
{
    RECT rct;
    POINT p1;
    POINT p2;
    GetWindowRect(client, &rct);
    p1.x = rct.left;
    p1.y = rct.top;
    ScreenToClient(parent, &p1);
    p2.x = rct.right;
    p2.y = rct.bottom;
    ScreenToClient(parent, &p2);

    HDC hdcParent = GetDC(parent);
    HDC hdcClient = GetDC(client);

    HDC hdcmem = CreateCompatibleDC(hdcClient);
    HBITMAP hbitmap = CreateCompatibleBitmap(hdcClient, p2.x - p1.x, p2.y - p1.y);
    SelectObject(hdcmem, hbitmap);
    BitBlt(hdcmem, 0, 0, p2.x - p1.x, p2.y - p1.y, hdcParent, p1.x, p1.y, SRCCOPY);

    HBRUSH pattern = CreatePatternBrush(hbitmap);

    DeleteDC(hdcmem);
    DeleteObject(hbitmap);
    ReleaseDC(client, hdcClient);
    ReleaseDC(parent, hdcParent);

    return pattern;
}

第二种方式。

在父级WndProc中,在WM_ERASEBKGND消息中绘制背景,则角不会。
case WM_ERASEBKGND:
    m_hdc = (HDC)wParam;

    // draw background.

    return TRUE;
break;

两种方法的结果:

最佳答案

对于对话框,处理WM_CTLCOLORDLG并为组合框返回背景画笔
如果要在对话框中显示此组合框,则技巧实际上是在对话框的窗口过程中处理WM_CTLCOLORDLG消息。响应此消息,您将画笔的句柄返回到对话框将用于绘制其背景的画笔。

case WM_CTLCOLORDLG:
{
   // NOTE: This code is wrong because it creates a new brush object each time it processes
   //       the message, which it promptly leaks. It is merely for demonstration purposes.
   //       Normally, you would create the brush once, in response to WM_INITDIALOG,
   //       cache it away, and return that same cached handle each time, finally destroying
   //       the brush in response to WM_NCDESTROY.
   HBRUSH hBrush = CreateSolidBrush(RGB(255, 120, 0));
   return reinterpret_cast<INT_PTR>(hBrush);
}

这是更改对话框背景颜色的标准记录方法,它也解决了组合框的问题。显然,无论出于何种原因,组合框控件也使用此笔刷绘制其背景。我想他们在绘画时会向 parent 发送WM_CTLCOLORDLG消息。
当然,这将限制您使用GDI画笔的图形功能。您可以绘制所需的任何系统色或纯色,甚至可以使用图案填充或图案/位图画笔,但是没有创建渐变画笔的简单方法。 (GDI +有一个,但没有GDI。)通常没关系-您只需在GradientFill(甚至WM_PAINT)消息处理程序中调用WM_ERASEBKGND函数即可。这对于对话框的背景来说效果很好,但是组合框仍然使用WM_CTLCOLORDLG返回的画笔绘制背景,因此它的角落上仍然有4个点用COLOR_3DFACE绘制(这是默认对话框过程返回的画笔)。

NULL_BRUSH返回空画笔(HOLLOW_BRUSH / WM_CTLCOLORDLG)也不起作用。它会稍微改变外观,以至于右上角和左下角像素现在都填充了类似于COLOR_3DSKSHADOW的内容,但仍然可见地填充了实际背景渐变以外的其他颜色。

因此,如果您真的希望它看起来不错,则只剩下一个选择:将手柄返回给GDI画笔。当然,它必须与用于绘制对话框背景的画笔相同。
如果要进行渐变填充,我能想到的唯一解决方案是使用图案/位图画笔,其中位图(DDB或DIB)就是您的渐变。效果不是很好,但是至少Windows 9x将我们限制为8×8模式的日子已经过去了。也许比我更有创造力的人可以使用此信息来考虑更好的解决方法?

对于其他窗口,请处理WM_CTLCOLORSTATIC并为组合框返回背景画笔
所有这些都用于对话框。但是,如果您在标准窗口(即对话框以外的其他窗口)中显示组合框,该怎么办?在这种情况下,永远不会发送WM_CTLCOLORDLG消息。
而是,组合框将WM_CTLCOLORSTATIC消息发送到其父窗口,然后使用响应该消息而返回的画笔句柄绘制其背景。
我知道这很奇怪。我只是通过进行实证检验而偶然发现了这一点,但我不确定是什么原因。如果我不得不猜测,我会说CBS_DROPDOWNLIST样式使组合框不可编辑(即,由于没有Edit控件,它不是真正的组合框),因此它使用WM_CTLCOLOREDIT而不是WM_CTLCOLORSTATIC。禁用的“编辑”框也会发送WM_CTLCOLORSTATIC,禁用的组合框也将发送“正常”的CBS_SIMPLECBS_DROPDOWN样式。
仍然很奇怪,这仅在启用Aero主题(Vista和7)时发生。在Windows 10或Luna主题(XP下为Visual Styles)或Classic主题中,不会发生这种情况。 (我没有在Windows 8或8.1上进行测试。)我想这并不重要,因为所有其他主题都绘制了一个简单的矩形组合框,没有角像素可显示背景。
无论采用哪种逻辑,解决方案仍然是处理WM_CTLCOLORSTATIC消息并返回希望组合框用来绘制其背景的画笔。
这里的注意事项与上述对话框的注意事项相同。如果您的窗口使用纯色背景或系统颜色,则您无家可归。只需将手柄返回到与窗口类的背景画笔相同的画笔即可。如果要使用渐变,则需要找出一种以GDI画笔形式表示该渐变的方法。
WNDCLASSEX wcex;
wcex.cbSize        = sizeof(WNDCLASSEX);
wcex.style         = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc   = WndProc;
wcex.cbClsExtra    = 0;
wcex.cbWndExtra    = 0;
wcex.hInstance     = hInstance;
wcex.hIcon         = LoadIcon(hInstance, MAKEINTRESOURCE(IDR_APPLICATION));
wcex.hIconSm       = LoadIcon(hInstance, MAKEINTRESOURCE(IDR_APPLICATION_SMALL));
wcex.hCursor       = LoadCursor(NULL, IDC_ARROW);
wcex.hbrBackground = reinterpret_cast<HBRUSH>(COLOR_3DDKSHADOW + 1);  // background brush
wcex.lpszMenuName  = NULL;
wcex.lpszClassName = TEXT("My Colored Window Class");
RegisterClassEx(&wcex);
case WM_CTLCOLORSTATIC:
{
  // NOTE: No leak here because we're using a system brush in this example.
  return reinterpret_cast<LRESULT>(GetSysColorBrush(COLOR_3DDKSHADOW)); // background brush
}

关于c++ - 渐变父窗口上的背景组合框?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37640434/

有关c++ - 渐变父窗口上的背景组合框?的更多相关文章

  1. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  2. ruby-on-rails - 使用 Sublime Text 3 突出显示 HTML 背景语法中的 ERB? - 2

    所以我在关注Railscast,我注意到在html.erb文件中,ruby代码有一个微弱的背景高亮效果,以区别于其他代码HTML文档。我知道Ryan使用TextMate。我正在使用SublimeText3。我怎样才能达到同样的效果?谢谢! 最佳答案 为SublimeText安装ERB包。假设您安装了SublimeText包管理器*,只需点击cmd+shift+P即可获得命令菜单,然后键入installpackage并选择PackageControl:InstallPackage获取包管理器菜单。在该菜单中,键入ERB并在看到包时选择

  3. ruby-on-rails - date_field_tag,如何设置默认日期? [ rails 上的 ruby ] - 2

    我想设置一个默认日期,例如实际日期,我该如何设置?还有如何在组合框中设置默认值顺便问一下,date_field_tag和date_field之间有什么区别? 最佳答案 试试这个:将默认日期作为第二个参数传递。youcorrectlysetthedefaultvalueofcomboboxasshowninyourquestion. 关于ruby-on-rails-date_field_tag,如何设置默认日期?[rails上的ruby],我们在StackOverflow上找到一个类似的问

  4. ruby-on-rails - openshift 上的 rails 控制台 - 2

    我将我的Rails应用程序部署到OpenShift,它运行良好,但我无法在生产服务器上运行“Rails控制台”。它给了我这个错误。我该如何解决这个问题?我尝试更新ruby​​gems,但它也给出了权限被拒绝的错误,我也无法做到。railsc错误:Warning:You'reusingRubygems1.8.24withSpring.UpgradetoatleastRubygems2.1.0andrun`gempristine--all`forbetterstartupperformance./opt/rh/ruby193/root/usr/share/rubygems/rubygems

  5. ruby-on-rails - 使用 Rmagick 或 ImageMagick 在背景上放置标题 - 2

    我有一张背景图片,我想在其中添加一个文本框。我想弄清楚如何将标题放置在其顶部的正确位置。(我使用标题是因为我需要自动换行功能)。现在,我只能让文本显示在左上角,但我需要能够手动定位它的开始位置。require'RMagick'require'Pry'includeMagicktext="Loremipsumdolorsitamet"img=ImageList.new('template001.jpg')img 最佳答案 这是使用convert的ImageMagick命令行的答案。如果你想在Rmagick中使用这个方法,你必须自己移植

  6. ruby-on-rails - 相关表上的范围为 "WHERE ... LIKE" - 2

    我正在尝试从Postgresql表(table1)中获取数据,该表由另一个相关表(property)的字段(table2)过滤。在纯SQL中,我会这样编写查询:SELECT*FROMtable1JOINtable2USING(table2_id)WHEREtable2.propertyLIKE'query%'这工作正常:scope:my_scope,->(query){includes(:table2).where("table2.property":query)}但我真正需要的是使用LIKE运算符进行过滤,而不是严格相等。然而,这是行不通的:scope:my_scope,->(que

  7. ruby - 使用 `+=` 和 `send` 方法 - 2

    如何将send与+=一起使用?a=20;a.send"+=",10undefinedmethod`+='for20:Fixnuma=20;a+=10=>30 最佳答案 恐怕你不能。+=不是方法,而是语法糖。参见http://www.ruby-doc.org/docs/ProgrammingRuby/html/tut_expressions.html它说Incommonwithmanyotherlanguages,Rubyhasasyntacticshortcut:a=a+2maybewrittenasa+=2.你能做的最好的事情是:

  8. ruby - 如何计算 Liquid 中的变量 +1 - 2

    我对如何计算通过{%assignvar=0%}赋值的变量加一完全感到困惑。这应该是最简单的任务。到目前为止,这是我尝试过的:{%assignamount=0%}{%forvariantinproduct.variants%}{%assignamount=amount+1%}{%endfor%}Amount:{{amount}}结果总是0。也许我忽略了一些明显的东西。也许有更好的方法。我想要存档的只是获取运行的迭代次数。 最佳答案 因为{{incrementamount}}将输出您的变量值并且不会影响{%assign%}定义的变量,我

  9. ruby-on-rails - Ruby - 如何从 ruby​​ 上的 .pfx 文件中提取公钥、rsa 私钥和 CA key - 2

    我有一个.pfx格式的证书,我需要使用ruby​​提取公共(public)、私有(private)和CA证书。使用shell我可以这样做:#ExtractPublicKey(askforpassword)opensslpkcs12-infile.pfx-outfile_public.pem-clcerts-nokeys#ExtractCertificateAuthorityKey(askforpassword)opensslpkcs12-infile.pfx-outfile_ca.pem-cacerts-nokeys#ExtractPrivateKey(askforpassword)o

  10. ruby - (Ruby || Python) 窗口管理器 - 2

    我想用这两种语言中的任何一种(最好是ruby​​)制作一个窗口管理器。老实说,除了我需要加载某种X模块外,我不知道从哪里开始。因此,如果有人有线索,如果您能指出正确的方向,那就太好了。谢谢 最佳答案 XCB,X的下一代API使用XML格式定义X协议(protocol),并使用脚本生成特定语言绑定(bind)。它在概念上与SWIG类似,只是它描述的不是CAPI,而是X协议(protocol)。目前,C和Python存在绑定(bind)。理论上,Ruby端口只是编写一个从XML协议(protocol)定义语言到Ruby的翻译器的问题。生

随机推荐