12.3 实现一个剪贴板查看器

个人分类: 《Windows 程序设计》学习之旅

摘录于《Windows程序(第5版,珍藏版).CHarles.Petzold 著》P462

        如果某个程序能获知剪贴板内容的改变,这个程序就被称为“剪贴板查看器”。Windows 带有一个剪贴板查看器,但是你也可以编写自己的剪贴板查看器程序。剪贴板查看器通过发送给查看器窗口过程的消息得知剪贴板内容的更改。

12.3.1  剪贴板查看器链

        在 Windows 下可以同时运行任意数目的剪贴板查看器应用程序,并且剪贴板的改变都能通知到它们。但是从 Windows 的角度来看,只有一个剪贴板查看器,我称之为“当前剪贴板查看器”。Windows 只维护一个标识当前剪贴板查看器的窗口句柄,并且只给那个窗口发送消息,通知它剪贴板内容改变了。

        剪贴板查看器应用程序应该加入“剪贴板查看器链”,这样所有运行的剪贴板查看器程序就都能收到 Windows 发给当前剪贴板查看器的消息。当某程序把自己注册为剪贴板查看器时,这个程序就成为当前剪贴板查看器。Windows 把上一个当前剪贴板查看器的窗口句柄交给这个程序,然后这个程序保存此句柄。当它收到剪贴板查看器消息时,它会把消息发送给剪贴板查看器链中下一个程序的窗口过程。

12.3.2  剪贴板查看器函数和消息

        通过调用 SetClipboardViewer 函数,程序就能成为剪贴板查看器链的一部分。如果此程序的主要目的就是作为一个剪贴板查看器,那么它可以在处理 WM_CREATE 消息时调用此函数。这个函数返回上一个当前剪贴板查看器的窗口句柄。程序应该用静态变量来保存这个句柄:

 
  1. static HWND hwndNextViewer;

  2. [其他程序行]

  3. case WM_CREATE:

  4. [其他程序行]

  5. hwndNextViewer = SetClipboardViewer (hwnd);

如果你的程序在 Windows 会话期间是头一个成为剪贴板查看器的程序,那么 hwndNextViewer 将为 NULL。

        一旦剪贴板内容改变了,Windows 就会向当前剪贴板查看器(即最新注册成为为剪贴板查看器的窗口)发送 WM_DRAWCLIPBOARD 消息。剪贴板查看器链的最后一个程序(即第一个注册成为剪贴板查看器的窗口)将会有一个值为 NULL 的 hwndNextViewer。如果 hwndNextViewer 值为 NULL,程序就直接返回,不需要把消息发送给另一个程序。(不要把 WM_DRAWCLIPBOARD 和 WM_PAINTCLIPBOARD 消息混为一谈。WM_PAINTCLIPBOARD 是由剪贴板查看器发给使用 CF_OWNERDISPLAY 剪贴板格式的程序的。WM_DRAWCLIPBOARD 则是由 Windows 发给当前剪贴板查看器的。)

        处理 WM_DRAWCLIPBOARD 消息的最简单的方法是把消息发给下一个剪贴板查看器(除非 hwndNextViewer 为 NULL),并且使窗口的客户区无效:

 
  1. case WM_DRAWCLIPBOARD:

  2. if (hwndNextViewer)

  3. SendMessage (hwndNextViewer, message, wParam, lParam);

  4. InvalidateRect (hwnd, NULL, TRUE);

  5. return 0;

处理 WM_PAINT 消息期间,你可以通过普通的 OpenClipboard、GetClipboardData 和 CloseClipboard 调用读取剪贴板内容。

        当某程序想从剪贴板查看器链中退出时,必须调用 ChangeClipboardChain。这个函数需要退出的程序的窗口句柄和它下一个剪贴板查看器的窗口句柄:

ChangeClipboardChain(hwnd, hwndNextViewer);

当程序调用 ChangeClipboardChain 时,Windows 会向当前剪贴板查看器发送一个 WM_CHANGECBCHAIN 消息。该消息的 wParam 参数是要从链里退出的窗口的句柄(即 ChangeClipboardChain 的第一个参数),lParam 是将要退出的窗口的下一个剪贴板查看器的窗口句柄(即 ChangeClipboardChain 的第二个参数)。

        当你的程序收到了 WM_CHANGECBCHAIN 消息,则必须检查 wParam 是否等于你保存的 hwndNextViewer 值。如果相等,你的程序必须把 hwndNextViewer 设为 lParam。这个操作确保将来你得到的任何 WM_CHANGECBCHAIN 消息不会被发送给从链里退出的那个窗口。如果 wParam 不等于 hwndNextViewer,并且 hwndNextViewer 不为 NULL,就把消息发送给下一个剪贴板查看器:

 
  1. case WM_CHANGECBCHAIN:

  2. if ((HWND) wParam == hwndNextViewer)

  3. hwndNextViewer = (HWND)lParam);

  4.  
  5. else if (hwndNextViewer)

  6. SendMessage (hwndNextViewer, message, wParam, lParam);

  7. return 0;

