mfplat/sample: Optimize copying to 2d buffer.
[wine.git] / dlls / comctl32 / pager.c
blob07484c6ebf5887271563fe751c3fb98dce7a533b
1 /*
2 * Pager control
4 * Copyright 1998, 1999 Eric Kohl
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20 * NOTES
22 * This code was audited for completeness against the documented features
23 * of Comctl32.dll version 6.0 on Sep. 18, 2004, by Robert Shearman.
25 * Unless otherwise noted, we believe this code to be complete, as per
26 * the specification mentioned above.
27 * If you discover missing features or bugs please note them below.
29 * TODO:
30 * Implement repetitive button press.
31 * Adjust arrow size relative to size of button.
32 * Allow border size changes.
33 * Styles:
34 * PGS_DRAGNDROP
35 * Notifications:
36 * PGN_HOTITEMCHANGE
37 * Messages:
38 * WM_PRINT and/or WM_PRINTCLIENT
40 * TESTING:
41 * Tested primarily with the controlspy Pager application.
42 * Susan Farley (susan@codeweavers.com)
44 * IMPLEMENTATION NOTES:
45 * This control uses WM_NCPAINT instead of WM_PAINT to paint itself
46 * as we need to scroll a child window. In order to do this we move
47 * the child window in the control's client area, using the clipping
48 * region that is automatically set around the client area. As the
49 * entire client area now consists of the child window, we must
50 * allocate space (WM_NCCALCSIZE) for the buttons and draw them as
51 * a non-client area (WM_NCPAINT).
52 * Robert Shearman <rob@codeweavers.com>
55 #include <stdarg.h>
56 #include <stdlib.h>
57 #include <string.h>
58 #include "windef.h"
59 #include "winbase.h"
60 #include "wingdi.h"
61 #include "winuser.h"
62 #include "winnls.h"
63 #include "commctrl.h"
64 #include "comctl32.h"
65 #include "wine/debug.h"
67 WINE_DEFAULT_DEBUG_CHANNEL(pager);
69 typedef struct
71 HWND hwndSelf; /* handle of the control wnd */
72 HWND hwndChild; /* handle of the contained wnd */
73 HWND hwndNotify; /* handle of the parent wnd */
74 BOOL bUnicode; /* send notifications in Unicode */
75 DWORD dwStyle; /* styles for this control */
76 COLORREF clrBk; /* background color */
77 INT nBorder; /* border size for the control */
78 INT nButtonSize;/* size of the pager btns */
79 INT nPos; /* scroll position */
80 INT nWidth; /* from child wnd's response to PGN_CALCSIZE */
81 INT nHeight; /* from child wnd's response to PGN_CALCSIZE */
82 BOOL bForward; /* forward WM_MOUSEMOVE msgs to the contained wnd */
83 BOOL bCapture; /* we have captured the mouse */
84 INT TLbtnState; /* state of top or left btn */
85 INT BRbtnState; /* state of bottom or right btn */
86 INT direction; /* direction of the scroll, (e.g. PGF_SCROLLUP) */
87 WCHAR *pwszBuffer;/* text buffer for converted notifications */
88 INT nBufferSize;/* size of the above buffer */
89 } PAGER_INFO;
91 #define TIMERID1 1
92 #define TIMERID2 2
93 #define INITIAL_DELAY 500
94 #define REPEAT_DELAY 50
96 /* Text field conversion behavior flags for PAGER_SendConvertedNotify() */
97 enum conversion_flags
99 /* Convert Unicode text to ANSI for parent before sending. If not set, do nothing */
100 CONVERT_SEND = 0x01,
101 /* Convert ANSI text from parent back to Unicode for children */
102 CONVERT_RECEIVE = 0x02,
103 /* Send empty text to parent if text is NULL. Original text pointer still remains NULL */
104 SEND_EMPTY_IF_NULL = 0x04,
105 /* Set text to null after parent received the notification if the required mask is not set before sending notification */
106 SET_NULL_IF_NO_MASK = 0x08,
107 /* Zero out the text buffer before sending it to parent */
108 ZERO_SEND = 0x10
111 static void
112 PAGER_GetButtonRects(const PAGER_INFO* infoPtr, RECT* prcTopLeft, RECT* prcBottomRight, BOOL bClientCoords)
114 RECT rcWindow;
115 GetWindowRect (infoPtr->hwndSelf, &rcWindow);
117 if (bClientCoords)
118 MapWindowPoints( 0, infoPtr->hwndSelf, (POINT *)&rcWindow, 2 );
119 else
120 OffsetRect(&rcWindow, -rcWindow.left, -rcWindow.top);
122 *prcTopLeft = *prcBottomRight = rcWindow;
123 if (infoPtr->dwStyle & PGS_HORZ)
125 prcTopLeft->right = prcTopLeft->left + infoPtr->nButtonSize;
126 prcBottomRight->left = prcBottomRight->right - infoPtr->nButtonSize;
128 else
130 prcTopLeft->bottom = prcTopLeft->top + infoPtr->nButtonSize;
131 prcBottomRight->top = prcBottomRight->bottom - infoPtr->nButtonSize;
135 static void
136 PAGER_DrawButton(HDC hdc, COLORREF clrBk, RECT rc,
137 BOOL horz, BOOL topLeft, INT btnState)
139 UINT flags;
141 TRACE("rc = %s, btnState = %d\n", wine_dbgstr_rect(&rc), btnState);
143 if (btnState == PGF_INVISIBLE)
144 return;
146 if ((rc.right - rc.left <= 0) || (rc.bottom - rc.top <= 0))
147 return;
149 if (horz)
150 flags = topLeft ? DFCS_SCROLLLEFT : DFCS_SCROLLRIGHT;
151 else
152 flags = topLeft ? DFCS_SCROLLUP : DFCS_SCROLLDOWN;
154 switch (btnState)
156 case PGF_HOT:
157 break;
158 case PGF_NORMAL:
159 flags |= DFCS_FLAT;
160 break;
161 case PGF_DEPRESSED:
162 flags |= DFCS_PUSHED;
163 break;
164 case PGF_GRAYED:
165 flags |= DFCS_INACTIVE | DFCS_FLAT;
166 break;
168 DrawFrameControl( hdc, &rc, DFC_SCROLL, flags );
171 /* << PAGER_GetDropTarget >> */
173 static inline LRESULT
174 PAGER_ForwardMouse (PAGER_INFO* infoPtr, BOOL bFwd)
176 TRACE("[%p]\n", infoPtr->hwndSelf);
178 infoPtr->bForward = bFwd;
180 return 0;
183 static inline LRESULT
184 PAGER_GetButtonState (const PAGER_INFO* infoPtr, INT btn)
186 LRESULT btnState = PGF_INVISIBLE;
187 TRACE("[%p]\n", infoPtr->hwndSelf);
189 if (btn == PGB_TOPORLEFT)
190 btnState = infoPtr->TLbtnState;
191 else if (btn == PGB_BOTTOMORRIGHT)
192 btnState = infoPtr->BRbtnState;
194 return btnState;
198 static inline INT
199 PAGER_GetPos(const PAGER_INFO *infoPtr)
201 TRACE("[%p] returns %d\n", infoPtr->hwndSelf, infoPtr->nPos);
202 return infoPtr->nPos;
205 static inline INT
206 PAGER_GetButtonSize(const PAGER_INFO *infoPtr)
208 TRACE("[%p] returns %d\n", infoPtr->hwndSelf, infoPtr->nButtonSize);
209 return infoPtr->nButtonSize;
212 static inline INT
213 PAGER_GetBorder(const PAGER_INFO *infoPtr)
215 TRACE("[%p] returns %d\n", infoPtr->hwndSelf, infoPtr->nBorder);
216 return infoPtr->nBorder;
219 static inline COLORREF
220 PAGER_GetBkColor(const PAGER_INFO *infoPtr)
222 TRACE("[%p] returns %#lx\n", infoPtr->hwndSelf, infoPtr->clrBk);
223 return infoPtr->clrBk;
226 static void
227 PAGER_CalcSize( PAGER_INFO *infoPtr )
229 NMPGCALCSIZE nmpgcs;
230 ZeroMemory (&nmpgcs, sizeof (NMPGCALCSIZE));
231 nmpgcs.hdr.hwndFrom = infoPtr->hwndSelf;
232 nmpgcs.hdr.idFrom = GetWindowLongPtrW (infoPtr->hwndSelf, GWLP_ID);
233 nmpgcs.hdr.code = PGN_CALCSIZE;
234 nmpgcs.dwFlag = (infoPtr->dwStyle & PGS_HORZ) ? PGF_CALCWIDTH : PGF_CALCHEIGHT;
235 nmpgcs.iWidth = infoPtr->nWidth;
236 nmpgcs.iHeight = infoPtr->nHeight;
237 SendMessageW (infoPtr->hwndNotify, WM_NOTIFY, nmpgcs.hdr.idFrom, (LPARAM)&nmpgcs);
239 if (infoPtr->dwStyle & PGS_HORZ)
240 infoPtr->nWidth = nmpgcs.iWidth;
241 else
242 infoPtr->nHeight = nmpgcs.iHeight;
244 TRACE("[%p] PGN_CALCSIZE returns %dx%d\n", infoPtr->hwndSelf, nmpgcs.iWidth, nmpgcs.iHeight );
247 static void
248 PAGER_PositionChildWnd(PAGER_INFO* infoPtr)
250 if (infoPtr->hwndChild)
252 RECT rcClient;
253 int nPos = infoPtr->nPos;
255 /* compensate for a grayed btn, which will soon become invisible */
256 if (infoPtr->TLbtnState == PGF_GRAYED)
257 nPos += infoPtr->nButtonSize;
259 GetClientRect(infoPtr->hwndSelf, &rcClient);
261 if (infoPtr->dwStyle & PGS_HORZ)
263 int wndSize = max(0, rcClient.right - rcClient.left);
264 if (infoPtr->nWidth < wndSize)
265 infoPtr->nWidth = wndSize;
267 TRACE("[%p] SWP %dx%d at (%d,%d)\n", infoPtr->hwndSelf,
268 infoPtr->nWidth, infoPtr->nHeight,
269 -nPos, 0);
270 SetWindowPos(infoPtr->hwndChild, HWND_TOP,
271 -nPos, 0,
272 infoPtr->nWidth, infoPtr->nHeight, 0);
274 else
276 int wndSize = max(0, rcClient.bottom - rcClient.top);
277 if (infoPtr->nHeight < wndSize)
278 infoPtr->nHeight = wndSize;
280 TRACE("[%p] SWP %dx%d at (%d,%d)\n", infoPtr->hwndSelf,
281 infoPtr->nWidth, infoPtr->nHeight,
282 0, -nPos);
283 SetWindowPos(infoPtr->hwndChild, HWND_TOP,
284 0, -nPos,
285 infoPtr->nWidth, infoPtr->nHeight, 0);
288 InvalidateRect(infoPtr->hwndChild, NULL, TRUE);
292 static INT
293 PAGER_GetScrollRange(PAGER_INFO* infoPtr, BOOL calc_size)
295 INT scrollRange = 0;
297 if (infoPtr->hwndChild)
299 INT wndSize, childSize;
300 RECT wndRect;
301 GetWindowRect(infoPtr->hwndSelf, &wndRect);
303 if (calc_size)
304 PAGER_CalcSize(infoPtr);
305 if (infoPtr->dwStyle & PGS_HORZ)
307 wndSize = wndRect.right - wndRect.left;
308 childSize = infoPtr->nWidth;
310 else
312 wndSize = wndRect.bottom - wndRect.top;
313 childSize = infoPtr->nHeight;
316 TRACE("childSize = %d, wndSize = %d\n", childSize, wndSize);
317 if (childSize > wndSize)
318 scrollRange = childSize - wndSize + infoPtr->nButtonSize;
321 TRACE("[%p] returns %d\n", infoPtr->hwndSelf, scrollRange);
322 return scrollRange;
325 static void
326 PAGER_UpdateBtns(PAGER_INFO *infoPtr, INT scrollRange, BOOL hideGrayBtns)
328 BOOL resizeClient;
329 BOOL repaintBtns;
330 INT oldTLbtnState = infoPtr->TLbtnState;
331 INT oldBRbtnState = infoPtr->BRbtnState;
332 POINT pt;
333 RECT rcTopLeft, rcBottomRight;
335 /* get button rects */
336 PAGER_GetButtonRects(infoPtr, &rcTopLeft, &rcBottomRight, TRUE);
338 GetCursorPos(&pt);
339 ScreenToClient( infoPtr->hwndSelf, &pt );
341 /* update states based on scroll position */
342 if (infoPtr->nPos > 0)
344 if (infoPtr->TLbtnState == PGF_INVISIBLE || infoPtr->TLbtnState == PGF_GRAYED)
345 infoPtr->TLbtnState = PGF_NORMAL;
347 else if (!hideGrayBtns && PtInRect(&rcTopLeft, pt))
348 infoPtr->TLbtnState = PGF_GRAYED;
349 else
350 infoPtr->TLbtnState = PGF_INVISIBLE;
352 if (scrollRange <= 0)
354 infoPtr->TLbtnState = PGF_INVISIBLE;
355 infoPtr->BRbtnState = PGF_INVISIBLE;
357 else if (infoPtr->nPos < scrollRange)
359 if (infoPtr->BRbtnState == PGF_INVISIBLE || infoPtr->BRbtnState == PGF_GRAYED)
360 infoPtr->BRbtnState = PGF_NORMAL;
362 else if (!hideGrayBtns && PtInRect(&rcBottomRight, pt))
363 infoPtr->BRbtnState = PGF_GRAYED;
364 else
365 infoPtr->BRbtnState = PGF_INVISIBLE;
367 /* only need to resize when entering or leaving PGF_INVISIBLE state */
368 resizeClient =
369 ((oldTLbtnState == PGF_INVISIBLE) != (infoPtr->TLbtnState == PGF_INVISIBLE)) ||
370 ((oldBRbtnState == PGF_INVISIBLE) != (infoPtr->BRbtnState == PGF_INVISIBLE));
371 /* initiate NCCalcSize to resize client wnd if necessary */
372 if (resizeClient)
373 SetWindowPos(infoPtr->hwndSelf, 0, 0, 0, 0, 0,
374 SWP_FRAMECHANGED | SWP_NOSIZE | SWP_NOMOVE |
375 SWP_NOZORDER | SWP_NOACTIVATE);
377 /* repaint when changing any state */
378 repaintBtns = (oldTLbtnState != infoPtr->TLbtnState) ||
379 (oldBRbtnState != infoPtr->BRbtnState);
380 if (repaintBtns)
381 SendMessageW(infoPtr->hwndSelf, WM_NCPAINT, 0, 0);
384 static LRESULT
385 PAGER_SetPos(PAGER_INFO* infoPtr, INT newPos, BOOL fromBtnPress, BOOL calc_size)
387 INT scrollRange = PAGER_GetScrollRange(infoPtr, calc_size);
388 INT oldPos = infoPtr->nPos;
390 if ((scrollRange <= 0) || (newPos < 0))
391 infoPtr->nPos = 0;
392 else if (newPos > scrollRange)
393 infoPtr->nPos = scrollRange;
394 else
395 infoPtr->nPos = newPos;
397 TRACE("[%p] pos=%d, oldpos=%d\n", infoPtr->hwndSelf, infoPtr->nPos, oldPos);
399 if (infoPtr->nPos != oldPos)
401 /* gray and restore btns, and if from WM_SETPOS, hide the gray btns */
402 PAGER_UpdateBtns(infoPtr, scrollRange, !fromBtnPress);
403 PAGER_PositionChildWnd(infoPtr);
406 return 0;
409 /******************************************************************
410 * For the PGM_RECALCSIZE message (but not the other uses in *
411 * this module), the native control does only the following: *
413 * if (some condition) *
414 * PostMessageW(hwnd, EM_FMTLINES, 0, 0); *
415 * return DefWindowProcW(hwnd, PGM_RECALCSIZE, 0, 0); *
417 * When we figure out what the "some condition" is we will *
418 * implement that for the message processing. *
419 ******************************************************************/
421 static LRESULT
422 PAGER_RecalcSize(PAGER_INFO *infoPtr)
424 TRACE("[%p]\n", infoPtr->hwndSelf);
426 if (infoPtr->hwndChild)
428 INT scrollRange = PAGER_GetScrollRange(infoPtr, TRUE);
430 if (scrollRange <= 0)
432 infoPtr->nPos = -1;
433 PAGER_SetPos(infoPtr, 0, FALSE, TRUE);
435 else
436 PAGER_PositionChildWnd(infoPtr);
439 return 1;
443 static COLORREF
444 PAGER_SetBkColor (PAGER_INFO* infoPtr, COLORREF clrBk)
446 COLORREF clrTemp = infoPtr->clrBk;
448 infoPtr->clrBk = clrBk;
449 TRACE("[%p] %#lx\n", infoPtr->hwndSelf, infoPtr->clrBk);
451 /* the native control seems to do things this way */
452 SetWindowPos(infoPtr->hwndSelf, 0, 0, 0, 0, 0,
453 SWP_FRAMECHANGED | SWP_NOSIZE | SWP_NOMOVE |
454 SWP_NOZORDER | SWP_NOACTIVATE);
456 RedrawWindow(infoPtr->hwndSelf, 0, 0, RDW_ERASE | RDW_INVALIDATE);
458 return clrTemp;
462 static INT
463 PAGER_SetBorder (PAGER_INFO* infoPtr, INT iBorder)
465 INT nTemp = infoPtr->nBorder;
467 infoPtr->nBorder = iBorder;
468 TRACE("[%p] %d\n", infoPtr->hwndSelf, infoPtr->nBorder);
470 PAGER_RecalcSize(infoPtr);
472 return nTemp;
476 static INT
477 PAGER_SetButtonSize (PAGER_INFO* infoPtr, INT iButtonSize)
479 INT nTemp = infoPtr->nButtonSize;
481 infoPtr->nButtonSize = iButtonSize;
482 TRACE("[%p] %d\n", infoPtr->hwndSelf, infoPtr->nButtonSize);
484 PAGER_RecalcSize(infoPtr);
486 return nTemp;
490 static LRESULT
491 PAGER_SetChild (PAGER_INFO* infoPtr, HWND hwndChild)
493 infoPtr->hwndChild = IsWindow (hwndChild) ? hwndChild : 0;
495 if (infoPtr->hwndChild)
497 TRACE("[%p] hwndChild=%p\n", infoPtr->hwndSelf, infoPtr->hwndChild);
499 SetWindowPos(infoPtr->hwndSelf, 0, 0, 0, 0, 0,
500 SWP_FRAMECHANGED | SWP_NOMOVE | SWP_NOZORDER | SWP_NOSIZE | SWP_NOACTIVATE);
502 infoPtr->nPos = -1;
503 PAGER_SetPos(infoPtr, 0, FALSE, FALSE);
506 return 0;
509 static void
510 PAGER_Scroll(PAGER_INFO* infoPtr, INT dir)
512 NMPGSCROLL nmpgScroll;
513 RECT rcWnd;
515 if (infoPtr->hwndChild)
517 ZeroMemory (&nmpgScroll, sizeof (NMPGSCROLL));
518 nmpgScroll.hdr.hwndFrom = infoPtr->hwndSelf;
519 nmpgScroll.hdr.idFrom = GetWindowLongPtrW (infoPtr->hwndSelf, GWLP_ID);
520 nmpgScroll.hdr.code = PGN_SCROLL;
522 GetWindowRect(infoPtr->hwndSelf, &rcWnd);
523 GetClientRect(infoPtr->hwndSelf, &nmpgScroll.rcParent);
524 nmpgScroll.iXpos = nmpgScroll.iYpos = 0;
525 nmpgScroll.iDir = dir;
527 if (infoPtr->dwStyle & PGS_HORZ)
529 nmpgScroll.iScroll = rcWnd.right - rcWnd.left;
530 nmpgScroll.iXpos = infoPtr->nPos;
532 else
534 nmpgScroll.iScroll = rcWnd.bottom - rcWnd.top;
535 nmpgScroll.iYpos = infoPtr->nPos;
537 nmpgScroll.iScroll -= 2*infoPtr->nButtonSize;
539 SendMessageW (infoPtr->hwndNotify, WM_NOTIFY, nmpgScroll.hdr.idFrom, (LPARAM)&nmpgScroll);
541 TRACE("[%p] PGN_SCROLL returns iScroll=%d\n", infoPtr->hwndSelf, nmpgScroll.iScroll);
543 if (nmpgScroll.iScroll > 0)
545 infoPtr->direction = dir;
547 if (dir == PGF_SCROLLLEFT || dir == PGF_SCROLLUP)
548 PAGER_SetPos(infoPtr, infoPtr->nPos - nmpgScroll.iScroll, TRUE, TRUE);
549 else
550 PAGER_SetPos(infoPtr, infoPtr->nPos + nmpgScroll.iScroll, TRUE, TRUE);
552 else
553 infoPtr->direction = -1;
557 static LRESULT
558 PAGER_FmtLines(const PAGER_INFO *infoPtr)
560 /* initiate NCCalcSize to resize client wnd and get size */
561 SetWindowPos(infoPtr->hwndSelf, 0, 0, 0, 0, 0,
562 SWP_FRAMECHANGED | SWP_NOSIZE | SWP_NOMOVE |
563 SWP_NOZORDER | SWP_NOACTIVATE);
565 SetWindowPos(infoPtr->hwndChild, 0,
566 0,0,infoPtr->nWidth,infoPtr->nHeight,
569 return DefWindowProcW (infoPtr->hwndSelf, EM_FMTLINES, 0, 0);
572 static LRESULT
573 PAGER_Create (HWND hwnd, const CREATESTRUCTW *lpcs)
575 PAGER_INFO *infoPtr;
576 INT ret;
578 /* allocate memory for info structure */
579 infoPtr = Alloc(sizeof(*infoPtr));
580 if (!infoPtr) return -1;
581 SetWindowLongPtrW (hwnd, 0, (DWORD_PTR)infoPtr);
583 /* set default settings */
584 infoPtr->hwndSelf = hwnd;
585 infoPtr->hwndChild = NULL;
586 infoPtr->hwndNotify = lpcs->hwndParent;
587 infoPtr->dwStyle = lpcs->style;
588 infoPtr->clrBk = GetSysColor(COLOR_BTNFACE);
589 infoPtr->nBorder = 0;
590 infoPtr->nButtonSize = 12;
591 infoPtr->nPos = 0;
592 infoPtr->nWidth = 0;
593 infoPtr->nHeight = 0;
594 infoPtr->bForward = FALSE;
595 infoPtr->bCapture = FALSE;
596 infoPtr->TLbtnState = PGF_INVISIBLE;
597 infoPtr->BRbtnState = PGF_INVISIBLE;
598 infoPtr->direction = -1;
600 if (infoPtr->dwStyle & PGS_DRAGNDROP)
601 FIXME("[%p] Drag and Drop style is not implemented yet.\n", infoPtr->hwndSelf);
603 ret = SendMessageW(infoPtr->hwndNotify, WM_NOTIFYFORMAT, (WPARAM)infoPtr->hwndSelf, NF_QUERY);
604 infoPtr->bUnicode = (ret == NFR_UNICODE);
606 return 0;
610 static LRESULT
611 PAGER_Destroy (PAGER_INFO *infoPtr)
613 SetWindowLongPtrW (infoPtr->hwndSelf, 0, 0);
614 Free (infoPtr->pwszBuffer);
615 Free (infoPtr);
616 return 0;
619 static LRESULT
620 PAGER_NCCalcSize(PAGER_INFO* infoPtr, WPARAM wParam, LPRECT lpRect)
622 RECT rcChild, rcWindow;
625 * lpRect points to a RECT struct. On entry, the struct
626 * contains the proposed wnd rectangle for the window.
627 * On exit, the struct should contain the screen
628 * coordinates of the corresponding window's client area.
631 DefWindowProcW (infoPtr->hwndSelf, WM_NCCALCSIZE, wParam, (LPARAM)lpRect);
633 TRACE("orig rect=%s\n", wine_dbgstr_rect(lpRect));
635 GetWindowRect (infoPtr->hwndChild, &rcChild);
636 MapWindowPoints (0, infoPtr->hwndSelf, (LPPOINT)&rcChild, 2); /* FIXME: RECT != 2 POINTS */
637 GetWindowRect (infoPtr->hwndSelf, &rcWindow);
639 infoPtr->nWidth = lpRect->right - lpRect->left;
640 infoPtr->nHeight = lpRect->bottom - lpRect->top;
641 PAGER_CalcSize( infoPtr );
643 if (infoPtr->dwStyle & PGS_HORZ)
645 if (infoPtr->TLbtnState && (lpRect->left + infoPtr->nButtonSize < lpRect->right))
646 lpRect->left += infoPtr->nButtonSize;
647 if (infoPtr->BRbtnState && (lpRect->right - infoPtr->nButtonSize > lpRect->left))
648 lpRect->right -= infoPtr->nButtonSize;
650 else
652 if (infoPtr->TLbtnState && (lpRect->top + infoPtr->nButtonSize < lpRect->bottom))
653 lpRect->top += infoPtr->nButtonSize;
654 if (infoPtr->BRbtnState && (lpRect->bottom - infoPtr->nButtonSize > lpRect->top))
655 lpRect->bottom -= infoPtr->nButtonSize;
658 TRACE("nPos=%d, nHeight=%d, window=%s\n", infoPtr->nPos, infoPtr->nHeight, wine_dbgstr_rect(&rcWindow));
659 TRACE("[%p] client rect set to %s BtnState[%d,%d]\n", infoPtr->hwndSelf, wine_dbgstr_rect(lpRect),
660 infoPtr->TLbtnState, infoPtr->BRbtnState);
662 return 0;
665 static LRESULT
666 PAGER_NCPaint (const PAGER_INFO* infoPtr, HRGN hRgn)
668 RECT rcBottomRight, rcTopLeft;
669 HDC hdc;
671 if (infoPtr->dwStyle & WS_MINIMIZE)
672 return 0;
674 DefWindowProcW (infoPtr->hwndSelf, WM_NCPAINT, (WPARAM)hRgn, 0);
676 if (!(hdc = GetDCEx (infoPtr->hwndSelf, 0, DCX_USESTYLE | DCX_WINDOW)))
677 return 0;
679 PAGER_GetButtonRects(infoPtr, &rcTopLeft, &rcBottomRight, FALSE);
681 PAGER_DrawButton(hdc, infoPtr->clrBk, rcTopLeft,
682 infoPtr->dwStyle & PGS_HORZ, TRUE, infoPtr->TLbtnState);
683 PAGER_DrawButton(hdc, infoPtr->clrBk, rcBottomRight,
684 infoPtr->dwStyle & PGS_HORZ, FALSE, infoPtr->BRbtnState);
686 ReleaseDC( infoPtr->hwndSelf, hdc );
687 return 0;
690 static INT
691 PAGER_HitTest (const PAGER_INFO* infoPtr, const POINT * pt)
693 RECT clientRect, rcTopLeft, rcBottomRight;
694 POINT ptWindow;
696 GetClientRect (infoPtr->hwndSelf, &clientRect);
698 if (PtInRect(&clientRect, *pt))
700 TRACE("child\n");
701 return -1;
704 ptWindow = *pt;
705 PAGER_GetButtonRects(infoPtr, &rcTopLeft, &rcBottomRight, TRUE);
707 if ((infoPtr->TLbtnState != PGF_INVISIBLE) && PtInRect(&rcTopLeft, ptWindow))
709 TRACE("PGB_TOPORLEFT\n");
710 return PGB_TOPORLEFT;
712 else if ((infoPtr->BRbtnState != PGF_INVISIBLE) && PtInRect(&rcBottomRight, ptWindow))
714 TRACE("PGB_BOTTOMORRIGHT\n");
715 return PGB_BOTTOMORRIGHT;
718 TRACE("nowhere\n");
719 return -1;
722 static LRESULT
723 PAGER_NCHitTest (const PAGER_INFO* infoPtr, INT x, INT y)
725 POINT pt;
726 INT nHit;
728 pt.x = x;
729 pt.y = y;
731 ScreenToClient (infoPtr->hwndSelf, &pt);
732 nHit = PAGER_HitTest(infoPtr, &pt);
734 return (nHit < 0) ? HTTRANSPARENT : HTCLIENT;
737 static LRESULT
738 PAGER_MouseMove (PAGER_INFO* infoPtr, INT keys, INT x, INT y)
740 POINT clpt, pt;
741 RECT wnrect;
742 BOOL topLeft = FALSE;
743 INT btnstate = 0;
744 INT hit;
745 HDC hdc;
747 pt.x = x;
748 pt.y = y;
750 TRACE("[%p] to (%d,%d)\n", infoPtr->hwndSelf, x, y);
751 ClientToScreen(infoPtr->hwndSelf, &pt);
752 GetWindowRect(infoPtr->hwndSelf, &wnrect);
753 if (PtInRect(&wnrect, pt)) {
754 RECT topleft, bottomright, *rect = NULL;
756 PAGER_GetButtonRects(infoPtr, &topleft, &bottomright, FALSE);
758 clpt = pt;
759 MapWindowPoints(0, infoPtr->hwndSelf, &clpt, 1);
760 hit = PAGER_HitTest(infoPtr, &clpt);
761 if ((hit == PGB_TOPORLEFT) && (infoPtr->TLbtnState == PGF_NORMAL))
763 topLeft = TRUE;
764 rect = &topleft;
765 infoPtr->TLbtnState = PGF_HOT;
766 btnstate = infoPtr->TLbtnState;
768 else if ((hit == PGB_BOTTOMORRIGHT) && (infoPtr->BRbtnState == PGF_NORMAL))
770 topLeft = FALSE;
771 rect = &bottomright;
772 infoPtr->BRbtnState = PGF_HOT;
773 btnstate = infoPtr->BRbtnState;
776 /* If in one of the buttons the capture and draw buttons */
777 if (rect)
779 TRACE("[%p] draw btn (%s), Capture %s, style %#lx\n",
780 infoPtr->hwndSelf, wine_dbgstr_rect(rect),
781 (infoPtr->bCapture) ? "TRUE" : "FALSE",
782 infoPtr->dwStyle);
783 if (!infoPtr->bCapture)
785 TRACE("[%p] SetCapture\n", infoPtr->hwndSelf);
786 SetCapture(infoPtr->hwndSelf);
787 infoPtr->bCapture = TRUE;
789 if (infoPtr->dwStyle & PGS_AUTOSCROLL)
790 SetTimer(infoPtr->hwndSelf, TIMERID1, 0x3e, 0);
791 hdc = GetWindowDC(infoPtr->hwndSelf);
792 /* OffsetRect(wnrect, 0 | 1, 0 | 1) */
793 PAGER_DrawButton(hdc, infoPtr->clrBk, *rect,
794 infoPtr->dwStyle & PGS_HORZ, topLeft, btnstate);
795 ReleaseDC(infoPtr->hwndSelf, hdc);
796 return 0;
800 /* If we think we are captured, then do release */
801 if (infoPtr->bCapture && (WindowFromPoint(pt) != infoPtr->hwndSelf))
803 NMHDR nmhdr;
805 infoPtr->bCapture = FALSE;
807 if (GetCapture() == infoPtr->hwndSelf)
809 ReleaseCapture();
811 if (infoPtr->TLbtnState == PGF_GRAYED)
813 infoPtr->TLbtnState = PGF_INVISIBLE;
814 SetWindowPos(infoPtr->hwndSelf, 0, 0, 0, 0, 0,
815 SWP_FRAMECHANGED | SWP_NOSIZE | SWP_NOMOVE |
816 SWP_NOZORDER | SWP_NOACTIVATE);
818 else if (infoPtr->TLbtnState == PGF_HOT)
820 infoPtr->TLbtnState = PGF_NORMAL;
821 /* FIXME: just invalidate button rect */
822 RedrawWindow(infoPtr->hwndSelf, NULL, NULL, RDW_FRAME | RDW_INVALIDATE);
825 if (infoPtr->BRbtnState == PGF_GRAYED)
827 infoPtr->BRbtnState = PGF_INVISIBLE;
828 SetWindowPos(infoPtr->hwndSelf, 0, 0, 0, 0, 0,
829 SWP_FRAMECHANGED | SWP_NOSIZE | SWP_NOMOVE |
830 SWP_NOZORDER | SWP_NOACTIVATE);
832 else if (infoPtr->BRbtnState == PGF_HOT)
834 infoPtr->BRbtnState = PGF_NORMAL;
835 /* FIXME: just invalidate button rect */
836 RedrawWindow(infoPtr->hwndSelf, NULL, NULL, RDW_FRAME | RDW_INVALIDATE);
839 /* Notify parent of released mouse capture */
840 memset(&nmhdr, 0, sizeof(NMHDR));
841 nmhdr.hwndFrom = infoPtr->hwndSelf;
842 nmhdr.idFrom = GetWindowLongPtrW(infoPtr->hwndSelf, GWLP_ID);
843 nmhdr.code = NM_RELEASEDCAPTURE;
844 SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, nmhdr.idFrom, (LPARAM)&nmhdr);
846 if (IsWindow(infoPtr->hwndSelf))
847 KillTimer(infoPtr->hwndSelf, TIMERID1);
849 return 0;
852 static LRESULT
853 PAGER_LButtonDown (PAGER_INFO* infoPtr, INT keys, INT x, INT y)
855 BOOL repaintBtns = FALSE;
856 POINT pt;
857 INT hit;
859 pt.x = x;
860 pt.y = y;
862 TRACE("[%p] at (%d,%d)\n", infoPtr->hwndSelf, x, y);
864 hit = PAGER_HitTest(infoPtr, &pt);
866 /* put btn in DEPRESSED state */
867 if (hit == PGB_TOPORLEFT)
869 repaintBtns = infoPtr->TLbtnState != PGF_DEPRESSED;
870 infoPtr->TLbtnState = PGF_DEPRESSED;
871 SetTimer(infoPtr->hwndSelf, TIMERID1, INITIAL_DELAY, 0);
873 else if (hit == PGB_BOTTOMORRIGHT)
875 repaintBtns = infoPtr->BRbtnState != PGF_DEPRESSED;
876 infoPtr->BRbtnState = PGF_DEPRESSED;
877 SetTimer(infoPtr->hwndSelf, TIMERID1, INITIAL_DELAY, 0);
880 if (repaintBtns)
881 SendMessageW(infoPtr->hwndSelf, WM_NCPAINT, 0, 0);
883 switch(hit)
885 case PGB_TOPORLEFT:
886 if (infoPtr->dwStyle & PGS_HORZ)
888 TRACE("[%p] PGF_SCROLLLEFT\n", infoPtr->hwndSelf);
889 PAGER_Scroll(infoPtr, PGF_SCROLLLEFT);
891 else
893 TRACE("[%p] PGF_SCROLLUP\n", infoPtr->hwndSelf);
894 PAGER_Scroll(infoPtr, PGF_SCROLLUP);
896 break;
897 case PGB_BOTTOMORRIGHT:
898 if (infoPtr->dwStyle & PGS_HORZ)
900 TRACE("[%p] PGF_SCROLLRIGHT\n", infoPtr->hwndSelf);
901 PAGER_Scroll(infoPtr, PGF_SCROLLRIGHT);
903 else
905 TRACE("[%p] PGF_SCROLLDOWN\n", infoPtr->hwndSelf);
906 PAGER_Scroll(infoPtr, PGF_SCROLLDOWN);
908 break;
909 default:
910 break;
913 return 0;
916 static LRESULT
917 PAGER_LButtonUp (PAGER_INFO* infoPtr, INT keys, INT x, INT y)
919 TRACE("[%p]\n", infoPtr->hwndSelf);
921 KillTimer (infoPtr->hwndSelf, TIMERID1);
922 KillTimer (infoPtr->hwndSelf, TIMERID2);
924 /* make PRESSED btns NORMAL but don't hide gray btns */
925 if (infoPtr->TLbtnState & (PGF_HOT | PGF_DEPRESSED))
926 infoPtr->TLbtnState = PGF_NORMAL;
927 if (infoPtr->BRbtnState & (PGF_HOT | PGF_DEPRESSED))
928 infoPtr->BRbtnState = PGF_NORMAL;
930 return 0;
933 static LRESULT
934 PAGER_Timer (PAGER_INFO* infoPtr, INT nTimerId)
936 INT dir;
938 /* if initial timer, kill it and start the repeat timer */
939 if (nTimerId == TIMERID1) {
940 if (infoPtr->TLbtnState == PGF_HOT)
941 dir = (infoPtr->dwStyle & PGS_HORZ) ?
942 PGF_SCROLLLEFT : PGF_SCROLLUP;
943 else
944 dir = (infoPtr->dwStyle & PGS_HORZ) ?
945 PGF_SCROLLRIGHT : PGF_SCROLLDOWN;
946 TRACE("[%p] TIMERID1: style=%#lx, dir=%d\n",
947 infoPtr->hwndSelf, infoPtr->dwStyle, dir);
948 KillTimer(infoPtr->hwndSelf, TIMERID1);
949 SetTimer(infoPtr->hwndSelf, TIMERID1, REPEAT_DELAY, 0);
950 if (infoPtr->dwStyle & PGS_AUTOSCROLL) {
951 PAGER_Scroll(infoPtr, dir);
952 SetWindowPos(infoPtr->hwndSelf, 0, 0, 0, 0, 0,
953 SWP_FRAMECHANGED | SWP_NOSIZE | SWP_NOMOVE |
954 SWP_NOZORDER | SWP_NOACTIVATE);
956 return 0;
960 TRACE("[%p] TIMERID2: dir=%d\n", infoPtr->hwndSelf, infoPtr->direction);
961 KillTimer(infoPtr->hwndSelf, TIMERID2);
962 if (infoPtr->direction > 0) {
963 PAGER_Scroll(infoPtr, infoPtr->direction);
964 SetTimer(infoPtr->hwndSelf, TIMERID2, REPEAT_DELAY, 0);
966 return 0;
969 static LRESULT PAGER_ThemeChanged (const PAGER_INFO* infoPtr)
971 InvalidateRect(infoPtr->hwndSelf, NULL, TRUE);
972 return 0;
975 static LRESULT
976 PAGER_EraseBackground (const PAGER_INFO* infoPtr, HDC hdc)
978 POINT pt, ptorig;
979 HWND parent;
980 LRESULT ret;
982 pt.x = 0;
983 pt.y = 0;
984 parent = GetParent(infoPtr->hwndSelf);
985 MapWindowPoints(infoPtr->hwndSelf, parent, &pt, 1);
986 OffsetWindowOrgEx (hdc, pt.x, pt.y, &ptorig);
987 ret = SendMessageW (parent, WM_ERASEBKGND, (WPARAM)hdc, 0);
988 SetWindowOrgEx (hdc, ptorig.x, ptorig.y, 0);
990 return ret;
994 static LRESULT
995 PAGER_Size (PAGER_INFO* infoPtr, INT type, INT x, INT y)
997 /* note that WM_SIZE is sent whenever NCCalcSize resizes the client wnd */
999 TRACE("[%p] %d,%d\n", infoPtr->hwndSelf, x, y);
1001 if (infoPtr->dwStyle & PGS_HORZ)
1002 infoPtr->nHeight = y;
1003 else
1004 infoPtr->nWidth = x;
1006 return PAGER_RecalcSize(infoPtr);
1010 static LRESULT
1011 PAGER_StyleChanged(PAGER_INFO *infoPtr, WPARAM wStyleType, const STYLESTRUCT *lpss)
1013 DWORD oldStyle = infoPtr->dwStyle;
1015 TRACE("styletype %Ix, styleOld %#lx, styleNew %#lx\n", wStyleType, lpss->styleOld, lpss->styleNew);
1017 if (wStyleType != GWL_STYLE) return 0;
1019 infoPtr->dwStyle = lpss->styleNew;
1021 if ((oldStyle ^ lpss->styleNew) & (PGS_HORZ | PGS_VERT))
1023 PAGER_RecalcSize(infoPtr);
1026 return 0;
1029 static LRESULT PAGER_NotifyFormat(PAGER_INFO *infoPtr, INT command)
1031 INT ret;
1032 switch (command)
1034 case NF_REQUERY:
1035 ret = SendMessageW(infoPtr->hwndNotify, WM_NOTIFYFORMAT, (WPARAM)infoPtr->hwndSelf, NF_QUERY);
1036 infoPtr->bUnicode = (ret == NFR_UNICODE);
1037 return ret;
1038 case NF_QUERY:
1039 /* Pager always wants Unicode notifications from children */
1040 return NFR_UNICODE;
1041 default:
1042 return 0;
1046 static UINT PAGER_GetAnsiNtfCode(UINT code)
1048 switch (code)
1050 /* ComboxBoxEx */
1051 case CBEN_DRAGBEGINW: return CBEN_DRAGBEGINA;
1052 case CBEN_ENDEDITW: return CBEN_ENDEDITA;
1053 case CBEN_GETDISPINFOW: return CBEN_GETDISPINFOA;
1054 /* Date and Time Picker */
1055 case DTN_FORMATW: return DTN_FORMATA;
1056 case DTN_FORMATQUERYW: return DTN_FORMATQUERYA;
1057 case DTN_USERSTRINGW: return DTN_USERSTRINGA;
1058 case DTN_WMKEYDOWNW: return DTN_WMKEYDOWNA;
1059 /* Header */
1060 case HDN_BEGINTRACKW: return HDN_BEGINTRACKA;
1061 case HDN_DIVIDERDBLCLICKW: return HDN_DIVIDERDBLCLICKA;
1062 case HDN_ENDTRACKW: return HDN_ENDTRACKA;
1063 case HDN_GETDISPINFOW: return HDN_GETDISPINFOA;
1064 case HDN_ITEMCHANGEDW: return HDN_ITEMCHANGEDA;
1065 case HDN_ITEMCHANGINGW: return HDN_ITEMCHANGINGA;
1066 case HDN_ITEMCLICKW: return HDN_ITEMCLICKA;
1067 case HDN_ITEMDBLCLICKW: return HDN_ITEMDBLCLICKA;
1068 case HDN_TRACKW: return HDN_TRACKA;
1069 /* List View */
1070 case LVN_BEGINLABELEDITW: return LVN_BEGINLABELEDITA;
1071 case LVN_ENDLABELEDITW: return LVN_ENDLABELEDITA;
1072 case LVN_GETDISPINFOW: return LVN_GETDISPINFOA;
1073 case LVN_GETINFOTIPW: return LVN_GETINFOTIPA;
1074 case LVN_INCREMENTALSEARCHW: return LVN_INCREMENTALSEARCHA;
1075 case LVN_ODFINDITEMW: return LVN_ODFINDITEMA;
1076 case LVN_SETDISPINFOW: return LVN_SETDISPINFOA;
1077 /* Toolbar */
1078 case TBN_GETBUTTONINFOW: return TBN_GETBUTTONINFOA;
1079 case TBN_GETINFOTIPW: return TBN_GETINFOTIPA;
1080 /* Tooltip */
1081 case TTN_GETDISPINFOW: return TTN_GETDISPINFOA;
1082 /* Tree View */
1083 case TVN_BEGINDRAGW: return TVN_BEGINDRAGA;
1084 case TVN_BEGINLABELEDITW: return TVN_BEGINLABELEDITA;
1085 case TVN_BEGINRDRAGW: return TVN_BEGINRDRAGA;
1086 case TVN_DELETEITEMW: return TVN_DELETEITEMA;
1087 case TVN_ENDLABELEDITW: return TVN_ENDLABELEDITA;
1088 case TVN_GETDISPINFOW: return TVN_GETDISPINFOA;
1089 case TVN_GETINFOTIPW: return TVN_GETINFOTIPA;
1090 case TVN_ITEMEXPANDEDW: return TVN_ITEMEXPANDEDA;
1091 case TVN_ITEMEXPANDINGW: return TVN_ITEMEXPANDINGA;
1092 case TVN_SELCHANGEDW: return TVN_SELCHANGEDA;
1093 case TVN_SELCHANGINGW: return TVN_SELCHANGINGA;
1094 case TVN_SETDISPINFOW: return TVN_SETDISPINFOA;
1096 return code;
1099 static BOOL PAGER_AdjustBuffer(PAGER_INFO *infoPtr, INT size)
1101 if (!infoPtr->pwszBuffer)
1102 infoPtr->pwszBuffer = Alloc(size);
1103 else if (infoPtr->nBufferSize < size)
1104 infoPtr->pwszBuffer = ReAlloc(infoPtr->pwszBuffer, size);
1106 if (!infoPtr->pwszBuffer) return FALSE;
1107 if (infoPtr->nBufferSize < size) infoPtr->nBufferSize = size;
1109 return TRUE;
1112 /* Convert text to Unicode and return the original text address */
1113 static WCHAR *PAGER_ConvertText(WCHAR **text)
1115 WCHAR *oldText = *text;
1116 *text = NULL;
1117 Str_SetPtrWtoA((CHAR **)text, oldText);
1118 return oldText;
1121 static void PAGER_RestoreText(WCHAR **text, WCHAR *oldText)
1123 if (!oldText) return;
1125 Free(*text);
1126 *text = oldText;
1129 static LRESULT PAGER_SendConvertedNotify(PAGER_INFO *infoPtr, NMHDR *hdr, UINT *mask, UINT requiredMask, WCHAR **text,
1130 INT *textMax, DWORD flags)
1132 CHAR *sendBuffer = NULL;
1133 CHAR *receiveBuffer;
1134 INT bufferSize;
1135 WCHAR *oldText;
1136 INT oldTextMax;
1137 LRESULT ret = NO_ERROR;
1139 oldText = *text;
1140 oldTextMax = textMax ? *textMax : 0;
1142 hdr->code = PAGER_GetAnsiNtfCode(hdr->code);
1144 if (mask && !(*mask & requiredMask))
1146 ret = SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)hdr);
1147 if (flags & SET_NULL_IF_NO_MASK) oldText = NULL;
1148 goto done;
1151 if (oldTextMax < 0) goto done;
1153 if ((*text && flags & (CONVERT_SEND | ZERO_SEND)) || (!*text && flags & SEND_EMPTY_IF_NULL))
1155 bufferSize = textMax ? *textMax : lstrlenW(*text) + 1;
1156 sendBuffer = Alloc(bufferSize);
1157 if (!sendBuffer) goto done;
1158 if (!(flags & ZERO_SEND)) WideCharToMultiByte(CP_ACP, 0, *text, -1, sendBuffer, bufferSize, NULL, FALSE);
1159 *text = (WCHAR *)sendBuffer;
1162 ret = SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)hdr);
1164 if (*text && oldText && (flags & CONVERT_RECEIVE))
1166 /* MultiByteToWideChar requires that source and destination are not the same buffer */
1167 if (*text == oldText)
1169 bufferSize = lstrlenA((CHAR *)*text) + 1;
1170 receiveBuffer = Alloc(bufferSize);
1171 if (!receiveBuffer) goto done;
1172 memcpy(receiveBuffer, *text, bufferSize);
1173 MultiByteToWideChar(CP_ACP, 0, receiveBuffer, bufferSize, oldText, oldTextMax);
1174 Free(receiveBuffer);
1176 else
1177 MultiByteToWideChar(CP_ACP, 0, (CHAR *)*text, -1, oldText, oldTextMax);
1180 done:
1181 Free(sendBuffer);
1182 *text = oldText;
1183 return ret;
1186 static LRESULT PAGER_Notify(PAGER_INFO *infoPtr, NMHDR *hdr)
1188 LRESULT ret;
1190 if (infoPtr->bUnicode) return SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)hdr);
1192 switch (hdr->code)
1194 /* ComboBoxEx */
1195 case CBEN_GETDISPINFOW:
1197 NMCOMBOBOXEXW *nmcbe = (NMCOMBOBOXEXW *)hdr;
1198 return PAGER_SendConvertedNotify(infoPtr, hdr, &nmcbe->ceItem.mask, CBEIF_TEXT, &nmcbe->ceItem.pszText,
1199 &nmcbe->ceItem.cchTextMax, ZERO_SEND | SET_NULL_IF_NO_MASK | CONVERT_RECEIVE);
1201 case CBEN_DRAGBEGINW:
1203 NMCBEDRAGBEGINW *nmdbW = (NMCBEDRAGBEGINW *)hdr;
1204 NMCBEDRAGBEGINA nmdbA = {{0}};
1205 nmdbA.hdr.code = PAGER_GetAnsiNtfCode(nmdbW->hdr.code);
1206 nmdbA.hdr.hwndFrom = nmdbW->hdr.hwndFrom;
1207 nmdbA.hdr.idFrom = nmdbW->hdr.idFrom;
1208 nmdbA.iItemid = nmdbW->iItemid;
1209 WideCharToMultiByte(CP_ACP, 0, nmdbW->szText, ARRAY_SIZE(nmdbW->szText), nmdbA.szText, ARRAY_SIZE(nmdbA.szText),
1210 NULL, FALSE);
1211 return SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)&nmdbA);
1213 case CBEN_ENDEDITW:
1215 NMCBEENDEDITW *nmedW = (NMCBEENDEDITW *)hdr;
1216 NMCBEENDEDITA nmedA = {{0}};
1217 nmedA.hdr.code = PAGER_GetAnsiNtfCode(nmedW->hdr.code);
1218 nmedA.hdr.hwndFrom = nmedW->hdr.hwndFrom;
1219 nmedA.hdr.idFrom = nmedW->hdr.idFrom;
1220 nmedA.fChanged = nmedW->fChanged;
1221 nmedA.iNewSelection = nmedW->iNewSelection;
1222 nmedA.iWhy = nmedW->iWhy;
1223 WideCharToMultiByte(CP_ACP, 0, nmedW->szText, ARRAY_SIZE(nmedW->szText), nmedA.szText, ARRAY_SIZE(nmedA.szText),
1224 NULL, FALSE);
1225 return SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)&nmedA);
1227 /* Date and Time Picker */
1228 case DTN_FORMATW:
1230 NMDATETIMEFORMATW *nmdtf = (NMDATETIMEFORMATW *)hdr;
1231 WCHAR *oldFormat;
1232 INT textLength;
1234 hdr->code = PAGER_GetAnsiNtfCode(hdr->code);
1235 oldFormat = PAGER_ConvertText((WCHAR **)&nmdtf->pszFormat);
1236 ret = SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)nmdtf);
1237 PAGER_RestoreText((WCHAR **)&nmdtf->pszFormat, oldFormat);
1239 if (nmdtf->pszDisplay)
1241 textLength = MultiByteToWideChar(CP_ACP, 0, (LPCSTR)nmdtf->pszDisplay, -1, 0, 0);
1242 if (!PAGER_AdjustBuffer(infoPtr, textLength * sizeof(WCHAR))) return ret;
1243 MultiByteToWideChar(CP_ACP, 0, (LPCSTR)nmdtf->pszDisplay, -1, infoPtr->pwszBuffer, textLength);
1244 if (nmdtf->pszDisplay != nmdtf->szDisplay)
1245 nmdtf->pszDisplay = infoPtr->pwszBuffer;
1246 else
1248 textLength = min(textLength, ARRAY_SIZE(nmdtf->szDisplay));
1249 memcpy(nmdtf->szDisplay, infoPtr->pwszBuffer, textLength * sizeof(WCHAR));
1253 return ret;
1255 case DTN_FORMATQUERYW:
1257 NMDATETIMEFORMATQUERYW *nmdtfq = (NMDATETIMEFORMATQUERYW *)hdr;
1258 return PAGER_SendConvertedNotify(infoPtr, hdr, NULL, 0, (WCHAR **)&nmdtfq->pszFormat, NULL, CONVERT_SEND);
1260 case DTN_WMKEYDOWNW:
1262 NMDATETIMEWMKEYDOWNW *nmdtkd = (NMDATETIMEWMKEYDOWNW *)hdr;
1263 return PAGER_SendConvertedNotify(infoPtr, hdr, NULL, 0, (WCHAR **)&nmdtkd->pszFormat, NULL, CONVERT_SEND);
1265 case DTN_USERSTRINGW:
1267 NMDATETIMESTRINGW *nmdts = (NMDATETIMESTRINGW *)hdr;
1268 return PAGER_SendConvertedNotify(infoPtr, hdr, NULL, 0, (WCHAR **)&nmdts->pszUserString, NULL, CONVERT_SEND);
1270 /* Header */
1271 case HDN_BEGINTRACKW:
1272 case HDN_DIVIDERDBLCLICKW:
1273 case HDN_ENDTRACKW:
1274 case HDN_ITEMCHANGEDW:
1275 case HDN_ITEMCHANGINGW:
1276 case HDN_ITEMCLICKW:
1277 case HDN_ITEMDBLCLICKW:
1278 case HDN_TRACKW:
1280 NMHEADERW *nmh = (NMHEADERW *)hdr;
1281 WCHAR *oldText = NULL, *oldFilterText = NULL;
1282 HD_TEXTFILTERW *tf = NULL;
1284 hdr->code = PAGER_GetAnsiNtfCode(hdr->code);
1286 if (!nmh->pitem) return SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)hdr);
1287 if (nmh->pitem->mask & HDI_TEXT) oldText = PAGER_ConvertText(&nmh->pitem->pszText);
1288 if ((nmh->pitem->mask & HDI_FILTER) && (nmh->pitem->type == HDFT_ISSTRING) && nmh->pitem->pvFilter)
1290 tf = (HD_TEXTFILTERW *)nmh->pitem->pvFilter;
1291 oldFilterText = PAGER_ConvertText(&tf->pszText);
1293 ret = SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)hdr);
1294 PAGER_RestoreText(&nmh->pitem->pszText, oldText);
1295 if (tf) PAGER_RestoreText(&tf->pszText, oldFilterText);
1296 return ret;
1298 case HDN_GETDISPINFOW:
1300 NMHDDISPINFOW *nmhddi = (NMHDDISPINFOW *)hdr;
1301 return PAGER_SendConvertedNotify(infoPtr, hdr, &nmhddi->mask, HDI_TEXT, &nmhddi->pszText, &nmhddi->cchTextMax,
1302 SEND_EMPTY_IF_NULL | CONVERT_SEND | CONVERT_RECEIVE);
1304 /* List View */
1305 case LVN_BEGINLABELEDITW:
1306 case LVN_ENDLABELEDITW:
1307 case LVN_SETDISPINFOW:
1309 NMLVDISPINFOW *nmlvdi = (NMLVDISPINFOW *)hdr;
1310 return PAGER_SendConvertedNotify(infoPtr, hdr, &nmlvdi->item.mask, LVIF_TEXT, &nmlvdi->item.pszText,
1311 &nmlvdi->item.cchTextMax, SET_NULL_IF_NO_MASK | CONVERT_SEND | CONVERT_RECEIVE);
1313 case LVN_GETDISPINFOW:
1315 NMLVDISPINFOW *nmlvdi = (NMLVDISPINFOW *)hdr;
1316 return PAGER_SendConvertedNotify(infoPtr, hdr, &nmlvdi->item.mask, LVIF_TEXT, &nmlvdi->item.pszText,
1317 &nmlvdi->item.cchTextMax, CONVERT_RECEIVE);
1319 case LVN_GETINFOTIPW:
1321 NMLVGETINFOTIPW *nmlvgit = (NMLVGETINFOTIPW *)hdr;
1322 return PAGER_SendConvertedNotify(infoPtr, hdr, NULL, 0, &nmlvgit->pszText, &nmlvgit->cchTextMax,
1323 CONVERT_SEND | CONVERT_RECEIVE);
1325 case LVN_INCREMENTALSEARCHW:
1326 case LVN_ODFINDITEMW:
1328 NMLVFINDITEMW *nmlvfi = (NMLVFINDITEMW *)hdr;
1329 return PAGER_SendConvertedNotify(infoPtr, hdr, &nmlvfi->lvfi.flags, LVFI_STRING | LVFI_SUBSTRING,
1330 (WCHAR **)&nmlvfi->lvfi.psz, NULL, CONVERT_SEND);
1332 /* Toolbar */
1333 case TBN_GETBUTTONINFOW:
1335 NMTOOLBARW *nmtb = (NMTOOLBARW *)hdr;
1336 return PAGER_SendConvertedNotify(infoPtr, hdr, NULL, 0, &nmtb->pszText, &nmtb->cchText,
1337 SEND_EMPTY_IF_NULL | CONVERT_SEND | CONVERT_RECEIVE);
1339 case TBN_GETINFOTIPW:
1341 NMTBGETINFOTIPW *nmtbgit = (NMTBGETINFOTIPW *)hdr;
1342 return PAGER_SendConvertedNotify(infoPtr, hdr, NULL, 0, &nmtbgit->pszText, &nmtbgit->cchTextMax, CONVERT_RECEIVE);
1344 /* Tooltip */
1345 case TTN_GETDISPINFOW:
1347 NMTTDISPINFOW *nmttdiW = (NMTTDISPINFOW *)hdr;
1348 NMTTDISPINFOA nmttdiA = {{0}};
1349 INT size;
1351 nmttdiA.hdr.code = PAGER_GetAnsiNtfCode(nmttdiW->hdr.code);
1352 nmttdiA.hdr.hwndFrom = nmttdiW->hdr.hwndFrom;
1353 nmttdiA.hdr.idFrom = nmttdiW->hdr.idFrom;
1354 nmttdiA.hinst = nmttdiW->hinst;
1355 nmttdiA.uFlags = nmttdiW->uFlags;
1356 nmttdiA.lParam = nmttdiW->lParam;
1357 nmttdiA.lpszText = nmttdiA.szText;
1358 WideCharToMultiByte(CP_ACP, 0, nmttdiW->szText, ARRAY_SIZE(nmttdiW->szText), nmttdiA.szText,
1359 ARRAY_SIZE(nmttdiA.szText), NULL, FALSE);
1361 ret = SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)&nmttdiA);
1363 nmttdiW->hinst = nmttdiA.hinst;
1364 nmttdiW->uFlags = nmttdiA.uFlags;
1365 nmttdiW->lParam = nmttdiA.lParam;
1367 MultiByteToWideChar(CP_ACP, 0, nmttdiA.szText, ARRAY_SIZE(nmttdiA.szText), nmttdiW->szText,
1368 ARRAY_SIZE(nmttdiW->szText));
1369 if (!nmttdiA.lpszText)
1370 nmttdiW->lpszText = nmttdiW->szText;
1371 else if (!IS_INTRESOURCE(nmttdiA.lpszText))
1373 size = MultiByteToWideChar(CP_ACP, 0, nmttdiA.lpszText, -1, 0, 0);
1374 if (size > ARRAY_SIZE(nmttdiW->szText))
1376 if (!PAGER_AdjustBuffer(infoPtr, size * sizeof(WCHAR))) return ret;
1377 MultiByteToWideChar(CP_ACP, 0, nmttdiA.lpszText, -1, infoPtr->pwszBuffer, size);
1378 nmttdiW->lpszText = infoPtr->pwszBuffer;
1379 /* Override content in szText */
1380 memcpy(nmttdiW->szText, nmttdiW->lpszText, min(sizeof(nmttdiW->szText), size * sizeof(WCHAR)));
1382 else
1384 MultiByteToWideChar(CP_ACP, 0, nmttdiA.lpszText, -1, nmttdiW->szText, ARRAY_SIZE(nmttdiW->szText));
1385 nmttdiW->lpszText = nmttdiW->szText;
1388 else
1390 nmttdiW->szText[0] = 0;
1391 nmttdiW->lpszText = (WCHAR *)nmttdiA.lpszText;
1394 return ret;
1396 /* Tree View */
1397 case TVN_BEGINDRAGW:
1398 case TVN_BEGINRDRAGW:
1399 case TVN_ITEMEXPANDEDW:
1400 case TVN_ITEMEXPANDINGW:
1402 NMTREEVIEWW *nmtv = (NMTREEVIEWW *)hdr;
1403 return PAGER_SendConvertedNotify(infoPtr, hdr, &nmtv->itemNew.mask, TVIF_TEXT, &nmtv->itemNew.pszText, NULL,
1404 CONVERT_SEND);
1406 case TVN_DELETEITEMW:
1408 NMTREEVIEWW *nmtv = (NMTREEVIEWW *)hdr;
1409 return PAGER_SendConvertedNotify(infoPtr, hdr, &nmtv->itemOld.mask, TVIF_TEXT, &nmtv->itemOld.pszText, NULL,
1410 CONVERT_SEND);
1412 case TVN_BEGINLABELEDITW:
1413 case TVN_ENDLABELEDITW:
1415 NMTVDISPINFOW *nmtvdi = (NMTVDISPINFOW *)hdr;
1416 return PAGER_SendConvertedNotify(infoPtr, hdr, &nmtvdi->item.mask, TVIF_TEXT, &nmtvdi->item.pszText,
1417 &nmtvdi->item.cchTextMax, SET_NULL_IF_NO_MASK | CONVERT_SEND | CONVERT_RECEIVE);
1419 case TVN_SELCHANGINGW:
1420 case TVN_SELCHANGEDW:
1422 NMTREEVIEWW *nmtv = (NMTREEVIEWW *)hdr;
1423 WCHAR *oldItemOldText = NULL;
1424 WCHAR *oldItemNewText = NULL;
1426 hdr->code = PAGER_GetAnsiNtfCode(hdr->code);
1428 if (!((nmtv->itemNew.mask | nmtv->itemOld.mask) & TVIF_TEXT))
1429 return SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)hdr);
1431 if (nmtv->itemOld.mask & TVIF_TEXT) oldItemOldText = PAGER_ConvertText(&nmtv->itemOld.pszText);
1432 if (nmtv->itemNew.mask & TVIF_TEXT) oldItemNewText = PAGER_ConvertText(&nmtv->itemNew.pszText);
1434 ret = SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)hdr);
1435 PAGER_RestoreText(&nmtv->itemOld.pszText, oldItemOldText);
1436 PAGER_RestoreText(&nmtv->itemNew.pszText, oldItemNewText);
1437 return ret;
1439 case TVN_GETDISPINFOW:
1441 NMTVDISPINFOW *nmtvdi = (NMTVDISPINFOW *)hdr;
1442 return PAGER_SendConvertedNotify(infoPtr, hdr, &nmtvdi->item.mask, TVIF_TEXT, &nmtvdi->item.pszText,
1443 &nmtvdi->item.cchTextMax, ZERO_SEND | CONVERT_RECEIVE);
1445 case TVN_SETDISPINFOW:
1447 NMTVDISPINFOW *nmtvdi = (NMTVDISPINFOW *)hdr;
1448 return PAGER_SendConvertedNotify(infoPtr, hdr, &nmtvdi->item.mask, TVIF_TEXT, &nmtvdi->item.pszText,
1449 &nmtvdi->item.cchTextMax, SET_NULL_IF_NO_MASK | CONVERT_SEND | CONVERT_RECEIVE);
1451 case TVN_GETINFOTIPW:
1453 NMTVGETINFOTIPW *nmtvgit = (NMTVGETINFOTIPW *)hdr;
1454 return PAGER_SendConvertedNotify(infoPtr, hdr, NULL, 0, &nmtvgit->pszText, &nmtvgit->cchTextMax, CONVERT_RECEIVE);
1457 /* Other notifications, no need to convert */
1458 return SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, hdr->idFrom, (LPARAM)hdr);
1461 static LRESULT WINAPI
1462 PAGER_WindowProc (HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
1464 PAGER_INFO *infoPtr = (PAGER_INFO *)GetWindowLongPtrW(hwnd, 0);
1466 TRACE("%p, %#x, %#Ix, %#Ix\n", hwnd, uMsg, wParam, lParam);
1468 if (!infoPtr && (uMsg != WM_CREATE))
1469 return DefWindowProcW (hwnd, uMsg, wParam, lParam);
1471 switch (uMsg)
1473 case EM_FMTLINES:
1474 return PAGER_FmtLines(infoPtr);
1476 case PGM_FORWARDMOUSE:
1477 return PAGER_ForwardMouse (infoPtr, (BOOL)wParam);
1479 case PGM_GETBKCOLOR:
1480 return PAGER_GetBkColor(infoPtr);
1482 case PGM_GETBORDER:
1483 return PAGER_GetBorder(infoPtr);
1485 case PGM_GETBUTTONSIZE:
1486 return PAGER_GetButtonSize(infoPtr);
1488 case PGM_GETPOS:
1489 return PAGER_GetPos(infoPtr);
1491 case PGM_GETBUTTONSTATE:
1492 return PAGER_GetButtonState (infoPtr, (INT)lParam);
1494 /* case PGM_GETDROPTARGET: */
1496 case PGM_RECALCSIZE:
1497 return PAGER_RecalcSize(infoPtr);
1499 case PGM_SETBKCOLOR:
1500 return PAGER_SetBkColor (infoPtr, (COLORREF)lParam);
1502 case PGM_SETBORDER:
1503 return PAGER_SetBorder (infoPtr, (INT)lParam);
1505 case PGM_SETBUTTONSIZE:
1506 return PAGER_SetButtonSize (infoPtr, (INT)lParam);
1508 case PGM_SETCHILD:
1509 return PAGER_SetChild (infoPtr, (HWND)lParam);
1511 case PGM_SETPOS:
1512 return PAGER_SetPos(infoPtr, (INT)lParam, FALSE, TRUE);
1514 case WM_CREATE:
1515 return PAGER_Create (hwnd, (LPCREATESTRUCTW)lParam);
1517 case WM_DESTROY:
1518 return PAGER_Destroy (infoPtr);
1520 case WM_SIZE:
1521 return PAGER_Size (infoPtr, (INT)wParam, (short)LOWORD(lParam), (short)HIWORD(lParam));
1523 case WM_NCPAINT:
1524 return PAGER_NCPaint (infoPtr, (HRGN)wParam);
1526 case WM_STYLECHANGED:
1527 return PAGER_StyleChanged(infoPtr, wParam, (LPSTYLESTRUCT)lParam);
1529 case WM_NCCALCSIZE:
1530 return PAGER_NCCalcSize (infoPtr, wParam, (LPRECT)lParam);
1532 case WM_NCHITTEST:
1533 return PAGER_NCHitTest (infoPtr, (short)LOWORD(lParam), (short)HIWORD(lParam));
1535 case WM_MOUSEMOVE:
1536 if (infoPtr->bForward && infoPtr->hwndChild)
1537 PostMessageW(infoPtr->hwndChild, WM_MOUSEMOVE, wParam, lParam);
1538 return PAGER_MouseMove (infoPtr, (INT)wParam, (short)LOWORD(lParam), (short)HIWORD(lParam));
1540 case WM_LBUTTONDOWN:
1541 return PAGER_LButtonDown (infoPtr, (INT)wParam, (short)LOWORD(lParam), (short)HIWORD(lParam));
1543 case WM_LBUTTONUP:
1544 return PAGER_LButtonUp (infoPtr, (INT)wParam, (short)LOWORD(lParam), (short)HIWORD(lParam));
1546 case WM_ERASEBKGND:
1547 return PAGER_EraseBackground (infoPtr, (HDC)wParam);
1549 case WM_TIMER:
1550 return PAGER_Timer (infoPtr, (INT)wParam);
1552 case WM_NOTIFYFORMAT:
1553 return PAGER_NotifyFormat (infoPtr, lParam);
1555 case WM_NOTIFY:
1556 return PAGER_Notify (infoPtr, (NMHDR *)lParam);
1558 case WM_COMMAND:
1559 return SendMessageW (infoPtr->hwndNotify, uMsg, wParam, lParam);
1561 case WM_THEMECHANGED:
1562 return PAGER_ThemeChanged (infoPtr);
1564 default:
1565 return DefWindowProcW (hwnd, uMsg, wParam, lParam);
1570 VOID
1571 PAGER_Register (void)
1573 WNDCLASSW wndClass;
1575 ZeroMemory (&wndClass, sizeof(WNDCLASSW));
1576 wndClass.style = CS_GLOBALCLASS;
1577 wndClass.lpfnWndProc = PAGER_WindowProc;
1578 wndClass.cbClsExtra = 0;
1579 wndClass.cbWndExtra = sizeof(PAGER_INFO *);
1580 wndClass.hCursor = LoadCursorW (0, (LPWSTR)IDC_ARROW);
1581 wndClass.hbrBackground = (HBRUSH)(COLOR_BTNFACE+1);
1582 wndClass.lpszClassName = WC_PAGESCROLLERW;
1584 RegisterClassW (&wndClass);
1588 VOID
1589 PAGER_Unregister (void)
1591 UnregisterClassW (WC_PAGESCROLLERW, NULL);