熟悉窗口超类化的请进

joshua0137 2014-02-25 09:00:10
如下代码,希望通过超类化方法更改对话框背景,但测试发现只对部分对话框有效
请问原因和修改方法。


BOOL CsttApp::InitInstance()
{
superclassfunc();
// 如果一个运行在 Windows XP 上的应用程序清单指定要
// 使用 ComCtl32.dll 版本 6 或更高版本来启用可视化方式,
//则需要 InitCommonControlsEx()。否则,将无法创建窗口。
INITCOMMONCONTROLSEX InitCtrls;
InitCtrls.dwSize = sizeof(InitCtrls);
// 将它设置为包括所有要在应用程序中使用的
// 公共控件类。
InitCtrls.dwICC = ICC_WIN95_CLASSES;
InitCommonControlsEx(&InitCtrls);
CWinAppEx::InitInstance();
// 初始化 OLE 库
if (!AfxOleInit())
{
AfxMessageBox(IDP_OLE_INIT_FAILED);
return FALSE;
}
AfxEnableControlContainer();
// 标准初始化
// 如果未使用这些功能并希望减小
// 最终可执行文件的大小,则应移除下列
// 不需要的特定初始化例程
// 更改用于存储设置的注册表项
// TODO: 应适当修改该字符串,
// 例如修改为公司或组织名
SetRegistryKey(_T("应用程序向导生成的本地应用程序"));
LoadStdProfileSettings(4); // 加载标准 INI 文件选项(包括 MRU)
InitContextMenuManager();
InitKeyboardManager();
InitTooltipManager();
CMFCToolTipInfo ttParams;
ttParams.m_bVislManagerTheme = TRUE;
theApp.GetTooltipManager()->SetTooltipParams(AFX_TOOLTIP_TYPE_ALL,
RUNTIME_CLASS(CMFCToolTipCtrl), &ttParams);


// 注册应用程序的文档模板。文档模板
// 将用作文档、框架窗口和视图之间的连接
CSingleDocTemplate* pDocTemplate;
pDocTemplate = new CSingleDocTemplate(
IDR_MAINFRAME,
RUNTIME_CLASS(CsttDoc),
RUNTIME_CLASS(CMainFrame), // 主 SDI 框架窗口
RUNTIME_CLASS(CsttView));
if (!pDocTemplate)
return FALSE;
AddDocTemplate(pDocTemplate);

// 分析标准外壳命令、DDE、打开文件操作的命令行
CCommandLineInfo cmdInfo;
ParseCommandLine(cmdInfo);

// 调度在命令行中指定的命令。如果
// 用 /RegServer、/Register、/Unregserver 或 /Unregister 启动应用程序,则返回 FALSE。
if (!ProcessShellCommand(cmdInfo))
return FALSE;
// 唯一的一个窗口已初始化,因此显示它并对其进行更新
m_pMainWnd->ShowWindow(SW_SHOW);
m_pMainWnd->UpdateWindow();
// 仅当具有后缀时才调用 DragAcceptFiles
// 在 SDI 应用程序中,这应在 ProcessShellCommand 之后发生
AfxMessageBox(_T("Hello, world"));
return TRUE;
}



WNDPROC lpOriDlgProc = NULL;
LRESULT CALLBACK JDlgProc(HWND hwnd, UINT msg, WPARAM w, LPARAM l);
void superclassfunc()
{
WNDCLASSEX stWC;
GetClassInfoEx(AfxGetInstanceHandle(), WC_DIALOGBOX, &stWC);
stWC.cbSize = sizeof(stWC);
lpOriDlgProc = stWC.lpfnWndProc;
stWC.lpfnWndProc = &JDlgProc;
stWC.hInstance = AfxGetInstanceHandle();
stWC.lpszClassName = WC_DIALOGBOX;
RegisterClassEx(&stWC);
}

LRESULT CALLBACK JDlgProc(HWND hwnd, UINT msg, WPARAM w, LPARAM l)
{
if (!(WS_POPUP & ::GetWindowLong(hwnd, GWL_STYLE)))
return (*lpOriDlgProc)(hwnd, msg, w, l);
switch(msg)
{
case WM_CTLCOLORBTN:
case WM_CTLCOLORDLG:
case WM_CTLCOLOREDIT:
case WM_CTLCOLORLISTBOX:
case WM_CTLCOLORMSGBOX:
case WM_CTLCOLORSCROLLBAR:
case WM_CTLCOLORSTATIC:
case WM_CTLCOLOR:
{
static CBrush br;
if (!br.m_hObject)
br.CreateSolidBrush(RGB(0, 128, 0));
if (msg == WM_CTLCOLORSTATIC)
{
HDC hdc = (HDC)w;
::SetBkMode(hdc, TRANSPARENT);
}
return reinterpret_cast<LRESULT>(br.m_hObject);
}
}
return (*lpOriDlgProc)(hwnd, msg, w, l);
}
...全文
49 点赞 收藏 3
写回复
3 条回复
切换为时间正序
请发表友善的回复…
发表回复
「已注销」 2014-02-25
我没细看,但我看你这里就有问题。WC_DIALOGBOX 我不知道是什么,但是 WC_DIALOG 我知道是系统的对话框类 ATOM。况且 GetClassInfoEx 调用前其参数中要求 WNDCLASSEX 的 cbSize 成员必须设置为该结构体的大小。你连判断都不判断,直接写个 void,错了都不知道哪些地方有错误。
BOOL superclassfunc()
{
    WNDCLASSEX stWC;
    stWC.cbSize = sizeof(stWC);
    if (GetClassInfoEx(AfxGetInstanceHandle(), WC_DIALOG, &stWC)) {
        lpOriDlgProc = stWC.lpfnWndProc;
        stWC.lpfnWndProc = &JDlgProc;
        stWC.hInstance = AfxGetInstanceHandle();
        stWC.lpszClassName = _T("你的类名");
        return RegisterClassEx(&stWC);
    }
    return FALSE;
}
超类化肯定要使用一个新的类名,那你这两处都是 WC_DIALOGBOX 当然有问题了。另外,超类化之后,对话框设计器里的对话框类名也要使用你这个新类名。
回复
joshua0137 2014-02-25
原因已查明,实例句柄问题
回复
gfm688 2014-02-25
WC_DIALOGBOX 是什么? 你可以用GetClassInfoEx获取,说明是一个已注册的窗口类,再用这个类名注册应该就不行了 如果一定要用这个类名,正确的做法应该是用SetClassLong修改WindowProc
回复
发动态
发帖子
界面
创建于2007-09-28

1.5w+

社区成员

VC/MFC 界面
申请成为版主
社区公告
暂无公告