240 lines
6.4 KiB
C++
Raw Permalink Normal View History

2024-11-16 21:51:11 +08:00

2024-11-05 15:46:45 +08:00
2024-04-02 15:36:52 +08:00
bool tryopenclipboard(HWND hwnd = 0)
{
2024-01-08 23:37:00 +08:00
bool success = false;
2024-04-02 15:36:52 +08:00
for (int i = 0; i < 50; i++)
{
if (OpenClipboard(hwnd))
{
2024-01-08 23:37:00 +08:00
success = true;
break;
}
2024-04-02 15:36:52 +08:00
else
{
2024-01-08 23:37:00 +08:00
Sleep(10);
}
}
return success;
}
2024-07-14 15:09:37 +08:00
std::optional<std::wstring> clipboard_get_internal()
2024-04-02 15:36:52 +08:00
{
2025-01-08 13:47:46 +08:00
if (!IsClipboardFormatAvailable(CF_UNICODETEXT))
return {};
2024-04-02 15:36:52 +08:00
if (tryopenclipboard() == false)
2024-07-14 15:09:37 +08:00
return {};
2025-01-08 13:47:46 +08:00
std::optional<std::wstring> data = {};
2024-04-02 15:36:52 +08:00
do
{
HANDLE hData = GetClipboardData(CF_UNICODETEXT);
if (hData == 0)
break;
LPWSTR pszText = static_cast<LPWSTR>(GlobalLock(hData));
if (pszText == 0)
break;
2024-09-14 10:39:56 +08:00
data = std::move(std::wstring(pszText));
2024-01-08 23:37:00 +08:00
GlobalUnlock(hData);
} while (false);
CloseClipboard();
return data;
}
2024-07-14 15:09:37 +08:00
2024-11-05 15:46:45 +08:00
DECLARE_API bool clipboard_get(void (*cb)(const wchar_t *))
2024-07-14 15:09:37 +08:00
{
auto data = std::move(clipboard_get_internal());
if (!data)
return false;
cb(data.value().c_str());
return true;
}
2024-11-05 15:46:45 +08:00
DECLARE_API bool clipboard_set(HWND hwnd, wchar_t *text)
2024-04-02 15:36:52 +08:00
{
2024-01-08 23:37:00 +08:00
bool success = false;
2024-04-02 15:36:52 +08:00
// static HWND hwnd=CreateWindowExA(0,"STATIC",0,0,0,0,0,0,0,0,0,0);
if (tryopenclipboard(hwnd) == false)
return false;
2024-01-08 23:37:00 +08:00
EmptyClipboard();
2024-04-02 15:36:52 +08:00
do
{
2024-01-08 23:37:00 +08:00
HGLOBAL hClipboardData;
size_t len = wcslen(text) + 1;
hClipboardData = GlobalAlloc(GMEM_MOVEABLE, len * sizeof(wchar_t));
2024-04-02 15:36:52 +08:00
if (hClipboardData == 0)
break;
auto pchData = (wchar_t *)GlobalLock(hClipboardData);
if (pchData == 0)
2024-10-01 22:58:41 +08:00
{
GlobalFree(hClipboardData);
2024-04-02 15:36:52 +08:00
break;
2024-10-01 22:58:41 +08:00
}
2024-04-02 15:36:52 +08:00
wcscpy_s(pchData, len, text);
GlobalUnlock(hClipboardData);
2024-10-01 22:58:41 +08:00
if (SetClipboardData(CF_UNICODETEXT, hClipboardData))
success = true;
else
{
GlobalFree(hClipboardData);
}
2024-04-02 15:36:52 +08:00
2024-01-08 23:37:00 +08:00
} while (false);
CloseClipboard();
return success;
2024-06-06 20:58:53 +08:00
}
2025-01-08 13:47:46 +08:00
inline bool iscurrentowndclipboard()
{
auto ohwnd = GetClipboardOwner();
DWORD pid;
GetWindowThreadProcessId(ohwnd, &pid);
return pid == GetCurrentProcessId();
}
2024-07-14 15:09:37 +08:00
static void clipboard_callback_1(void (*callback)(const wchar_t *, bool), HANDLE hsema, HWND *hwnd)
2024-06-06 20:58:53 +08:00
{
const wchar_t CLASS_NAME[] = L"LunaClipboardListener";
WNDCLASS wc = {};
wc.lpfnWndProc = [](HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
2024-11-15 07:46:53 +08:00
static auto callbackx = [](HWND hWnd)
2024-06-06 20:58:53 +08:00
{
2024-07-14 15:09:37 +08:00
auto data = clipboard_get_internal();
2025-01-08 13:47:46 +08:00
if (!data)
return;
2024-06-06 20:58:53 +08:00
auto callback_ = reinterpret_cast<decltype(callback)>(GetWindowLongPtrW(hWnd, GWLP_USERDATA));
2025-01-08 13:47:46 +08:00
if (!callback_)
return;
callback_(data.value().c_str(), iscurrentowndclipboard());
2024-11-15 07:46:53 +08:00
};
#ifndef WINXP
if (WM_CLIPBOARDUPDATE == message)
{
callbackx(hWnd);
2024-06-06 20:58:53 +08:00
}
2024-11-15 07:43:50 +08:00
#else
2024-11-15 09:11:45 +08:00
// 根据文档这样做是正确的且在win11下管用。但到xp上就读不到了。。
2024-11-15 07:43:50 +08:00
static HWND nextviewer;
switch (message)
{
case WM_CREATE:
{
nextviewer = SetClipboardViewer(hWnd);
}
break;
case WM_CHANGECBCHAIN:
{
if ((HWND)wParam == nextviewer)
nextviewer = (HWND)lParam;
if (nextviewer)
SendMessage(nextviewer, message, wParam, lParam);
}
break;
case WM_DESTROY:
{
ChangeClipboardChain(hWnd, nextviewer);
}
break;
case WM_DRAWCLIPBOARD:
{
2024-11-15 07:46:53 +08:00
callbackx(hWnd);
2024-11-15 07:43:50 +08:00
if (nextviewer)
SendMessage(nextviewer, message, wParam, lParam);
}
}
#endif
2024-06-06 20:58:53 +08:00
return DefWindowProc(hWnd, message, wParam, lParam);
};
wc.hInstance = GetModuleHandle(0);
wc.lpszClassName = CLASS_NAME;
static auto _ = RegisterClass(&wc);
HWND hWnd = CreateWindowEx(
WS_EX_CLIENTEDGE, CLASS_NAME, CLASS_NAME, WS_OVERLAPPEDWINDOW,
0, 0, 0, 0,
NULL, NULL, GetModuleHandle(0), 0);
SetWindowLongPtrW(hWnd, GWLP_USERDATA, (LONG_PTR)callback);
*hwnd = hWnd;
ReleaseSemaphore(hsema, 1, 0);
MSG msg = {};
while (GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
2024-11-04 23:10:41 +08:00
DECLARE_API HWND clipboard_callback(void (*callback)(const wchar_t *, bool))
2024-06-06 20:58:53 +08:00
{
2025-01-08 13:47:46 +08:00
#ifndef WINXP
2024-06-06 20:58:53 +08:00
HANDLE hsema = CreateSemaphoreW(0, 0, 10, 0);
HWND hwnd;
2025-01-02 17:47:21 +08:00
std::thread(clipboard_callback_1, callback, hsema, &hwnd).detach();
2024-06-06 20:58:53 +08:00
WaitForSingleObject(hsema, INFINITE);
CloseHandle(hsema);
if (AddClipboardFormatListener(hwnd))
return hwnd;
else
return NULL;
2024-11-15 07:49:38 +08:00
#else
2025-01-08 13:47:46 +08:00
static HANDLE clipboardUpdate;
clipboardUpdate = CreateEventW(nullptr, FALSE, TRUE, NULL);
auto __ = SetWindowsHookExW(WH_GETMESSAGE, [](int statusCode, WPARAM wParam, LPARAM lParam)
{
if (statusCode == HC_ACTION && wParam == PM_REMOVE && ((MSG*)lParam)->message == WM_CLIPBOARDUPDATE) SetEvent(clipboardUpdate);
return CallNextHookEx(NULL, statusCode, wParam, lParam); }, NULL, GetCurrentThreadId());
std::thread([=]
2024-11-15 09:11:45 +08:00
{
2025-01-08 13:47:46 +08:00
while (WaitForSingleObject(clipboardUpdate, INFINITE) == WAIT_OBJECT_0)
{
auto data = clipboard_get_internal();
if(data)
callback(data.value().c_str(), iscurrentowndclipboard());
2025-01-08 14:04:05 +08:00
} })
2024-11-15 09:11:45 +08:00
.detach();
2025-01-08 13:47:46 +08:00
return (HWND) new std::pair<HANDLE, HHOOK>{clipboardUpdate, __};
2024-11-15 09:11:45 +08:00
#endif
2025-01-08 13:47:46 +08:00
}
2024-11-04 23:10:41 +08:00
DECLARE_API void clipboard_callback_stop(HWND hwnd)
2024-06-06 20:58:53 +08:00
{
2024-11-15 09:11:45 +08:00
#ifndef WINXP
2024-06-06 20:58:53 +08:00
if (!hwnd)
return;
RemoveClipboardFormatListener(hwnd);
DestroyWindow(hwnd);
2024-11-15 09:11:45 +08:00
#else
2025-01-08 13:47:46 +08:00
auto __ = (std::pair<HANDLE, HHOOK> *)(hwnd);
UnhookWindowsHookEx(__->second);
CloseHandle(__->first);
delete __;
2024-11-15 09:11:45 +08:00
#endif
2024-06-06 20:58:53 +08:00
}
2024-10-01 22:58:41 +08:00
2024-11-04 23:10:41 +08:00
DECLARE_API bool clipboard_set_image(HWND hwnd, void *ptr, size_t size)
2024-10-01 22:58:41 +08:00
{
size -= sizeof(BITMAPFILEHEADER);
HGLOBAL hDib = GlobalAlloc(GMEM_MOVEABLE, size);
if (!hDib)
return false;
void *pDib = GlobalLock(hDib);
if (!pDib)
{
GlobalFree(hDib);
return false;
}
memcpy((char *)pDib, (char *)ptr + sizeof(BITMAPFILEHEADER), size);
if (tryopenclipboard(hwnd) == false)
return false;
EmptyClipboard();
if (!SetClipboardData(CF_DIB, hDib))
{
GlobalFree(hDib);
CloseClipboard();
return false;
}
CloseClipboard();
return true;
}