| /* |
| * Trackbar control |
| * |
| * Copyright 1998, 1999 Eric Kohl <ekohl@abo.rhein-zeitung.de> |
| * Copyright 1998, 1999 Alex Priem <alexp@sci.kun.nl> |
| * Copyright 2002 Dimitrie O. Paun <dimi@bigfoot.com> |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| * |
| * NOTE |
| * |
| * This code was audited for completeness against the documented features |
| * of Comctl32.dll version 6.0 on Sep. 12, 2002, by Dimitrie O. Paun. |
| * |
| * Unless otherwise noted, we believe this code to be complete, as per |
| * the specification mentioned above. |
| * If you discover missing features, or bugs, please note them below. |
| * |
| */ |
| |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| #include "winbase.h" |
| #include "commctrl.h" |
| #include "wine/debug.h" |
| |
| #include "comctl32.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(trackbar); |
| |
| typedef struct |
| { |
| HWND hwndSelf; |
| LONG lRangeMin; |
| LONG lRangeMax; |
| LONG lLineSize; |
| LONG lPageSize; |
| LONG lSelMin; |
| LONG lSelMax; |
| LONG lPos; |
| UINT uThumbLen; |
| UINT uNumTics; |
| UINT uTicFreq; |
| HWND hwndNotify; |
| HWND hwndToolTip; |
| HWND hwndBuddyLA; |
| HWND hwndBuddyRB; |
| INT fLocation; |
| INT flags; |
| BOOL bUnicode; |
| RECT rcChannel; |
| RECT rcSelection; |
| RECT rcThumb; |
| LPLONG tics; |
| } TRACKBAR_INFO; |
| |
| DEFINE_COMMON_NOTIFICATIONS(TRACKBAR_INFO, hwndSelf); |
| |
| #define TB_REFRESH_TIMER 1 |
| #define TB_REFRESH_DELAY 500 |
| |
| #define TOOLTIP_OFFSET 2 /* distance from ctrl edge to tooltip */ |
| |
| /* Used by TRACKBAR_Refresh to find out which parts of the control |
| need to be recalculated */ |
| |
| #define TB_THUMBPOSCHANGED 1 |
| #define TB_THUMBSIZECHANGED 2 |
| #define TB_THUMBCHANGED (TB_THUMBPOSCHANGED | TB_THUMBSIZECHANGED) |
| #define TB_SELECTIONCHANGED 4 |
| #define TB_DRAG_MODE 8 /* we're dragging the slider */ |
| #define TB_AUTO_PAGE_LEFT 16 |
| #define TB_AUTO_PAGE_RIGHT 32 |
| #define TB_AUTO_PAGE (TB_AUTO_PAGE_LEFT | TB_AUTO_PAGE_RIGHT) |
| |
| /* helper defines for TRACKBAR_DrawTic */ |
| #define TIC_EDGE 0x20 |
| #define TIC_SELECTIONMARKMAX 0x80 |
| #define TIC_SELECTIONMARKMIN 0x100 |
| #define TIC_SELECTIONMARK (TIC_SELECTIONMARKMAX | TIC_SELECTIONMARKMIN) |
| |
| static BOOL TRACKBAR_SendNotify (TRACKBAR_INFO *infoPtr, UINT code); |
| |
| static inline int |
| notify_customdraw(NMCUSTOMDRAW *pnmcd, int stage) |
| { |
| pnmcd->dwDrawStage = stage; |
| return SendMessageW (GetParent(pnmcd->hdr.hwndFrom), WM_NOTIFY, |
| pnmcd->hdr.idFrom, (LPARAM)pnmcd); |
| } |
| |
| static void TRACKBAR_RecalculateTics (TRACKBAR_INFO *infoPtr) |
| { |
| int i, tic, nrTics; |
| |
| if (infoPtr->uTicFreq && infoPtr->lRangeMax >= infoPtr->lRangeMin) |
| nrTics=(infoPtr->lRangeMax - infoPtr->lRangeMin)/infoPtr->uTicFreq; |
| else { |
| nrTics = 0; |
| COMCTL32_Free (infoPtr->tics); |
| infoPtr->tics = NULL; |
| infoPtr->uNumTics = 0; |
| return; |
| } |
| |
| if (nrTics != infoPtr->uNumTics) { |
| infoPtr->tics=COMCTL32_ReAlloc (infoPtr->tics, |
| (nrTics+1)*sizeof (DWORD)); |
| if (!infoPtr->tics) { |
| infoPtr->uNumTics = 0; |
| notify_outofmemory(infoPtr); |
| return; |
| } |
| infoPtr->uNumTics = nrTics; |
| } |
| |
| tic = infoPtr->lRangeMin + infoPtr->uTicFreq; |
| for (i = 0; i < nrTics; i++, tic += infoPtr->uTicFreq) |
| infoPtr->tics[i] = tic; |
| } |
| |
| /* converts from physical (mouse) position to logical position |
| (in range of trackbar) */ |
| |
| static inline LONG |
| TRACKBAR_ConvertPlaceToPosition (TRACKBAR_INFO *infoPtr, int place, |
| int vertical) |
| { |
| double range, width, pos; |
| |
| range = infoPtr->lRangeMax - infoPtr->lRangeMin; |
| if (vertical) { |
| width = infoPtr->rcChannel.bottom - infoPtr->rcChannel.top; |
| pos = (range*(place - infoPtr->rcChannel.top)) / width; |
| } else { |
| width = infoPtr->rcChannel.right - infoPtr->rcChannel.left; |
| pos = (range*(place - infoPtr->rcChannel.left)) / width; |
| } |
| pos += infoPtr->lRangeMin; |
| if (pos > infoPtr->lRangeMax) |
| pos = infoPtr->lRangeMax; |
| else if (pos < infoPtr->lRangeMin) |
| pos = infoPtr->lRangeMin; |
| |
| TRACE("%.2f\n", pos); |
| return (LONG)(pos + 0.5); |
| } |
| |
| |
| /* return: 0> prev, 0 none, >0 next */ |
| static LONG |
| TRACKBAR_GetAutoPageDirection (TRACKBAR_INFO *infoPtr, POINT clickPoint) |
| { |
| DWORD dwStyle = GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE); |
| RECT pageRect; |
| |
| if (dwStyle & TBS_VERT) { |
| pageRect.top = infoPtr->rcChannel.top; |
| pageRect.bottom = infoPtr->rcChannel.bottom; |
| pageRect.left = infoPtr->rcThumb.left; |
| pageRect.right = infoPtr->rcThumb.right; |
| } else { |
| pageRect.top = infoPtr->rcThumb.top; |
| pageRect.bottom = infoPtr->rcThumb.bottom; |
| pageRect.left = infoPtr->rcChannel.left; |
| pageRect.right = infoPtr->rcChannel.right; |
| } |
| |
| |
| if (PtInRect(&pageRect, clickPoint)) |
| { |
| int clickPlace = (dwStyle & TBS_VERT) ? clickPoint.y : clickPoint.x; |
| |
| LONG clickPos = TRACKBAR_ConvertPlaceToPosition(infoPtr, clickPlace, |
| dwStyle & TBS_VERT); |
| return clickPos - infoPtr->lPos; |
| } |
| |
| return 0; |
| } |
| |
| static void inline |
| TRACKBAR_PageUp (TRACKBAR_INFO *infoPtr) |
| { |
| if (infoPtr->lPos == infoPtr->lRangeMax) return; |
| |
| infoPtr->lPos += infoPtr->lPageSize; |
| if (infoPtr->lPos > infoPtr->lRangeMax) |
| infoPtr->lPos = infoPtr->lRangeMax; |
| TRACKBAR_SendNotify (infoPtr, TB_PAGEUP); |
| } |
| |
| |
| static void inline |
| TRACKBAR_PageDown (TRACKBAR_INFO *infoPtr) |
| { |
| if (infoPtr->lPos == infoPtr->lRangeMin) return; |
| |
| infoPtr->lPos -= infoPtr->lPageSize; |
| if (infoPtr->lPos < infoPtr->lRangeMin) |
| infoPtr->lPos = infoPtr->lRangeMin; |
| TRACKBAR_SendNotify (infoPtr, TB_PAGEDOWN); |
| } |
| |
| static void |
| TRACKBAR_CalcChannel (TRACKBAR_INFO *infoPtr) |
| { |
| DWORD dwStyle = GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE); |
| INT cyChannel, offsetthumb, offsetedge; |
| RECT lpRect, *channel = & infoPtr->rcChannel; |
| |
| GetClientRect (infoPtr->hwndSelf, &lpRect); |
| |
| offsetthumb = (int)(infoPtr->uThumbLen/4.5); |
| offsetedge = offsetthumb + 3; |
| cyChannel = (dwStyle & TBS_ENABLESELRANGE) ? (offsetthumb+1)*3 : 4; |
| |
| if (dwStyle & TBS_VERT) { |
| channel->top = lpRect.top + offsetedge; |
| channel->bottom = lpRect.bottom - offsetedge; |
| channel->left = lpRect.left + offsetthumb; |
| if (dwStyle & (TBS_BOTH | TBS_LEFT)) channel->left += 8; |
| channel->right = channel->left + cyChannel; |
| } else { |
| channel->left = lpRect.left + offsetedge; |
| channel->right = lpRect.right - offsetedge; |
| channel->top = lpRect.top + offsetthumb; |
| if (dwStyle & (TBS_BOTH | TBS_TOP)) channel->top += 8; |
| channel->bottom = channel->top + cyChannel; |
| } |
| } |
| |
| static void |
| TRACKBAR_CalcThumb (TRACKBAR_INFO *infoPtr, LONG lPos, RECT *thumb) |
| { |
| int range, width, thumbdepth; |
| DWORD dwStyle = GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE); |
| |
| range=infoPtr->lRangeMax - infoPtr->lRangeMin; |
| thumbdepth = ((int)(infoPtr->uThumbLen / 4.5) * 2) + 2; |
| |
| if (!range) range = 1; |
| |
| if (dwStyle & TBS_VERT) |
| { |
| width=infoPtr->rcChannel.bottom - infoPtr->rcChannel.top; |
| |
| if (dwStyle & (TBS_BOTH | TBS_LEFT)) |
| thumb->left = 10; |
| else |
| thumb->left = 2; |
| thumb->right = thumb -> left + infoPtr->uThumbLen; |
| thumb->top = infoPtr->rcChannel.top + |
| (width*(lPos - infoPtr->lRangeMin))/range - |
| thumbdepth/2; |
| thumb->bottom = thumb->top + thumbdepth; |
| } |
| else |
| { |
| width=infoPtr->rcChannel.right - infoPtr->rcChannel.left; |
| |
| thumb->left = infoPtr->rcChannel.left + |
| (width*(lPos - infoPtr->lRangeMin))/range - |
| thumbdepth/2; |
| thumb->right = thumb->left + thumbdepth; |
| if (dwStyle & (TBS_BOTH | TBS_TOP)) |
| thumb->top = 10; |
| else |
| thumb->top = 2; |
| thumb->bottom = thumb->top + infoPtr->uThumbLen; |
| } |
| } |
| |
| inline static void |
| TRACKBAR_UpdateThumb (TRACKBAR_INFO *infoPtr) |
| { |
| TRACKBAR_CalcThumb(infoPtr, infoPtr->lPos, &infoPtr->rcThumb); |
| } |
| |
| static inline void |
| TRACKBAR_InvalidateAll(TRACKBAR_INFO * infoPtr) |
| { |
| InvalidateRect(infoPtr->hwndSelf, NULL, FALSE); |
| } |
| |
| static void |
| TRACKBAR_InvalidateThumb (TRACKBAR_INFO *infoPtr, LONG thumbPos) |
| { |
| RECT rcThumb; |
| |
| TRACKBAR_CalcThumb(infoPtr, thumbPos, &rcThumb); |
| InflateRect(&rcThumb, 1, 1); |
| InvalidateRect(infoPtr->hwndSelf, &rcThumb, FALSE); |
| } |
| |
| static inline void |
| TRACKBAR_InvalidateThumbMove (TRACKBAR_INFO *infoPtr, LONG oldPos, LONG newPos) |
| { |
| TRACKBAR_InvalidateThumb (infoPtr, oldPos); |
| if (newPos != oldPos) |
| TRACKBAR_InvalidateThumb (infoPtr, newPos); |
| } |
| |
| static BOOL inline |
| TRACKBAR_HasSelection (TRACKBAR_INFO *infoPtr) |
| { |
| return infoPtr->lSelMin != infoPtr->lSelMax; |
| } |
| |
| static void |
| TRACKBAR_CalcSelection (TRACKBAR_INFO *infoPtr) |
| { |
| RECT *selection = &infoPtr->rcSelection; |
| int range = infoPtr->lRangeMax - infoPtr->lRangeMin; |
| |
| if (range <= 0) { |
| SetRectEmpty (selection); |
| } else { |
| if (GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE) & TBS_VERT) { |
| int height = infoPtr->rcChannel.right - infoPtr->rcChannel.left; |
| selection->top = infoPtr->rcChannel.top + |
| (height*infoPtr->lSelMin)/range; |
| selection->bottom = infoPtr->rcChannel.top + |
| (height*infoPtr->lSelMax)/range; |
| selection->left = infoPtr->rcChannel.left + 3; |
| selection->right = infoPtr->rcChannel.right - 3; |
| } else { |
| int width = infoPtr->rcChannel.right - infoPtr->rcChannel.left; |
| selection->left = infoPtr->rcChannel.left + |
| (width*infoPtr->lSelMin)/range; |
| selection->right = infoPtr->rcChannel.left + |
| (width*infoPtr->lSelMax)/range; |
| selection->top = infoPtr->rcChannel.top + 3; |
| selection->bottom = infoPtr->rcChannel.bottom - 3; |
| } |
| } |
| |
| TRACE("selection[left=%d, top=%d, right=%d, bottom=%d]\n", |
| selection->left, selection->top, selection->right, selection->bottom); |
| } |
| |
| static BOOL |
| TRACKBAR_AutoPage (TRACKBAR_INFO *infoPtr, POINT clickPoint) |
| { |
| LONG dir = TRACKBAR_GetAutoPageDirection(infoPtr, clickPoint); |
| LONG prevPos = infoPtr->lPos; |
| |
| TRACE("x=%ld, y=%ld, dir=%ld\n", clickPoint.x, clickPoint.y, dir); |
| |
| if (dir > 0 && (infoPtr->flags & TB_AUTO_PAGE_RIGHT)) |
| TRACKBAR_PageUp(infoPtr); |
| else if (dir < 0 && (infoPtr->flags & TB_AUTO_PAGE_LEFT)) |
| TRACKBAR_PageDown(infoPtr); |
| else return FALSE; |
| |
| infoPtr->flags |= TB_THUMBPOSCHANGED; |
| TRACKBAR_InvalidateThumbMove (infoPtr, prevPos, infoPtr->lPos); |
| |
| return TRUE; |
| } |
| |
| /* Trackbar drawing code. I like my spaghetti done milanese. */ |
| |
| static void |
| TRACKBAR_DrawChannel (TRACKBAR_INFO *infoPtr, HDC hdc, DWORD dwStyle) |
| { |
| RECT rcChannel = infoPtr->rcChannel; |
| |
| DrawEdge (hdc, &rcChannel, EDGE_SUNKEN, BF_RECT | BF_ADJUST); |
| if (dwStyle & TBS_ENABLESELRANGE) { /* fill the channel */ |
| FillRect (hdc, &rcChannel, GetStockObject(WHITE_BRUSH)); |
| if (TRACKBAR_HasSelection(infoPtr)) |
| FillRect (hdc, &infoPtr->rcSelection, GetSysColorBrush(COLOR_HIGHLIGHT)); |
| } |
| } |
| |
| static void |
| TRACKBAR_DrawOneTic (TRACKBAR_INFO *infoPtr, HDC hdc, LONG ticPos, int flags) |
| { |
| int x, y, ox, oy, range, side, offset = 5, indent = 0, len = 3; |
| RECT rcTics; |
| |
| TRACE("\n"); |
| |
| GetClientRect(infoPtr->hwndSelf, &rcTics); |
| if (flags & TBS_VERT) { |
| rcTics.top = infoPtr->rcChannel.top; |
| rcTics.bottom = infoPtr->rcChannel.bottom; |
| } else { |
| rcTics.left = infoPtr->rcChannel.left; |
| rcTics.right = infoPtr->rcChannel.right; |
| } |
| |
| if (flags & (TBS_TOP | TBS_LEFT)) { |
| x = rcTics.left; |
| y = rcTics.top; |
| side = -1; |
| } else { |
| x = rcTics.right; |
| y = rcTics.bottom; |
| side = 1; |
| } |
| |
| range = infoPtr->lRangeMax - infoPtr->lRangeMin; |
| if (range == 0) |
| range = 1; /* to avoid division by zero */ |
| |
| if (flags & TIC_SELECTIONMARK) { |
| indent = (flags & TIC_SELECTIONMARKMIN) ? -1 : 1; |
| } else if (flags & TIC_EDGE) { |
| len++; |
| } |
| |
| if (flags & TBS_VERT) { |
| int height = rcTics.bottom = rcTics.top; |
| y = rcTics.top + (height*(ticPos - infoPtr->lRangeMin))/range; |
| x -= (offset + 2) * side; |
| y += indent; |
| } else { |
| int width = rcTics.right - rcTics.left; |
| x = rcTics.left + (width*(ticPos - infoPtr->lRangeMin))/range; |
| x += indent; |
| y -= (offset + 2) * side; |
| } |
| |
| ox = x; |
| oy = y; |
| MoveToEx(hdc, x, y, 0); |
| if (flags & TBS_VERT) x += len * side; |
| else y += len * side; |
| LineTo(hdc, x, y); |
| |
| if (flags & TIC_SELECTIONMARK) { |
| if (flags & TBS_VERT) { |
| x -= side; |
| } else { |
| y -= side; |
| } |
| MoveToEx(hdc, x, y, 0); |
| if (flags & TBS_VERT) { |
| y += 2 * indent; |
| } else { |
| x += 2 * indent; |
| } |
| |
| LineTo(hdc, x, y); |
| LineTo(hdc, ox, oy); |
| } |
| } |
| |
| |
| static inline void |
| TRACKBAR_DrawTic (TRACKBAR_INFO *infoPtr, HDC hdc, LONG ticPos, int flags) |
| { |
| if ((flags & (TBS_LEFT | TBS_TOP)) || (flags & TBS_BOTH)) |
| TRACKBAR_DrawOneTic (infoPtr, hdc, ticPos, flags | TBS_LEFT); |
| |
| if (!(flags & (TBS_LEFT | TBS_TOP)) || (flags & TBS_BOTH)) |
| TRACKBAR_DrawOneTic (infoPtr, hdc, ticPos, flags); |
| } |
| |
| static void |
| TRACKBAR_DrawTics (TRACKBAR_INFO *infoPtr, HDC hdc, DWORD dwStyle) |
| { |
| int i, ticFlags = dwStyle & 0x0f; |
| LOGPEN ticPen = { PS_SOLID, {1, 0}, GetSysColor (COLOR_3DDKSHADOW) }; |
| HPEN hOldPen, hTicPen; |
| |
| /* create the pen to draw the tics with */ |
| hTicPen = CreatePenIndirect(&ticPen); |
| hOldPen = hTicPen ? SelectObject(hdc, hTicPen) : 0; |
| |
| /* actually draw the tics */ |
| for (i=0; i<infoPtr->uNumTics; i++) |
| TRACKBAR_DrawTic (infoPtr, hdc, infoPtr->tics[i], ticFlags); |
| |
| TRACKBAR_DrawTic (infoPtr, hdc, infoPtr->lRangeMin, ticFlags | TIC_EDGE); |
| TRACKBAR_DrawTic (infoPtr, hdc, infoPtr->lRangeMax, ticFlags | TIC_EDGE); |
| |
| if ((dwStyle & TBS_ENABLESELRANGE) && TRACKBAR_HasSelection(infoPtr)) { |
| TRACKBAR_DrawTic (infoPtr, hdc, infoPtr->lSelMin, |
| ticFlags | TIC_SELECTIONMARKMIN); |
| TRACKBAR_DrawTic (infoPtr, hdc, infoPtr->lSelMax, |
| ticFlags | TIC_SELECTIONMARKMAX); |
| } |
| |
| /* clean up the pen, if we created one */ |
| if (hTicPen) { |
| SelectObject(hdc, hOldPen); |
| DeleteObject(hTicPen); |
| } |
| } |
| |
| static void |
| TRACKBAR_DrawThumb(TRACKBAR_INFO *infoPtr, HDC hdc, DWORD dwStyle) |
| { |
| HBRUSH oldbr; |
| HPEN oldpen; |
| RECT thumb = infoPtr->rcThumb; |
| int BlackUntil = 3; |
| int PointCount = 6; |
| POINT points[6]; |
| int fillClr; |
| |
| static INT PointDepth = 4; |
| |
| fillClr = infoPtr->flags & TB_DRAG_MODE ? COLOR_BTNHILIGHT : COLOR_BTNFACE; |
| oldbr = SelectObject (hdc, GetSysColorBrush(fillClr)); |
| SetPolyFillMode (hdc, WINDING); |
| |
| if (dwStyle & TBS_BOTH) |
| { |
| points[0].x=thumb.right; |
| points[0].y=thumb.top; |
| points[1].x=thumb.right; |
| points[1].y=thumb.bottom; |
| points[2].x=thumb.left; |
| points[2].y=thumb.bottom; |
| points[3].x=thumb.left; |
| points[3].y=thumb.top; |
| points[4].x=points[0].x; |
| points[4].y=points[0].y; |
| PointCount = 5; |
| BlackUntil = 3; |
| } |
| else |
| { |
| if (dwStyle & TBS_VERT) |
| { |
| if (dwStyle & TBS_LEFT) |
| { |
| points[0].x=thumb.right; |
| points[0].y=thumb.top; |
| points[1].x=thumb.right; |
| points[1].y=thumb.bottom; |
| points[2].x=thumb.left + PointDepth; |
| points[2].y=thumb.bottom; |
| points[3].x=thumb.left; |
| points[3].y=(thumb.bottom - thumb.top) / 2 + thumb.top; |
| points[4].x=thumb.left + PointDepth; |
| points[4].y=thumb.top; |
| points[5].x=points[0].x; |
| points[5].y=points[0].y; |
| BlackUntil = 4; |
| } |
| else |
| { |
| points[0].x=thumb.right; |
| points[0].y=(thumb.bottom - thumb.top) / 2 + thumb.top; |
| points[1].x=thumb.right - PointDepth; |
| points[1].y=thumb.bottom; |
| points[2].x=thumb.left; |
| points[2].y=thumb.bottom; |
| points[3].x=thumb.left; |
| points[3].y=thumb.top; |
| points[4].x=thumb.right - PointDepth; |
| points[4].y=thumb.top; |
| points[5].x=points[0].x; |
| points[5].y=points[0].y; |
| } |
| } |
| else |
| { |
| if (dwStyle & TBS_TOP) |
| { |
| points[0].x=(thumb.right - thumb.left) / 2 + thumb.left ; |
| points[0].y=thumb.top; |
| points[1].x=thumb.right; |
| points[1].y=thumb.top + PointDepth; |
| points[2].x=thumb.right; |
| points[2].y=thumb.bottom; |
| points[3].x=thumb.left; |
| points[3].y=thumb.bottom; |
| points[4].x=thumb.left; |
| points[4].y=thumb.top + PointDepth; |
| points[5].x=points[0].x; |
| points[5].y=points[0].y; |
| BlackUntil = 4; |
| } |
| else |
| { |
| points[0].x=thumb.right; |
| points[0].y=thumb.top; |
| points[1].x=thumb.right; |
| points[1].y=thumb.bottom - PointDepth; |
| points[2].x=(thumb.right - thumb.left) / 2 + thumb.left ; |
| points[2].y=thumb.bottom; |
| points[3].x=thumb.left; |
| points[3].y=thumb.bottom - PointDepth; |
| points[4].x=thumb.left; |
| points[4].y=thumb.top; |
| points[5].x=points[0].x; |
| points[5].y=points[0].y; |
| } |
| } |
| |
| } |
| |
| /* Draw the thumb now */ |
| Polygon (hdc, points, PointCount); |
| oldpen = SelectObject(hdc, GetStockObject(BLACK_PEN)); |
| Polyline(hdc,points, BlackUntil); |
| SelectObject(hdc, GetStockObject(WHITE_PEN)); |
| Polyline(hdc, &points[BlackUntil-1], PointCount+1-BlackUntil); |
| SelectObject(hdc, oldpen); |
| SelectObject(hdc, oldbr); |
| } |
| |
| |
| static void inline |
| TRACKBAR_ActivateToolTip (TRACKBAR_INFO *infoPtr, BOOL fShow) |
| { |
| TTTOOLINFOW ti; |
| |
| if (!infoPtr->hwndToolTip) return; |
| |
| ZeroMemory(&ti, sizeof(ti)); |
| ti.cbSize = sizeof(ti); |
| ti.hwnd = infoPtr->hwndSelf; |
| |
| SendMessageW (infoPtr->hwndToolTip, TTM_TRACKACTIVATE, fShow, (LPARAM)&ti); |
| } |
| |
| |
| static void |
| TRACKBAR_UpdateToolTip (TRACKBAR_INFO *infoPtr) |
| { |
| DWORD dwStyle = GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE); |
| WCHAR buf[80], fmt[] = { '%', 'l', 'd', 0 }; |
| TTTOOLINFOW ti; |
| POINT pt; |
| RECT rcClient; |
| LRESULT size; |
| |
| if (!infoPtr->hwndToolTip) return; |
| |
| ZeroMemory(&ti, sizeof(ti)); |
| ti.cbSize = sizeof(ti); |
| ti.hwnd = infoPtr->hwndSelf; |
| ti.uFlags = TTF_IDISHWND | TTF_TRACK | TTF_ABSOLUTE; |
| |
| wsprintfW (buf, fmt, infoPtr->lPos); |
| ti.lpszText = buf; |
| SendMessageW (infoPtr->hwndToolTip, TTM_UPDATETIPTEXTW, 0, (LPARAM)&ti); |
| |
| GetClientRect (infoPtr->hwndSelf, &rcClient); |
| size = SendMessageW (infoPtr->hwndToolTip, TTM_GETBUBBLESIZE, 0, (LPARAM)&ti); |
| if (dwStyle & TBS_VERT) { |
| if (infoPtr->fLocation == TBTS_LEFT) |
| pt.x = 0 - LOWORD(size) - TOOLTIP_OFFSET; |
| else |
| pt.x = rcClient.right + TOOLTIP_OFFSET; |
| pt.y = (infoPtr->rcThumb.top + infoPtr->rcThumb.bottom - HIWORD(size))/2; |
| } else { |
| if (infoPtr->fLocation == TBTS_TOP) |
| pt.y = 0 - HIWORD(size) - TOOLTIP_OFFSET; |
| else |
| pt.y = rcClient.bottom + TOOLTIP_OFFSET; |
| pt.x = (infoPtr->rcThumb.left + infoPtr->rcThumb.right - LOWORD(size))/2; |
| } |
| ClientToScreen(infoPtr->hwndSelf, &pt); |
| |
| SendMessageW (infoPtr->hwndToolTip, TTM_TRACKPOSITION, |
| 0, (LPARAM)MAKELPARAM(pt.x, pt.y)); |
| } |
| |
| |
| static void |
| TRACKBAR_Refresh (TRACKBAR_INFO *infoPtr, HDC hdcDst) |
| { |
| DWORD dwStyle = GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE); |
| RECT rcClient; |
| HDC hdc; |
| HBITMAP hOldBmp = 0, hOffScreenBmp = 0; |
| NMCUSTOMDRAW nmcd; |
| int gcdrf, icdrf; |
| |
| if (infoPtr->flags & TB_THUMBCHANGED) { |
| TRACKBAR_UpdateThumb (infoPtr); |
| if (infoPtr->flags & TB_THUMBSIZECHANGED) |
| TRACKBAR_CalcChannel (infoPtr); |
| } |
| if (infoPtr->flags & TB_SELECTIONCHANGED) |
| TRACKBAR_CalcSelection (infoPtr); |
| |
| if (infoPtr->flags & TB_DRAG_MODE) |
| TRACKBAR_UpdateToolTip (infoPtr); |
| |
| infoPtr->flags &= ~ (TB_THUMBCHANGED | TB_SELECTIONCHANGED); |
| |
| GetClientRect (infoPtr->hwndSelf, &rcClient); |
| |
| /* try to render offscreen, if we fail, carrry onscreen */ |
| hdc = CreateCompatibleDC(hdcDst); |
| if (hdc) { |
| hOffScreenBmp = CreateCompatibleBitmap(hdcDst, rcClient.right, rcClient.bottom); |
| if (hOffScreenBmp) { |
| hOldBmp = SelectObject(hdc, hOffScreenBmp); |
| } else { |
| DeleteObject(hdc); |
| hdc = hdcDst; |
| } |
| } else { |
| hdc = hdcDst; |
| } |
| |
| ZeroMemory(&nmcd, sizeof(nmcd)); |
| nmcd.hdr.hwndFrom = infoPtr->hwndSelf; |
| nmcd.hdr.idFrom = GetWindowLongW (infoPtr->hwndSelf, GWL_ID); |
| nmcd.hdr.code = NM_CUSTOMDRAW; |
| nmcd.hdc = hdc; |
| |
| /* start the paint cycle */ |
| nmcd.rc = rcClient; |
| gcdrf = notify_customdraw(&nmcd, CDDS_PREPAINT); |
| if (gcdrf & CDRF_SKIPDEFAULT) goto cleanup; |
| |
| /* Erase backbround */ |
| if (gcdrf == CDRF_DODEFAULT || |
| notify_customdraw(&nmcd, CDDS_PREERASE) != CDRF_SKIPDEFAULT) { |
| FillRect (hdc, &rcClient, GetSysColorBrush(COLOR_BTNFACE)); |
| if (gcdrf != CDRF_DODEFAULT) |
| notify_customdraw(&nmcd, CDDS_POSTERASE); |
| } |
| |
| /* draw channel */ |
| if (gcdrf & CDRF_NOTIFYITEMDRAW) { |
| nmcd.dwItemSpec = TBCD_CHANNEL; |
| nmcd.uItemState = CDIS_DEFAULT; |
| nmcd.rc = infoPtr->rcChannel; |
| icdrf = notify_customdraw(&nmcd, CDDS_ITEMPREPAINT); |
| } else icdrf = CDRF_DODEFAULT; |
| if ( !(icdrf & CDRF_SKIPDEFAULT) ) { |
| TRACKBAR_DrawChannel (infoPtr, hdc, dwStyle); |
| if (icdrf & CDRF_NOTIFYPOSTPAINT) |
| notify_customdraw(&nmcd, CDDS_ITEMPOSTPAINT); |
| } |
| |
| |
| /* draw tics */ |
| if (!(dwStyle & TBS_NOTICKS)) { |
| if (gcdrf & CDRF_NOTIFYITEMDRAW) { |
| nmcd.dwItemSpec = TBCD_TICS; |
| nmcd.uItemState = CDIS_DEFAULT; |
| nmcd.rc = rcClient; |
| icdrf = notify_customdraw(&nmcd, CDDS_ITEMPREPAINT); |
| } else icdrf = CDRF_DODEFAULT; |
| if ( !(icdrf & CDRF_SKIPDEFAULT) ) { |
| TRACKBAR_DrawTics (infoPtr, hdc, dwStyle); |
| if (icdrf & CDRF_NOTIFYPOSTPAINT) |
| notify_customdraw(&nmcd, CDDS_ITEMPOSTPAINT); |
| } |
| } |
| |
| /* draw thumb */ |
| if (!(dwStyle & TBS_NOTHUMB)) { |
| if (gcdrf & CDRF_NOTIFYITEMDRAW) { |
| nmcd.dwItemSpec = TBCD_THUMB; |
| nmcd.uItemState = infoPtr->flags & TB_DRAG_MODE ? CDIS_HOT : CDIS_DEFAULT; |
| nmcd.rc = infoPtr->rcThumb; |
| icdrf = notify_customdraw(&nmcd, CDDS_ITEMPREPAINT); |
| } else icdrf = CDRF_DODEFAULT; |
| if ( !(icdrf & CDRF_SKIPDEFAULT) ) { |
| TRACKBAR_DrawThumb(infoPtr, hdc, dwStyle); |
| if (icdrf & CDRF_NOTIFYPOSTPAINT) |
| notify_customdraw(&nmcd, CDDS_ITEMPOSTPAINT); |
| } |
| } |
| |
| /* finish up the painting */ |
| if (gcdrf & CDRF_NOTIFYPOSTPAINT) |
| notify_customdraw(&nmcd, CDDS_POSTPAINT); |
| |
| cleanup: |
| /* cleanup, if we rendered offscreen */ |
| if (hdc != hdcDst) { |
| BitBlt(hdcDst, 0, 0, rcClient.right, rcClient.bottom, hdc, 0, 0, SRCCOPY); |
| SelectObject(hdc, hOldBmp); |
| DeleteObject(hOffScreenBmp); |
| DeleteObject(hdc); |
| } |
| } |
| |
| |
| static void |
| TRACKBAR_AlignBuddies (TRACKBAR_INFO *infoPtr) |
| { |
| DWORD dwStyle = GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE); |
| HWND hwndParent = GetParent (infoPtr->hwndSelf); |
| RECT rcSelf, rcBuddy; |
| INT x, y; |
| |
| GetWindowRect (infoPtr->hwndSelf, &rcSelf); |
| MapWindowPoints (HWND_DESKTOP, hwndParent, (LPPOINT)&rcSelf, 2); |
| |
| /* align buddy left or above */ |
| if (infoPtr->hwndBuddyLA) { |
| GetWindowRect (infoPtr->hwndBuddyLA, &rcBuddy); |
| MapWindowPoints (HWND_DESKTOP, hwndParent, (LPPOINT)&rcBuddy, 2); |
| |
| if (dwStyle & TBS_VERT) { |
| x = (infoPtr->rcChannel.right + infoPtr->rcChannel.left) / 2 - |
| (rcBuddy.right - rcBuddy.left) / 2 + rcSelf.left; |
| y = rcSelf.top - (rcBuddy.bottom - rcBuddy.top); |
| } |
| else { |
| x = rcSelf.left - (rcBuddy.right - rcBuddy.left); |
| y = (infoPtr->rcChannel.bottom + infoPtr->rcChannel.top) / 2 - |
| (rcBuddy.bottom - rcBuddy.top) / 2 + rcSelf.top; |
| } |
| |
| SetWindowPos (infoPtr->hwndBuddyLA, 0, x, y, 0, 0, |
| SWP_NOZORDER | SWP_NOSIZE); |
| } |
| |
| |
| /* align buddy right or below */ |
| if (infoPtr->hwndBuddyRB) { |
| GetWindowRect (infoPtr->hwndBuddyRB, &rcBuddy); |
| MapWindowPoints (HWND_DESKTOP, hwndParent, (LPPOINT)&rcBuddy, 2); |
| |
| if (dwStyle & TBS_VERT) { |
| x = (infoPtr->rcChannel.right + infoPtr->rcChannel.left) / 2 - |
| (rcBuddy.right - rcBuddy.left) / 2 + rcSelf.left; |
| y = rcSelf.bottom; |
| } |
| else { |
| x = rcSelf.right; |
| y = (infoPtr->rcChannel.bottom + infoPtr->rcChannel.top) / 2 - |
| (rcBuddy.bottom - rcBuddy.top) / 2 + rcSelf.top; |
| } |
| SetWindowPos (infoPtr->hwndBuddyRB, 0, x, y, 0, 0, |
| SWP_NOZORDER | SWP_NOSIZE); |
| } |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_ClearSel (TRACKBAR_INFO *infoPtr, BOOL fRedraw) |
| { |
| infoPtr->lSelMin = 0; |
| infoPtr->lSelMax = 0; |
| infoPtr->flags |= TB_SELECTIONCHANGED; |
| |
| if (fRedraw) TRACKBAR_InvalidateAll(infoPtr); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_ClearTics (TRACKBAR_INFO *infoPtr, BOOL fRedraw) |
| { |
| if (infoPtr->tics) { |
| COMCTL32_Free (infoPtr->tics); |
| infoPtr->tics = NULL; |
| infoPtr->uNumTics = 0; |
| } |
| |
| if (fRedraw) TRACKBAR_InvalidateAll(infoPtr); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_GetChannelRect (TRACKBAR_INFO *infoPtr, LPRECT lprc) |
| { |
| if (lprc == NULL) return 0; |
| |
| lprc->left = infoPtr->rcChannel.left; |
| lprc->right = infoPtr->rcChannel.right; |
| lprc->bottom = infoPtr->rcChannel.bottom; |
| lprc->top = infoPtr->rcChannel.top; |
| |
| return 0; |
| } |
| |
| |
| static LONG inline |
| TRACKBAR_GetNumTics (TRACKBAR_INFO *infoPtr) |
| { |
| if (GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE) & TBS_NOTICKS) |
| return 0; |
| |
| return infoPtr->uNumTics + 2; |
| } |
| |
| |
| static int comp_tics(const void *ap, const void *bp) |
| { |
| DWORD a = *((DWORD *)ap); |
| DWORD b = *((DWORD *)bp); |
| |
| TRACE("(a=%ld, b=%ld)\n", a, b); |
| if (a < b) return -1; |
| if (a > b) return 1; |
| return 0; |
| } |
| |
| |
| static LONG inline |
| TRACKBAR_GetTic (TRACKBAR_INFO *infoPtr, INT iTic) |
| { |
| if ((iTic < 0) || (iTic >= infoPtr->uNumTics) || !infoPtr->tics) |
| return -1; |
| |
| qsort(infoPtr->tics, infoPtr->uNumTics, sizeof(DWORD), comp_tics); |
| return infoPtr->tics[iTic]; |
| } |
| |
| |
| static LONG inline |
| TRACKBAR_GetTicPos (TRACKBAR_INFO *infoPtr, INT iTic) |
| { |
| LONG range, width, pos, tic; |
| |
| if ((iTic < 0) || (iTic >= infoPtr->uNumTics) || !infoPtr->tics) |
| return -1; |
| |
| tic = TRACKBAR_GetTic (infoPtr, iTic); |
| range = infoPtr->lRangeMax - infoPtr->lRangeMin; |
| width = infoPtr->rcChannel.right - infoPtr->rcChannel.left; |
| pos = infoPtr->rcChannel.left + (width * tic) / range; |
| |
| return pos; |
| } |
| |
| |
| static HWND |
| TRACKBAR_SetBuddy (TRACKBAR_INFO *infoPtr, BOOL fLocation, HWND hwndBuddy) |
| { |
| HWND hwndTemp; |
| |
| if (fLocation) { |
| /* buddy is left or above */ |
| hwndTemp = infoPtr->hwndBuddyLA; |
| infoPtr->hwndBuddyLA = hwndBuddy; |
| } |
| else { |
| /* buddy is right or below */ |
| hwndTemp = infoPtr->hwndBuddyRB; |
| infoPtr->hwndBuddyRB = hwndBuddy; |
| } |
| |
| TRACKBAR_AlignBuddies (infoPtr); |
| |
| return hwndTemp; |
| } |
| |
| |
| static LONG inline |
| TRACKBAR_SetLineSize (TRACKBAR_INFO *infoPtr, LONG lLineSize) |
| { |
| LONG lTemp = infoPtr->lLineSize; |
| |
| infoPtr->lLineSize = lLineSize; |
| |
| return lTemp; |
| } |
| |
| |
| static LONG inline |
| TRACKBAR_SetPageSize (TRACKBAR_INFO *infoPtr, LONG lPageSize) |
| { |
| LONG lTemp = infoPtr->lPageSize; |
| |
| infoPtr->lPageSize = lPageSize; |
| |
| return lTemp; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetPos (TRACKBAR_INFO *infoPtr, BOOL fPosition, LONG lPosition) |
| { |
| LONG oldPos = infoPtr->lPos; |
| infoPtr->lPos = lPosition; |
| |
| if (infoPtr->lPos < infoPtr->lRangeMin) |
| infoPtr->lPos = infoPtr->lRangeMin; |
| |
| if (infoPtr->lPos > infoPtr->lRangeMax) |
| infoPtr->lPos = infoPtr->lRangeMax; |
| infoPtr->flags |= TB_THUMBPOSCHANGED; |
| |
| if (fPosition) TRACKBAR_InvalidateThumbMove(infoPtr, oldPos, lPosition); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetRange (TRACKBAR_INFO *infoPtr, BOOL fRedraw, LONG lRange) |
| { |
| infoPtr->lRangeMin = (SHORT)LOWORD(lRange); |
| infoPtr->lRangeMax = (SHORT)HIWORD(lRange); |
| |
| if (infoPtr->lPos < infoPtr->lRangeMin) { |
| infoPtr->lPos = infoPtr->lRangeMin; |
| infoPtr->flags |= TB_THUMBPOSCHANGED; |
| } |
| |
| if (infoPtr->lPos > infoPtr->lRangeMax) { |
| infoPtr->lPos = infoPtr->lRangeMax; |
| infoPtr->flags |= TB_THUMBPOSCHANGED; |
| } |
| |
| infoPtr->lPageSize = (infoPtr->lRangeMax - infoPtr->lRangeMin) / 5; |
| if (infoPtr->lPageSize == 0) infoPtr->lPageSize = 1; |
| |
| if (fRedraw) TRACKBAR_InvalidateAll(infoPtr); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetRangeMax (TRACKBAR_INFO *infoPtr, BOOL fRedraw, LONG lMax) |
| { |
| infoPtr->lRangeMax = lMax; |
| if (infoPtr->lPos > infoPtr->lRangeMax) { |
| infoPtr->lPos = infoPtr->lRangeMax; |
| infoPtr->flags |= TB_THUMBPOSCHANGED; |
| } |
| |
| infoPtr->lPageSize = (infoPtr->lRangeMax - infoPtr->lRangeMin) / 5; |
| if (infoPtr->lPageSize == 0) infoPtr->lPageSize = 1; |
| |
| if (fRedraw) TRACKBAR_InvalidateAll(infoPtr); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetRangeMin (TRACKBAR_INFO *infoPtr, BOOL fRedraw, LONG lMin) |
| { |
| infoPtr->lRangeMin = lMin; |
| if (infoPtr->lPos < infoPtr->lRangeMin) { |
| infoPtr->lPos = infoPtr->lRangeMin; |
| infoPtr->flags |= TB_THUMBPOSCHANGED; |
| } |
| |
| infoPtr->lPageSize = (infoPtr->lRangeMax - infoPtr->lRangeMin) / 5; |
| if (infoPtr->lPageSize == 0) infoPtr->lPageSize = 1; |
| |
| if (fRedraw) TRACKBAR_InvalidateAll(infoPtr); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetSel (TRACKBAR_INFO *infoPtr, BOOL fRedraw, LONG lSel) |
| { |
| if (!GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE) & TBS_ENABLESELRANGE) |
| return 0; |
| |
| infoPtr->lSelMin = (SHORT)LOWORD(lSel); |
| infoPtr->lSelMax = (SHORT)HIWORD(lSel); |
| infoPtr->flags |= TB_SELECTIONCHANGED; |
| |
| if (infoPtr->lSelMin < infoPtr->lRangeMin) |
| infoPtr->lSelMin = infoPtr->lRangeMin; |
| if (infoPtr->lSelMax > infoPtr->lRangeMax) |
| infoPtr->lSelMax = infoPtr->lRangeMax; |
| |
| if (fRedraw) TRACKBAR_InvalidateAll(infoPtr); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetSelEnd (TRACKBAR_INFO *infoPtr, BOOL fRedraw, LONG lEnd) |
| { |
| if (!GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE) & TBS_ENABLESELRANGE) |
| return 0; |
| |
| infoPtr->lSelMax = lEnd; |
| infoPtr->flags |= TB_SELECTIONCHANGED; |
| |
| if (infoPtr->lSelMax > infoPtr->lRangeMax) |
| infoPtr->lSelMax = infoPtr->lRangeMax; |
| |
| if (fRedraw) TRACKBAR_InvalidateAll(infoPtr); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetSelStart (TRACKBAR_INFO *infoPtr, BOOL fRedraw, LONG lStart) |
| { |
| if (!GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE) & TBS_ENABLESELRANGE) |
| return 0; |
| |
| infoPtr->lSelMin = lStart; |
| infoPtr->flags |=TB_SELECTIONCHANGED; |
| |
| if (infoPtr->lSelMin < infoPtr->lRangeMin) |
| infoPtr->lSelMin = infoPtr->lRangeMin; |
| |
| if (fRedraw) TRACKBAR_InvalidateAll(infoPtr); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetThumbLength (TRACKBAR_INFO *infoPtr, UINT iLength) |
| { |
| if (GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE) & TBS_FIXEDLENGTH) |
| infoPtr->uThumbLen = iLength; |
| |
| infoPtr->flags |= TB_THUMBSIZECHANGED; |
| |
| InvalidateRect (infoPtr->hwndSelf, &infoPtr->rcThumb, FALSE); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetTic (TRACKBAR_INFO *infoPtr, LONG lPos) |
| { |
| if (GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE) & TBS_AUTOTICKS) |
| return FALSE; |
| |
| if ((lPos < infoPtr->lRangeMin) || (lPos> infoPtr->lRangeMax)) |
| return FALSE; |
| |
| TRACE("lPos=%ld\n", lPos); |
| |
| infoPtr->uNumTics++; |
| infoPtr->tics=COMCTL32_ReAlloc( infoPtr->tics, |
| (infoPtr->uNumTics)*sizeof (DWORD)); |
| if (!infoPtr->tics) { |
| infoPtr->uNumTics = 0; |
| notify_outofmemory(infoPtr); |
| return FALSE; |
| } |
| infoPtr->tics[infoPtr->uNumTics-1] = lPos; |
| |
| TRACKBAR_InvalidateAll(infoPtr); |
| |
| return TRUE; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetTicFreq (TRACKBAR_INFO *infoPtr, WORD wFreq) |
| { |
| if (GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE) & TBS_AUTOTICKS) { |
| infoPtr->uTicFreq = wFreq; |
| TRACKBAR_RecalculateTics (infoPtr); |
| TRACKBAR_InvalidateAll(infoPtr); |
| } |
| |
| return 0; |
| } |
| |
| |
| static INT inline |
| TRACKBAR_SetTipSide (TRACKBAR_INFO *infoPtr, INT fLocation) |
| { |
| INT fTemp = infoPtr->fLocation; |
| |
| infoPtr->fLocation = fLocation; |
| |
| return fTemp; |
| } |
| |
| |
| static LRESULT inline |
| TRACKBAR_SetToolTips (TRACKBAR_INFO *infoPtr, HWND hwndTT) |
| { |
| infoPtr->hwndToolTip = hwndTT; |
| |
| return 0; |
| } |
| |
| |
| static BOOL inline |
| TRACKBAR_SetUnicodeFormat (TRACKBAR_INFO *infoPtr, BOOL fUnicode) |
| { |
| BOOL bTemp = infoPtr->bUnicode; |
| |
| infoPtr->bUnicode = fUnicode; |
| |
| return bTemp; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_InitializeThumb (TRACKBAR_INFO *infoPtr) |
| { |
| infoPtr->uThumbLen = 23; /* initial thumb length */ |
| |
| TRACKBAR_CalcChannel (infoPtr); |
| TRACKBAR_UpdateThumb (infoPtr); |
| infoPtr->flags &= ~TB_SELECTIONCHANGED; |
| |
| return 0; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_Create (HWND hwnd, LPCREATESTRUCTW lpcs) |
| { |
| TRACKBAR_INFO *infoPtr; |
| DWORD oldStyle, newStyle; |
| |
| infoPtr = (TRACKBAR_INFO *)COMCTL32_Alloc (sizeof(TRACKBAR_INFO)); |
| if (!infoPtr) return -1; |
| SetWindowLongW (hwnd, 0, (DWORD)infoPtr); |
| |
| /* set default values */ |
| infoPtr->hwndSelf = hwnd; |
| infoPtr->lRangeMin = 0; |
| infoPtr->lRangeMax = 100; |
| infoPtr->lLineSize = 1; |
| infoPtr->lPageSize = 20; |
| infoPtr->lSelMin = 0; |
| infoPtr->lSelMax = 0; |
| infoPtr->lPos = 0; |
| infoPtr->fLocation = -1; |
| infoPtr->uNumTics = 0; /* start and end tic are not included in count*/ |
| infoPtr->uTicFreq = 1; |
| infoPtr->tics = NULL; |
| infoPtr->hwndNotify= GetParent (hwnd); |
| |
| TRACKBAR_InitializeThumb (infoPtr); |
| |
| oldStyle = newStyle = GetWindowLongW (hwnd, GWL_STYLE); |
| if (oldStyle & TBS_VERT) { |
| if (! (oldStyle & (TBS_LEFT | TBS_RIGHT | TBS_BOTH)) ) |
| newStyle |= TBS_RIGHT; |
| } else { |
| if (! (oldStyle & (TBS_TOP | TBS_BOTTOM | TBS_BOTH)) ) |
| newStyle |= TBS_BOTTOM; |
| } |
| if (newStyle != oldStyle) |
| SetWindowLongW (hwnd, GWL_STYLE, newStyle); |
| |
| /* Create tooltip control */ |
| if (newStyle & TBS_TOOLTIPS) { |
| |
| infoPtr->hwndToolTip = |
| CreateWindowExW (0, TOOLTIPS_CLASSW, NULL, 0, |
| CW_USEDEFAULT, CW_USEDEFAULT, |
| CW_USEDEFAULT, CW_USEDEFAULT, |
| hwnd, 0, 0, 0); |
| |
| if (infoPtr->hwndToolTip) { |
| TTTOOLINFOW ti; |
| notify_tooltipscreated(infoPtr); |
| |
| ZeroMemory (&ti, sizeof(ti)); |
| ti.cbSize = sizeof(ti); |
| ti.uFlags = TTF_IDISHWND | TTF_TRACK | TTF_ABSOLUTE; |
| ti.hwnd = hwnd; |
| |
| SendMessageW (infoPtr->hwndToolTip, TTM_ADDTOOLW, 0, (LPARAM)&ti); |
| } |
| } |
| |
| return 0; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_Destroy (TRACKBAR_INFO *infoPtr) |
| { |
| /* delete tooltip control */ |
| if (infoPtr->hwndToolTip) |
| DestroyWindow (infoPtr->hwndToolTip); |
| |
| COMCTL32_Free (infoPtr); |
| SetWindowLongW (infoPtr->hwndSelf, 0, 0); |
| return 0; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_KillFocus (TRACKBAR_INFO *infoPtr, HWND hwndGetFocus) |
| { |
| TRACE("\n"); |
| |
| TRACKBAR_InvalidateAll(infoPtr); |
| |
| return 0; |
| } |
| |
| static LRESULT |
| TRACKBAR_LButtonDown (TRACKBAR_INFO *infoPtr, DWORD fwKeys, POINTS pts) |
| { |
| POINT clickPoint = { pts.x, pts.y }; |
| |
| SetFocus(infoPtr->hwndSelf); |
| |
| if (PtInRect(&infoPtr->rcThumb, clickPoint)) { |
| infoPtr->flags |= TB_DRAG_MODE; |
| SetCapture (infoPtr->hwndSelf); |
| TRACKBAR_UpdateToolTip (infoPtr); |
| TRACKBAR_ActivateToolTip (infoPtr, TRUE); |
| TRACKBAR_InvalidateThumb(infoPtr, infoPtr->lPos); |
| } else { |
| LONG dir = TRACKBAR_GetAutoPageDirection(infoPtr, clickPoint); |
| if (dir == 0) return 0; |
| infoPtr->flags |= (dir < 0) ? TB_AUTO_PAGE_LEFT : TB_AUTO_PAGE_RIGHT; |
| TRACKBAR_AutoPage (infoPtr, clickPoint); |
| SetCapture (infoPtr->hwndSelf); |
| SetTimer(infoPtr->hwndSelf, TB_REFRESH_TIMER, TB_REFRESH_DELAY, 0); |
| } |
| |
| return 0; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_LButtonUp (TRACKBAR_INFO *infoPtr, DWORD fwKeys, POINTS pts) |
| { |
| if (infoPtr->flags & TB_DRAG_MODE) { |
| TRACKBAR_SendNotify (infoPtr, TB_ENDTRACK); |
| infoPtr->flags &= ~TB_DRAG_MODE; |
| ReleaseCapture (); |
| notify_releasedcapture(infoPtr); |
| TRACKBAR_ActivateToolTip(infoPtr, FALSE); |
| TRACKBAR_InvalidateThumb(infoPtr, infoPtr->lPos); |
| } |
| if (infoPtr->flags & TB_AUTO_PAGE) { |
| KillTimer (infoPtr->hwndSelf, TB_REFRESH_TIMER); |
| infoPtr->flags &= ~TB_AUTO_PAGE; |
| ReleaseCapture (); |
| notify_releasedcapture(infoPtr); |
| } |
| |
| return 0; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_CaptureChanged (TRACKBAR_INFO *infoPtr) |
| { |
| TRACKBAR_SendNotify (infoPtr, TB_ENDTRACK); |
| return 0; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_Paint (TRACKBAR_INFO *infoPtr, HDC hdc) |
| { |
| if (hdc) { |
| TRACKBAR_Refresh(infoPtr, hdc); |
| } else { |
| PAINTSTRUCT ps; |
| hdc = BeginPaint (infoPtr->hwndSelf, &ps); |
| TRACKBAR_Refresh (infoPtr, hdc); |
| EndPaint (infoPtr->hwndSelf, &ps); |
| } |
| |
| return 0; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_SetFocus (TRACKBAR_INFO *infoPtr, HWND hwndLoseFocus) |
| { |
| TRACE("\n"); |
| |
| TRACKBAR_InvalidateAll(infoPtr); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_Size (TRACKBAR_INFO *infoPtr, DWORD fwSizeType, INT nWidth, INT nHeight) |
| { |
| TRACKBAR_CalcChannel (infoPtr); |
| TRACKBAR_AlignBuddies (infoPtr); |
| |
| return 0; |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_Timer (TRACKBAR_INFO *infoPtr, INT wTimerID, TIMERPROC *tmrpc) |
| { |
| if (infoPtr->flags & TB_AUTO_PAGE) { |
| POINT pt; |
| if (GetCursorPos(&pt)) |
| if (ScreenToClient(infoPtr->hwndSelf, &pt)) |
| TRACKBAR_AutoPage(infoPtr, pt); |
| } |
| return 0; |
| } |
| |
| |
| static BOOL |
| TRACKBAR_SendNotify (TRACKBAR_INFO *infoPtr, UINT code) |
| { |
| BOOL bVert = GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE) & TBS_VERT; |
| |
| TRACE("%x\n", code); |
| |
| return (BOOL) SendMessageW (GetParent (infoPtr->hwndSelf), |
| bVert ? WM_VSCROLL : WM_HSCROLL, |
| (WPARAM)code, (LPARAM)infoPtr->hwndSelf); |
| } |
| |
| |
| static LRESULT |
| TRACKBAR_MouseMove (TRACKBAR_INFO *infoPtr, DWORD fwKeys, POINTS pts) |
| { |
| DWORD dwStyle = GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE); |
| INT clickPlace = (dwStyle & TBS_VERT) ? pts.y : pts.x; |
| DOUBLE dragPos, oldPos = infoPtr->lPos; |
| |
| TRACE("(x=%d. y=%d)\n", pts.x, pts.y); |
| |
| if (infoPtr->flags & TB_AUTO_PAGE) { |
| POINT pt; |
| POINTSTOPOINT(pt, pts); |
| TRACKBAR_AutoPage (infoPtr, pt); |
| return TRUE; |
| } |
| |
| if (!(infoPtr->flags & TB_DRAG_MODE)) return TRUE; |
| |
| dragPos = TRACKBAR_ConvertPlaceToPosition (infoPtr, clickPlace, |
| dwStyle & TBS_VERT); |
| if (dragPos > ((INT)dragPos) + 0.5) dragPos++; |
| |
| if (dragPos == oldPos) return TRUE; |
| |
| infoPtr->lPos = dragPos; |
| |
| infoPtr->flags |= TB_THUMBPOSCHANGED; |
| TRACKBAR_SendNotify (infoPtr, TB_THUMBTRACK | (infoPtr->lPos<<16)); |
| |
| |
| TRACKBAR_InvalidateThumbMove(infoPtr, oldPos, dragPos); |
| UpdateWindow (infoPtr->hwndSelf); |
| |
| return TRUE; |
| } |
| |
| |
| static BOOL |
| TRACKBAR_KeyDown (TRACKBAR_INFO *infoPtr, INT nVirtKey, DWORD lKeyData) |
| { |
| BOOL downIsLeft = GetWindowLongW (infoPtr->hwndSelf, GWL_STYLE) & TBS_DOWNISLEFT; |
| LONG pos = infoPtr->lPos; |
| |
| TRACE("%x\n", nVirtKey); |
| |
| switch (nVirtKey) { |
| case VK_UP: |
| if (downIsLeft) goto step_right; |
| case VK_LEFT: |
| step_left: |
| if (infoPtr->lPos == infoPtr->lRangeMin) return FALSE; |
| infoPtr->lPos -= infoPtr->lLineSize; |
| if (infoPtr->lPos < infoPtr->lRangeMin) |
| infoPtr->lPos = infoPtr->lRangeMin; |
| TRACKBAR_SendNotify (infoPtr, TB_LINEUP); |
| break; |
| case VK_DOWN: |
| if (downIsLeft) goto step_left; |
| case VK_RIGHT: |
| step_right: |
| if (infoPtr->lPos == infoPtr->lRangeMax) return FALSE; |
| infoPtr->lPos += infoPtr->lLineSize; |
| if (infoPtr->lPos > infoPtr->lRangeMax) |
| infoPtr->lPos = infoPtr->lRangeMax; |
| TRACKBAR_SendNotify (infoPtr, TB_LINEDOWN); |
| break; |
| case VK_NEXT: |
| if (downIsLeft) goto page_left; |
| page_right: |
| TRACKBAR_PageUp(infoPtr); |
| break; |
| case VK_PRIOR: |
| if (downIsLeft) goto page_right; |
| page_left: |
| TRACKBAR_PageDown(infoPtr); |
| break; |
| case VK_HOME: |
| if (infoPtr->lPos == infoPtr->lRangeMin) return FALSE; |
| infoPtr->lPos = infoPtr->lRangeMin; |
| TRACKBAR_SendNotify (infoPtr, TB_TOP); |
| break; |
| case VK_END: |
| if (infoPtr->lPos == infoPtr->lRangeMax) return FALSE; |
| infoPtr->lPos = infoPtr->lRangeMax; |
| TRACKBAR_SendNotify (infoPtr, TB_BOTTOM); |
| break; |
| } |
| |
| if (pos != infoPtr->lPos) { |
| infoPtr->flags |=TB_THUMBPOSCHANGED; |
| TRACKBAR_InvalidateThumbMove (infoPtr, pos, infoPtr->lPos); |
| } |
| |
| return TRUE; |
| } |
| |
| |
| static BOOL inline |
| TRACKBAR_KeyUp (TRACKBAR_INFO *infoPtr, INT nVirtKey, DWORD lKeyData) |
| { |
| switch (nVirtKey) { |
| case VK_LEFT: |
| case VK_UP: |
| case VK_RIGHT: |
| case VK_DOWN: |
| case VK_NEXT: |
| case VK_PRIOR: |
| case VK_HOME: |
| case VK_END: |
| TRACKBAR_SendNotify (infoPtr, TB_ENDTRACK); |
| } |
| return TRUE; |
| } |
| |
| |
| static LRESULT WINAPI |
| TRACKBAR_WindowProc (HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) |
| { |
| TRACKBAR_INFO *infoPtr = (TRACKBAR_INFO *)GetWindowLongW (hwnd, 0); |
| |
| TRACE("hwnd=%p msg=%x wparam=%x lparam=%lx\n", hwnd, uMsg, wParam, lParam); |
| |
| if (!infoPtr && (uMsg != WM_CREATE)) |
| return DefWindowProcW (hwnd, uMsg, wParam, lParam); |
| |
| switch (uMsg) |
| { |
| case TBM_CLEARSEL: |
| return TRACKBAR_ClearSel (infoPtr, (BOOL)wParam); |
| |
| case TBM_CLEARTICS: |
| return TRACKBAR_ClearTics (infoPtr, (BOOL)wParam); |
| |
| case TBM_GETBUDDY: |
| return (LRESULT)(wParam ? infoPtr->hwndBuddyLA : infoPtr->hwndBuddyRB); |
| |
| case TBM_GETCHANNELRECT: |
| return TRACKBAR_GetChannelRect (infoPtr, (LPRECT)lParam); |
| |
| case TBM_GETLINESIZE: |
| return infoPtr->lLineSize; |
| |
| case TBM_GETNUMTICS: |
| return TRACKBAR_GetNumTics (infoPtr); |
| |
| case TBM_GETPAGESIZE: |
| return infoPtr->lPageSize; |
| |
| case TBM_GETPOS: |
| return infoPtr->lPos; |
| |
| case TBM_GETPTICS: |
| return (LRESULT)infoPtr->tics; |
| |
| case TBM_GETRANGEMAX: |
| return infoPtr->lRangeMax; |
| |
| case TBM_GETRANGEMIN: |
| return infoPtr->lRangeMin; |
| |
| case TBM_GETSELEND: |
| return infoPtr->lSelMax; |
| |
| case TBM_GETSELSTART: |
| return infoPtr->lSelMin; |
| |
| case TBM_GETTHUMBLENGTH: |
| return infoPtr->uThumbLen; |
| |
| case TBM_GETTHUMBRECT: |
| return CopyRect((LPRECT)lParam, &infoPtr->rcThumb); |
| |
| case TBM_GETTIC: |
| return TRACKBAR_GetTic (infoPtr, (INT)wParam); |
| |
| case TBM_GETTICPOS: |
| return TRACKBAR_GetTicPos (infoPtr, (INT)wParam); |
| |
| case TBM_GETTOOLTIPS: |
| return (LRESULT)infoPtr->hwndToolTip; |
| |
| case TBM_GETUNICODEFORMAT: |
| return infoPtr->bUnicode; |
| |
| case TBM_SETBUDDY: |
| return (LRESULT) TRACKBAR_SetBuddy(infoPtr, (BOOL)wParam, (HWND)lParam); |
| |
| case TBM_SETLINESIZE: |
| return TRACKBAR_SetLineSize (infoPtr, (LONG)lParam); |
| |
| case TBM_SETPAGESIZE: |
| return TRACKBAR_SetPageSize (infoPtr, (LONG)lParam); |
| |
| case TBM_SETPOS: |
| return TRACKBAR_SetPos (infoPtr, (BOOL)wParam, (LONG)lParam); |
| |
| case TBM_SETRANGE: |
| return TRACKBAR_SetRange (infoPtr, (BOOL)wParam, (LONG)lParam); |
| |
| case TBM_SETRANGEMAX: |
| return TRACKBAR_SetRangeMax (infoPtr, (BOOL)wParam, (LONG)lParam); |
| |
| case TBM_SETRANGEMIN: |
| return TRACKBAR_SetRangeMin (infoPtr, (BOOL)wParam, (LONG)lParam); |
| |
| case TBM_SETSEL: |
| return TRACKBAR_SetSel (infoPtr, (BOOL)wParam, (LONG)lParam); |
| |
| case TBM_SETSELEND: |
| return TRACKBAR_SetSelEnd (infoPtr, (BOOL)wParam, (LONG)lParam); |
| |
| case TBM_SETSELSTART: |
| return TRACKBAR_SetSelStart (infoPtr, (BOOL)wParam, (LONG)lParam); |
| |
| case TBM_SETTHUMBLENGTH: |
| return TRACKBAR_SetThumbLength (infoPtr, (UINT)wParam); |
| |
| case TBM_SETTIC: |
| return TRACKBAR_SetTic (infoPtr, (LONG)lParam); |
| |
| case TBM_SETTICFREQ: |
| return TRACKBAR_SetTicFreq (infoPtr, (WORD)wParam); |
| |
| case TBM_SETTIPSIDE: |
| return TRACKBAR_SetTipSide (infoPtr, (INT)wParam); |
| |
| case TBM_SETTOOLTIPS: |
| return TRACKBAR_SetToolTips (infoPtr, (HWND)wParam); |
| |
| case TBM_SETUNICODEFORMAT: |
| return TRACKBAR_SetUnicodeFormat (infoPtr, (BOOL)wParam); |
| |
| |
| case WM_CAPTURECHANGED: |
| return TRACKBAR_CaptureChanged (infoPtr); |
| |
| case WM_CREATE: |
| return TRACKBAR_Create (hwnd, (LPCREATESTRUCTW)lParam); |
| |
| case WM_DESTROY: |
| return TRACKBAR_Destroy (infoPtr); |
| |
| /* case WM_ENABLE: */ |
| |
| case WM_ERASEBKGND: |
| return 0; |
| |
| case WM_GETDLGCODE: |
| return DLGC_WANTARROWS; |
| |
| case WM_KEYDOWN: |
| return TRACKBAR_KeyDown (infoPtr, (INT)wParam, (DWORD)lParam); |
| |
| case WM_KEYUP: |
| return TRACKBAR_KeyUp (infoPtr, (INT)wParam, (DWORD)lParam); |
| |
| case WM_KILLFOCUS: |
| return TRACKBAR_KillFocus (infoPtr, (HWND)wParam); |
| |
| case WM_LBUTTONDOWN: |
| return TRACKBAR_LButtonDown (infoPtr, wParam, MAKEPOINTS(lParam)); |
| |
| case WM_LBUTTONUP: |
| return TRACKBAR_LButtonUp (infoPtr, wParam, MAKEPOINTS(lParam)); |
| |
| case WM_MOUSEMOVE: |
| return TRACKBAR_MouseMove (infoPtr, wParam, MAKEPOINTS(lParam)); |
| |
| case WM_PAINT: |
| return TRACKBAR_Paint (infoPtr, (HDC)wParam); |
| |
| case WM_SETFOCUS: |
| return TRACKBAR_SetFocus (infoPtr, (HWND)wParam); |
| |
| case WM_SIZE: |
| return TRACKBAR_Size (infoPtr, wParam, LOWORD(lParam), HIWORD(lParam)); |
| |
| case WM_TIMER: |
| return TRACKBAR_Timer (infoPtr, (INT)wParam, (TIMERPROC *)lParam); |
| |
| case WM_WININICHANGE: |
| return TRACKBAR_InitializeThumb (infoPtr); |
| |
| default: |
| if ((uMsg >= WM_USER) && (uMsg < WM_APP)) |
| ERR("unknown msg %04x wp=%08x lp=%08lx\n", uMsg, wParam, lParam); |
| return DefWindowProcW (hwnd, uMsg, wParam, lParam); |
| } |
| return 0; |
| } |
| |
| |
| void TRACKBAR_Register (void) |
| { |
| WNDCLASSW wndClass; |
| |
| ZeroMemory (&wndClass, sizeof(WNDCLASSW)); |
| wndClass.style = CS_GLOBALCLASS; |
| wndClass.lpfnWndProc = (WNDPROC)TRACKBAR_WindowProc; |
| wndClass.cbClsExtra = 0; |
| wndClass.cbWndExtra = sizeof(TRACKBAR_INFO *); |
| wndClass.hCursor = LoadCursorW (0, IDC_ARROWW); |
| wndClass.hbrBackground = (HBRUSH)(COLOR_3DFACE + 1); |
| wndClass.lpszClassName = TRACKBAR_CLASSW; |
| |
| RegisterClassW (&wndClass); |
| } |
| |
| |
| void TRACKBAR_Unregister (void) |
| { |
| UnregisterClassW (TRACKBAR_CLASSW, (HINSTANCE)NULL); |
| } |