并不一定要加上 else if 语句,它只是检查 hwndNextViewer 是否为非 NULL。hwndNextViewer 值为 NULL 意味着执行这段代码的程序是链里的最后一个剪贴板查看器,在这种情况下,消息不应该再传递。

        如果你的程序在要终止时仍然在剪贴板查看器链里,则必须把它从链里删除。可以在处理 WM_DESTROY 消息时调用 ChangeClipboardChain 来完成删除:

 
  1. case WM_DESTROY:

  2. ChangeClipboardChain (hwnd, hwndNextViewer);

  3. PostQuitMessage (0);

  4. return 0;

        Windows 也有一个让程序获得第一个剪贴板查看器的窗口句柄的函数:

hwndViewer = GetClipboardViewer();

这个函数通常没有什么用。如果当前剪贴板查看器不存在,则返回值为 NULL。

        这里有一个例子说明剪贴板查看器链是怎样工作的。当 Windows 刚开始启动时,当前剪贴板查看器为 NULL:

        当前剪贴板查看器:                                                      NULL

        拥有窗口句柄 hwnd1 的程序调用了 SetClipboardViewer。该函数会返回 NULL,这个值成为此程序里的 hwndNextViewer 值:

        当前剪贴板查看器:                                                      hwnd1

        hwnd1 的下一个查看器:                                             NULL

        拥有窗口句柄 hwnd2 的第二个程序现在调用 SetClipboardViewer,并且取得 hwnd1:

        当前剪贴板查看器:                                                      hwnd2

        hwnd2 的下一个查看器:                                            hwnd1

        hwnd1 的下一个查看器:                                            NULL

        第三个程序(hwnd3)和第四个程序(hwnd4)也调用了 SetClipboardViewer,并且取得 hwnd2 和 hwnd3:

        当前剪贴板查看器:                                                      hwnd4

        hwnd4 的下一个查看器:                                            hwnd3

        hwnd3 的下一个查看器:                                            hwnd2

        hwnd2 的下一个查看器:                                            hwnd1

        hwnd1 的下一个查看器:                                            NULL

        当剪贴板内容发生变化时,Windows 向 hwnd4 发送 WM_DRAWCLIPBOARD 消息,hwnd4 把消息发给 hwnd3,hwnd3 再发给 hwnd2,hwnd2 发给 hwnd1,hwnd1 返回。

        现在 hwnd2 决定从链中退出,它调用以下函数:

ChangeClipboardChain (hwnd2, hwnd1);

        Windows 会向 hwnd4 发送 WM_CHANGECBCHAIN 消息,相应的 wParam 等于 hwnd2,lParam 等于 hwnd1。由于 hwnd4 的下一个查看器时 hwnd3,因此 hwnd4 把这个消息发给 hwnd3。现在 hwnd3 注意到 wParam 等于它的下一个查看器(hwnd2),所以它把它的下一个查看器设置成等于 lParam(hwnd1)并且返回。这个使命就完成了。剪贴板查看器现在看起来如下:

        当前剪贴板查看器:                                                      hwnd4

        hwnd4 的下一个查看器:                                            hwnd3

        hwnd3 的下一个查看器:                                            hwnd1

        hwnd1 的下一个查看器:                                            NULL

