2009-10-28 73 views

回答

29

您可以在您的对话框类ON_WM_CTLCOLOR,而无需创建一个新的CStatic派生类:

BEGIN_MESSAGE_MAP(CMyDialog, CDialog) 
    //{{AFX_MSG_MAP(CMyDialog) 
    ON_WM_CTLCOLOR() 
    //}}AFX_MSG_MAP 
END_MESSAGE_MAP() 

HBRUSH CMyDialog::OnCtlColor(CDC* pDC, CWnd *pWnd, UINT nCtlColor) 
{ 
    switch (nCtlColor) 
    { 
    case CTLCOLOR_STATIC: 
     pDC->SetTextColor(RGB(255, 0, 0)); 
     return (HBRUSH)GetStockObject(NULL_BRUSH); 
    default: 
     return CDialog::OnCtlColor(pDC, pWnd, nCtlColor); 
    } 
} 

注意,上面的代码将所有静态控件的对话框中文字。但是你可以使用pWnd变量来过滤你想要的控件。

+0

你说得对。这是另一种方式。 我刚才提到了我认为更好的方式。在第二种情况下,您必须将代码添加到每个想要以其他颜色显示标签的对话框中。 – 2009-10-29 13:54:20

+1

是的,我同意,在这种情况下,你的方式是更好的方法。如果有人想在整个对话框(或应用程序)中进行重大修改,我可以使用我的方式。 – djeidot 2009-10-29 19:07:37

+2

返回(HBRUSH)GetStockObject(NULL_BRUSH);会导致绘画问题(或时髦的透明背景'功能')。您可能想要返回(HBRUSH)GetStockObject(WHITE_BRUSH); – 2010-05-28 17:45:59

7

不幸的是,你不会找到在静态控件类SetTextColor方法。如果你想改变一个CStatic的文本颜色,你将不得不编码更多。

在我看来,最好的方法是创建自己的CStatic派生类(CMyStatic),并在那里获取ON_WM_CTLCOLOR_REFLECT通知消息。

BEGIN_MESSAGE_MAP(CMyStatic, CStatic) 
    //{{AFX_MSG_MAP(CMyStatic) 
    ON_WM_CTLCOLOR_REFLECT() 
    //}}AFX_MSG_MAP 
END_MESSAGE_MAP() 

HBRUSH CColorStatic::CtlColor(CDC* pDC, UINT nCtlColor) 
{ 
    pDC->SetTextColor(RGB(255,0,0)); 

    return (HBRUSH)GetStockObject(NULL_BRUSH); 
} 

很明显,您可以使用成员变量和setter方法来替换红色(RGB(255,0,0))。

问候。

+1

感谢您的解决方案。它有一些意想不到的副作用。现在我的静态控件从我的主视图中有不同的控件背景颜色,并且文本背景也是白色的。 – Sheen 2010-12-24 10:55:25

6

只是跟进的涂装问题(透明背景),这引起*回报(HBRUSH)GetStockObject(NULL_BRUSH); *

轻易地变更如下:

HBRUSH hBrush = CDialog::OnCtlColor(pDC, pWnd, nCtlColor); 

if (nCtlColor == CTLCOLOR_STATIC && 
    pWnd->GetSafeHwnd() == GetDlgItem(XXX)->GetSafeHwnd() 
) pDC->SetTextColor(RGB(255, 0, 0));  

return hBrush; 

希望这帮助。

2

从这里和其他地方给出的答案,如何创建派生类来代替处理着色本身的CStatic并不明显。

因此,以下是适用于我的方法,使用MSVS 2013版本12.0.40629.00更新5.我可以在资源编辑器中放置一个“静态文本” - 控件,然后用TColorText替换成员变量的类型。

在.H文件:

class TColorText : public CStatic 
{ 
protected: 
    DECLARE_MESSAGE_MAP() 

public: 
    // make the background transparent (or if ATransparent == true, restore the previous background color) 
    void setTransparent(bool ATransparent = true); 
    // set background color and make the background opaque 
    void SetBackgroundColor(COLORREF); 
    void SetTextColor(COLORREF); 

protected: 
    HBRUSH CtlColor(CDC* pDC, UINT nCtlColor); 

private: 
    bool MTransparent = true; 
    COLORREF MBackgroundColor = RGB(255, 255, 255); // default is white (in case someone sets opaque without setting a color) 
    COLORREF MTextColor = RGB(0, 0, 0); // default is black. it would be more clean 
             // to not use the color before set with SetTextColor(..), but whatever... 
}; 

中的.cpp文件:

void TColorText::setTransparent(bool ATransparent) 
{ 
    MTransparent = ATransparent; 
    Invalidate(); 
} 

void TColorText::SetBackgroundColor(COLORREF AColor) 
{ 
    MBackgroundColor = AColor; 
    MTransparent = false; 
    Invalidate(); 
} 

void TColorText::SetTextColor(COLORREF AColor) 
{ 
    MTextColor = AColor; 
    Invalidate(); 
} 

BEGIN_MESSAGE_MAP(TColorText, CStatic) 
    ON_WM_CTLCOLOR_REFLECT() 
END_MESSAGE_MAP() 

HBRUSH TColorText::CtlColor(CDC* pDC, UINT nCtlColor) 
{ 
    pDC->SetTextColor(MTextColor); 
    pDC->SetBkMode(TRANSPARENT); // we do not want to draw background when drawing text. 
            // background color comes from drawing the control background. 
    if(MTransparent) 
    return nullptr; // return nullptr to indicate that the parent object 
        // should supply the brush. it has the appropriate background color. 
    else 
    return (HBRUSH) CreateSolidBrush(MBackgroundColor); // color for the empty area of the control 
} 
1

非常有帮助。

https://msdn.microsoft.com/de-de/library/0wwk06hc.aspx

相同方式来

HBRUSH hBrush = CDialog::OnCtlColor(pDC, pWnd, nCtlColor); 
if (nCtlColor == CTLCOLOR_STATIC && 
    pWnd->GetSafeHwnd() == GetDlgItem(XXX)->GetSafeHwnd() 
) pDC->SetTextColor(RGB(255, 0, 0));  
return hBrush;