I have written a small test application that inserts files (with hardcoded paths) into the currently active folder/application via delayed rendering. It works as expected. But I have a question - why does PeekMessage
always return FALSE
? But if you remove the PeekMessage
call, Wndproc
will never be called. I read a similar post, but I'm creating a window in the same thread in which I'm trying to process messages.
Code:
static LRESULT CALLBACK WindProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam) {
switch (Msg) {
case WM_RENDERALLFORMATS: {
OpenClipboard(hWnd);
EmptyClipboard();
}
case WM_RENDERFORMAT: {
printf("WM_RENDERFORMAT received");
<Here the file paths are copied to the clipboard>
if (Msg == WM_RENDERALLFORMATS)
CloseClipboard();
return 0;
}
case WM_DESTROYCLIPBOARD:
return 0;
}
return DefWindowProc(hWnd, Msg, wParam, lParam);
}
HWND hwnd_;
void thread_(void* ignored) {
WNDCLASSEX wcx = { 0 };
wcx.cbSize = sizeof(WNDCLASSEX);
wcx.lpfnWndProc = WindProc;
wcx.hInstance = GetModuleHandle(NULL);
wcx.lpszClassName = TEXT("my_class");
RegisterClassEx(&wcx);
hwnd_ = CreateWindowEx(0, TEXT("my_class"), TEXT(""), 0, 0, 0, 0, 0, HWND_MESSAGE, NULL, GetModuleHandle(NULL), NULL);
MSG msg;
while (true) {
if (PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE)) {
printf("PeekMessage returned TRUE\n");
TranslateMessage(&msg);
DispatchMessage(&msg);
break;
}
Sleep(1000);
}
}
void main() {
CloseHandle((HANDLE)_beginthread(thread_, 0, NULL));
// let's give some time to thread to create msg window
Sleep(100);
if (OpenClipboard(hwnd_)) {
EmptyClipboard();
SetClipboardData(CF_HDROP, NULL);
CloseClipboard();
}
while (true) {
Sleep(100);
}
}
PeekMessage()
(andGetMessage()
) only returns messages that are posted viaPostMessage()
orPostThreadMessage()
to the calling thread's message queue.PeekMessage()
returns FALSE when there is no posted message available at the moment it is called.GetMessage()
blocks until a posted message becomes available.However, the messages in question (
WM_RENDERFORMAT
andWM_RENDERALLFORMATS
) are instead being sent viaSendMessage...()
directly to the target window. But, the messages are being sent by another thread, soPeekMessage()
(orGetMessage()
) is still needed, as they internally dispatch messages that are sent across thread boundaries.This is stated in
PeekMessage()
's documentation:As well as the
GetMessage()
documentation:And
SendMessage()
's documentation:With that said, your handling of the
WM_RENDERALLFORMATS
message is wrong. For one thing, it must not callEmptyClipboard()
, and for another, it is not checking whether your app is still the clipboard owner before rendering its data. See What is the proper handling of WM_RENDERFORMAT and WM_RENDERALLFORMATS? for why these points are important.Also, you have a race condition in your main thread. Before it calls
OpenClipboard()
, it sleeps for only 100ms to wait for the window to be created first, but there is no guarantee thathwnd_
will have been assigned within that 100ms. It could take that long just for the worker thread to start running, for instance.A better option is to have the worker thread signal an event when the window is created, and then have the main thread wait on that event (even better would be to simply move the initial
SetClipboardData()
call into the worker thread itself after creating the window, but you have dismissed that option).Try something more like this instead: