C++ WTL8.1 如何在具有消息循环的新线程中创建窗口

C++ WTL8.1 How to create a window in a new thread with message loop?

本文关键字:新线程 线程 窗口 创建 循环 WTL8 消息 C++      更新时间:2023-10-16

我看到这问了很多方法,但从未见过实际的工作代码截图。

我尝试过这个无疑是天真的尝试创建一个具有新窗口和消息循环的新线程。我有一个函数,它必须打开一个窗口并处理其消息,并且必须在可以从没有现有消息循环(也没有其他窗口)或通常的 mfc 消息循环或 WTL 消息循环的应用程序调用它的环境中运行。我看到了一些关于AddMessageLoop和Modules的东西?但这似乎是针对主要应用程序的。在任何情况下,那里可能有也可能没有 WTL 模块。需要一个具有基本消息循环的独立窗口。传入窗口尚未打开的 WTL 类,因此窗口在与循环相同的线程中打开。是否还必须在新线程中创建类对象?

// does not work.... 
static DWORD WINAPI MyRunThread(__in LPVOID lpParameter)
{
CMessageLoop theLoop;
WTLsubclass *nav = (WTLsubclass *) lpParameter;
nav->CreateWindow(); 
int nRet = theLoop.Run();
return nRet;
}
CreateThread(0,0,MyRunThread,&nav,0,0);

有关需要更改的内容,请参阅注释。陷阱 #1 是和运算符不适用于 WTL 类陷阱#2,永远不要命名您的事件。如果与未知的其他对象链接,很可能有人已经使用了相同的名称,并且Windows认为相同的名称意味着相同的事件,即使使用了多个CreateEvent()调用。没有名字,没有所谓的不同事件的"别名"。

非陷阱,如果您不希望"主要"消息处理程序处理您的消息,则不需要_Module或"告诉"有关消息循环的任何信息。如果你想要独立具有独立循环的窗口,请参见下文。

====

======================================================
static DWORD WINAPI MyRunThread(__in LPVOID lpParameter)
{
CMessageLoop theLoop;
WTLsubclass *nav = (WTLsubclass *) lpParameter;
nav->CreateWindow(); 
SetEvent(WindowCreatedEvent) // signal event HANDLE type for worker thread
int nRet = theLoop.Run();
return nRet;
}
//CreateThread(0,0,MyRunThread,&nav,0,0); // & wont work on WTL class objects
// some genius overloaded the & operator on WTL class objects, just because in C++ you can
//Workaround to get address of WTL class object
CLASSnav nav[1];
CLASSnav *pnav = nav; // because arrays is address of first element.
CreateThread(0,0,MyRunThread,pnav,0,0);

SetWindowLongPtrGetWindowLongPtr ,使用这些从窗口 proc 中访问您的实例,然后在 WM_ACTIVATE 上设置条件变量。

在客户端代码中:

// Create window, this setups the window and runs a message loop, its guts are ran in a different thread
createwindow();
// Wait to be shown, this presumes the existence of a mutext (m) and a condition_variable (cv) in your instance
std::unique_lock<std::mutex> lock(m);
cv.wait_for(lock, std::chrono::milliseconds(800));

在窗口设置和创建过程中,这可以在您的线程进程中:

// Create window (RegisterClass, CreateWindow, etc)
...
SetWindowLongPtr(hwnd, GWLP_USERDATA, /*your instance*/ ...);   
ShowWindow(hwnd, ...);
...
// Pump

在窗口进程中:

...
switch(msg)
{
    case WM_ACTIVATE:
    {
        your_instance* inst = (your_instance*)GetWindowLongPtr(hwnd, GWLP_USERDATA);
        inst->cv->notify_one();
    }
    break;
...