12.3.3  一个简单的剪贴板查看器

        剪贴板查看器不一定要和 Windows 提供的查看器一样复杂。比如,剪贴板查看器可以只显示一种剪贴板格式。图 12-2 所示的 CLIPVIEW 程序就是一个只显示 CF_TEXT 格式的剪贴板查看器。

 
  1. /*---------------------------------------------

  2. CLIPVIEW.C -- Simple Clipboard Viewer

  3. (c) Charles Petzold, 1998

  4. ----------------------------------------------*/

  5.  
  6. #include <windows.h>

  7.  
  8. LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);

  9.  
  10. int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,

  11. PSTR szCmdLine, int iCmdShow)

  12. {

  13. static TCHAR szAppName[] = TEXT(".ClipView.");

  14. HWND hwnd;

  15. MSG msg;

  16. WNDCLASS wndclass;

  17.  
  18. wndclass.style = CS_HREDRAW | CS_VREDRAW;

  19. wndclass.lpfnWndProc = WndProc;

  20. wndclass.cbClsExtra = 0;

  21. wndclass.cbWndExtra = 0;

  22. wndclass.hInstance = hInstance;

  23. wndclass.hIcon = LoadIcon(NULL, IDI_APPLICATION);

  24. wndclass.hCursor = LoadCursor(NULL, IDC_ARROW);

  25. wndclass.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);

  26. wndclass.lpszMenuName = NULL;

  27. wndclass.lpszClassName = szAppName;

  28.  
  29. if (!RegisterClass(&wndclass))

  30. {

  31. MessageBox(NULL, TEXT("This program requires Windows NT!"),

  32. szAppName, MB_ICONERROR);

  33. return 0;

  34. }

  35.  
  36. hwnd = CreateWindow(szAppName,

  37. TEXT("Simple Clipboard Viewer (Text Only)"),

  38. WS_OVERLAPPEDWINDOW,

  39. CW_USEDEFAULT, CW_USEDEFAULT,

  40. CW_USEDEFAULT, CW_USEDEFAULT,

  41. NULL, NULL, hInstance, NULL);

  42.  
  43. ShowWindow(hwnd, iCmdShow);

  44. UpdateWindow(hwnd);

  45.  
  46. while (GetMessage(&msg, NULL, 0, 0))

  47. {

  48. TranslateMessage(&msg);

  49. DispatchMessage(&msg);

  50. }

  51. return msg.wParam;

  52. }

  53.  
  54. LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)

  55. {

  56. static HWND hwndNextViewer;

  57. HGLOBAL hGlobal;

  58. HDC hdc;

  59. PTSTR pGlobal;

  60. PAINTSTRUCT ps;

  61. RECT rect;

  62.  
  63. switch (message)

  64. {

  65. case WM_CREATE:

  66. hwndNextViewer = SetClipboardViewer(hwnd);

  67. return 0;

  68.  
  69. case WM_CHANGECBCHAIN:

  70. if ((HWND)wParam == hwndNextViewer)

  71. hwndNextViewer = (HWND)lParam;

  72.  
  73. else if (hwndNextViewer)

  74. SendMessage(hwndNextViewer, message, wParam, lParam);

  75.  
  76. return 0;

  77.  
  78. case WM_DRAWCLIPBOARD:

  79. if (hwndNextViewer)

  80. SendMessage(hwndNextViewer, message, wParam, lParam);

  81. InvalidateRect(hwnd, NULL, TRUE);

  82. return 0;

  83.  
  84. case WM_PAINT:

  85. hdc = BeginPaint(hwnd, &ps);

  86. GetClientRect(hwnd, &rect);

  87. OpenClipboard(hwnd);

  88.  
  89. #ifdef UNICODE

  90. hGlobal = GetClilpboardData(CF_UNICODETEXT);

  91. #else

  92. hGlobal = GetClipboardData(CF_TEXT);

  93. #endif

  94.  
  95. if (hGlobal != NULL)

  96. {

  97. pGlobal = (PTSTR)GlobalLock(hGlobal);

  98. DrawText(hdc, pGlobal, -1, &rect, DT_EXPANDTABS);

  99. GlobalUnlock(hGlobal);

  100. }

  101.  
  102. CloseClipboard();

  103. EndPaint(hwnd, &ps);

  104. return 0;

  105.  
  106. case WM_DESTROY:

  107. ChangeClipboardChain(hwnd, hwndNextViewer);

  108. PostQuitMessage(0);

  109. return 0;

  110. }

  111. return DefWindowProc(hwnd, message, wParam, lParam);

  112. }

        按照前面讨论的,CLIPVIEW 处理 WM_CREATE、WM_CHANGECBCHAIN、WM_DRAWCLIPBOARD 和 WM_DESTROY 消息。WM_PAINT 消息打开剪贴板并用 CF_TEXT 格式调用 GetClipboardData。如果该函数返回一个全局内存句柄,CLIPVIEW 就锁定它并用 DrawText 在客户区显示文本。

        能处理标准格式之外的其他格式的剪贴板查看器(比如 Windows 提供的剪贴板查看器)还需要作一些额外的工作,比如显示当前剪贴板里所有格式的名称。你可以调用 EnumClipboardFormats 得到格式名,还可以调用 GetClipboardFormatName 来取得非标准格式的名称。使用 CF_OWNERDISPLAY 格式的剪贴板查看器必须向剪贴板所有者发送以下四个消息才能显示数据:

        WM_PAINTCLIPBOARD                                             WM_VSCROLLCLIPBOARD

        WM_SIZECLIPBOARD                                                WM_HSCROLLCLIPBOARD

        如果想实现这种剪贴板查看器,你得通过 GetClipboardOwner 取得剪贴板所有者的窗口句柄。而且,在需要更新剪贴板查看器的客户区时,还需要向剪贴板所有者的窗口发送上述这些信息。

猜你喜欢

转载自blog.csdn.net/lazyxi/article/details/82828420
今日推荐