#include "main.h" #include "./widget.h" typedef enum WidgetState { WIDGET_STATE_MOUSE_MOVE_TRACKED = (1 << 0), WIDGET_STATE_DISABLE_CHILDREN_SCROLL = (1 << 1), } WidgetState; DEFINE_ENUM_FLAG_OPERATORS(WidgetState); #define WIDGET_IS_FROZEN(_widget) (0 != (_widget)->freezer) #define WIDGET_IS_MOUSE_MOVE_TRACKED(_widget) (0 != (WIDGET_STATE_MOUSE_MOVE_TRACKED & (_widget)->state)) #define WIDGET_SET_MOUSE_MOVE_TRACK(_widget) (((_widget)->state) |= WIDGET_STATE_MOUSE_MOVE_TRACKED) #define WIDGET_UNSET_MOUSE_MOVE_TRACK(_widget) (((_widget)->state) &= ~WIDGET_STATE_MOUSE_MOVE_TRACKED) #define WIDGET_IS_CHILDREN_SCROLL_DISABLED(_widget) (0 != (WIDGET_STATE_DISABLE_CHILDREN_SCROLL & (_widget)->state)) #define WIDGET_SET_DISABLE_CHILDREN_SCROLL(_widget) (((_widget)->state) |= WIDGET_STATE_DISABLE_CHILDREN_SCROLL) #define WIDGET_UNSET_DISABLE_CHILDREN_SCROLL(_widget) (((_widget)->state) &= ~WIDGET_STATE_DISABLE_CHILDREN_SCROLL) typedef struct Widget { unsigned int type; WidgetState state; const WidgetInterface *callbacks; void *object; WidgetStyle *style; wchar_t *text; HFONT font; SIZE viewSize; POINT viewOrigin; int wheelCarryover; size_t freezer; } Widget; typedef struct WidgetCreateParam { unsigned int type; const WidgetInterface *callbacks; void *param; const wchar_t *text; } WidgetCreateParam; #define WIDGET(_hwnd) ((Widget*)(LONGX86)GetWindowLongPtrW((_hwnd), 0)) #define WIDGET_RET_VOID(_view, _hwnd) {(_view) = WIDGET((_hwnd)); if (NULL == (_view)) return;} #define WIDGET_RET_VAL(_view, _hwnd, _error) {(_view) = WIDGET((_hwnd)); if (NULL == (_view)) return (_error);} #define WIDGETSTYLE(_widget) (((Widget*)(_widget))->style) #define WIDGETOBJECT(_widget) (((Widget*)(_widget))->object) #define WIDGETCALLBACKS(_widget) (((Widget*)(_widget))->callbacks) static UINT WINAMP_WM_DIRECT_MOUSE_WHEEL = WM_NULL; static LRESULT CALLBACK Widget_WindowProc(HWND hwnd, unsigned int uMsg, WPARAM wParam, LPARAM lParam); static ATOM Widget_GetClassAtom(HINSTANCE instance) { WNDCLASSEXW klass; ATOM klassAtom; klassAtom = (ATOM)GetClassInfoExW(instance, WIDGET_WINDOW_CLASS, &klass); if (0 != klassAtom) return klassAtom; memset(&klass, 0, sizeof(klass)); klass.cbSize = sizeof(klass); klass.style = CS_DBLCLKS; klass.lpfnWndProc = Widget_WindowProc; klass.cbClsExtra = 0; klass.cbWndExtra = sizeof(Widget*); klass.hInstance = instance; klass.hIcon = NULL; klass.hCursor = LoadCursorW(NULL, (LPCWSTR)IDC_ARROW); klass.hbrBackground = NULL; klass.lpszMenuName = NULL; klass.lpszClassName = WIDGET_WINDOW_CLASS; klass.hIconSm = NULL; klassAtom = RegisterClassExW(&klass); return klassAtom; } HWND Widget_CreateWindow(unsigned int type, const WidgetInterface *callbacks, const wchar_t *text, unsigned long windowExStyle, unsigned long windowStyle, int x, int y, int width, int height, HWND parentWindow, unsigned int controlId, void *param) { HINSTANCE instance; ATOM klassAtom; HWND hwnd; WidgetCreateParam createParam; if (NULL == parentWindow || FALSE == IsWindow(parentWindow)) return NULL; instance = GetModuleHandleW(NULL); klassAtom = Widget_GetClassAtom(instance); if (0 == klassAtom) return NULL; createParam.type = type; createParam.param = param; createParam.callbacks = callbacks; createParam.text = text; hwnd = CreateWindowExW(WS_EX_NOPARENTNOTIFY | windowExStyle, (LPCWSTR)MAKEINTATOM(klassAtom), NULL, WS_CHILD | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | windowStyle, x, y, width, height, parentWindow, (HMENU)controlId, instance, &createParam); return hwnd; } static LRESULT Widget_DefWindowProc(HWND hwnd, unsigned int uMsg, WPARAM wParam, LPARAM lParam) { Widget *self; self = WIDGET(hwnd); if (NULL != self && NULL != self->callbacks->messageProc) { LRESULT result; result = 0; if (FALSE != self->callbacks->messageProc(self->object, hwnd, uMsg, wParam, lParam, &result)) { return result; } } return DefWindowProc(hwnd, uMsg, wParam, lParam); } static void Widget_Freeze(Widget *self) { if (NULL != self) self->freezer++; } static void Widget_Thaw(Widget *self) { if (NULL != self && 0 != self->freezer) self->freezer--; } static INT Widget_ScrollBarOffsetPos(HWND hwnd, INT barType, INT delta, BOOL redraw) { Widget *self; INT position; SCROLLINFO scrollInfo; self = WIDGET(hwnd); scrollInfo.cbSize = sizeof(scrollInfo); scrollInfo.fMask = SIF_POS | SIF_RANGE | SIF_PAGE; if (FALSE == GetScrollInfo(hwnd, barType, &scrollInfo)) return 0; position = scrollInfo.nPos + delta; if (position < scrollInfo.nMin) position = scrollInfo.nMin; else if (position > (scrollInfo.nMax - (INT)scrollInfo.nPage)) position = scrollInfo.nMax - (INT)scrollInfo.nPage + 1; delta = position - scrollInfo.nPos; scrollInfo.fMask = SIF_POS; scrollInfo.nPos = position; SetScrollInfo(hwnd, barType, &scrollInfo, redraw); if (NULL != self) { if (SB_HORZ == barType) self->viewOrigin.x = -position; else self->viewOrigin.y = -position; } return delta; } static BOOL Widget_ScrollContent(HWND hwnd, int dx, int dy, BOOL redraw) { Widget *self; UINT scrollFlags; HRGN invalidRgn; INT scrollError; if (0 == dx && 0 == dy) return FALSE; self = WIDGET(hwnd); if (NULL != self && NULL != self->callbacks && NULL != self->callbacks->scrollBefore) { self->callbacks->scrollBefore(WIDGETOBJECT(self), hwnd, &dx, &dy); if (0 == dx && 0 == dy) return FALSE; } scrollFlags = (FALSE == WIDGET_IS_CHILDREN_SCROLL_DISABLED(self)) ? SW_SCROLLCHILDREN : 0; if (FALSE != redraw) { invalidRgn = CreateRectRgn(0, 0, 0, 0); scrollFlags |= SW_INVALIDATE | SW_ERASE; } else { invalidRgn = NULL; } scrollError = ScrollWindowEx(hwnd, -dx, -dy, NULL, NULL, invalidRgn, NULL, scrollFlags); if (ERROR != scrollError) { if (NULL != self && NULL != self->callbacks && NULL != self->callbacks->scroll) { self->callbacks->scroll(WIDGETOBJECT(self), hwnd, &dx, &dy); } if (FALSE != redraw && NULLREGION != scrollError) { RedrawWindow(hwnd, NULL, invalidRgn, RDW_ERASENOW | RDW_UPDATENOW | RDW_ALLCHILDREN); } } if (NULL != invalidRgn) DeleteObject(invalidRgn); return (ERROR != scrollError); } static BOOL Widget_SyncContentOrigin(HWND hwnd, BOOL redraw) { Widget *self; RECT clientRect; SCROLLINFO scrollInfo; INT dx, dy; WIDGET_RET_VAL(self, hwnd, FALSE); scrollInfo.cbSize = sizeof(scrollInfo); scrollInfo.fMask = SIF_POS; if (FALSE == GetClientRect(hwnd, &clientRect)) SetRectEmpty(&clientRect); if (self->viewSize.cx < RECTWIDTH(clientRect)) { scrollInfo.nPos = 0; dx = scrollInfo.nPos + self->viewOrigin.x; self->viewOrigin.x = 0; } else if (FALSE != GetScrollInfo(hwnd, SB_HORZ, &scrollInfo)) { dx = scrollInfo.nPos + self->viewOrigin.x; self->viewOrigin.x = -scrollInfo.nPos; } else dx = 0; if (FALSE != GetScrollInfo(hwnd, SB_VERT, &scrollInfo)) { dy = scrollInfo.nPos + self->viewOrigin.y; self->viewOrigin.y = -scrollInfo.nPos; } else dy = 0; if (0 == dx && 0 == dy) return FALSE; return Widget_ScrollContent(hwnd, dx, dy, redraw); } static BOOL Widget_ScrollWindow(HWND hwnd, INT dx, INT dy, BOOL redraw) { if (0 != dx) dx = Widget_ScrollBarOffsetPos(hwnd, SB_HORZ, dx, redraw); if (0 != dy) dy = Widget_ScrollBarOffsetPos(hwnd, SB_VERT, dy, redraw); return Widget_ScrollContent(hwnd, dx, dy, redraw); } static BOOL Widget_ScrollBarAction(HWND hwnd, INT barType, INT actionLayout, INT line, BOOL redraw) { INT delta; SCROLLINFO scrollInfo; scrollInfo.cbSize = sizeof(scrollInfo); scrollInfo.fMask = SIF_PAGE | SIF_POS | SIF_RANGE | SIF_TRACKPOS; if (FALSE == GetScrollInfo(hwnd, barType, &scrollInfo)) return FALSE; switch(actionLayout) { case SB_BOTTOM: delta = scrollInfo.nMax - scrollInfo.nPos; break; case SB_TOP: delta = scrollInfo.nMin - scrollInfo.nPos; break; case SB_LINEDOWN: delta = line; break; case SB_LINEUP: delta = -line; break; case SB_PAGEDOWN: delta = (INT)scrollInfo.nPage; break; case SB_PAGEUP: delta = -(INT)scrollInfo.nPage; break; case SB_THUMBTRACK: delta = scrollInfo.nTrackPos - scrollInfo.nPos; break; case SB_THUMBPOSITION: case SB_ENDSCROLL: delta = 0; break; default: return FALSE; } if(0 != delta) { Widget_ScrollWindow(hwnd, (SB_HORZ == barType) ? delta : 0, (SB_VERT == barType) ? delta : 0, redraw); } else Widget_SyncContentOrigin(hwnd, redraw); return TRUE; } static BOOL Widget_ScrollBarUpdate(HWND hwnd, INT barType, UINT page, INT max, BOOL redraw) { Widget *self; SCROLLINFO scrollInfo; UINT windowStyle, styleFilter; WIDGET_RET_VAL(self, hwnd, FALSE); windowStyle = GetWindowStyle(hwnd); switch(barType) { case SB_HORZ: styleFilter = WS_HSCROLL; break; case SB_VERT: styleFilter = WS_VSCROLL; break; default: return FALSE; } scrollInfo.cbSize = sizeof(SCROLLINFO); scrollInfo.fMask = SIF_PAGE | SIF_RANGE; if (page >= (UINT)max) { if (0 == (styleFilter & windowStyle)) return FALSE; scrollInfo.nPage = page + 1; scrollInfo.nMin = 0; scrollInfo.nMax = max; scrollInfo.nPos = scrollInfo.nMin; scrollInfo.nTrackPos = scrollInfo.nPos; scrollInfo.fMask |= (SIF_POS | SIF_TRACKPOS); Widget_Freeze(self); SetScrollInfo(hwnd, barType, &scrollInfo, redraw); Widget_Thaw(self); windowStyle = GetWindowStyle(hwnd); if (0 != (styleFilter & windowStyle)) SetWindowStyle(hwnd, windowStyle & ~styleFilter); return TRUE; } if (FALSE == GetScrollInfo(hwnd, barType, &scrollInfo)) { if (ERROR_NO_SCROLLBARS == GetLastError()) { scrollInfo.nPage = 0; scrollInfo.nMax = 0; scrollInfo.nMin = 0; scrollInfo.nPos = scrollInfo.nMin; scrollInfo.nTrackPos = scrollInfo.nPos; } else return FALSE; } scrollInfo.fMask = 0; if (scrollInfo.nPage != page) { scrollInfo.nPage = page; scrollInfo.fMask |= SIF_PAGE; } if (scrollInfo.nMax != max) { scrollInfo.nMax = max; scrollInfo.fMask |= SIF_RANGE; } if (0 == (styleFilter & windowStyle)) { scrollInfo.fMask |= (SIF_POS | SIF_TRACKPOS); scrollInfo.nPos = scrollInfo.nMin; scrollInfo.nTrackPos = scrollInfo.nMin; } if (0 == scrollInfo.fMask) return FALSE; Widget_Freeze(self); SetScrollInfo(hwnd, barType, &scrollInfo, redraw); Widget_Thaw(self); if (0 == (styleFilter & windowStyle)) { windowStyle = GetWindowStyle(hwnd); if (0 == (styleFilter & windowStyle)) SetWindowStyle(hwnd, windowStyle | styleFilter); return TRUE; } return FALSE; } static void Widget_Layout(HWND hwnd, BOOL redraw) { Widget *self; RECT rect; size_t iteration; WIDGET_RET_VOID(self, hwnd); iteration = 0; do { if (iteration++ > 2) break; if (FALSE == GetClientRect(hwnd, &rect)) break; SetSize(&self->viewSize, 0, 0); if (NULL != self->callbacks->layout) { self->callbacks->layout(self->object, hwnd, self->style, &rect, &self->viewSize, redraw); } if (FALSE != IsSizeEmpty(&self->viewSize)) SetSize(&self->viewSize, RECTWIDTH(rect), RECTHEIGHT(rect)); } while(FALSE != Widget_ScrollBarUpdate(hwnd, SB_HORZ, RECTWIDTH(rect), self->viewSize.cx, redraw) || FALSE != Widget_ScrollBarUpdate(hwnd, SB_VERT, RECTHEIGHT(rect), self->viewSize.cy, redraw)); Widget_SyncContentOrigin(hwnd, redraw); } static BOOL Widget_Paint(HWND hwnd, HDC hdc, const RECT *paintRect, BOOL erase) { Widget *self; BOOL result; self = WIDGET(hwnd); if (NULL == self || NULL == self->style) return FALSE; if (NULL != self->callbacks->paint) { POINT prevOrigin; RECT viewRect; CopyRect(&viewRect, paintRect); OffsetRect(&viewRect, -self->viewOrigin.x, -self->viewOrigin.y); OffsetViewportOrgEx(hdc, self->viewOrigin.x, self->viewOrigin.y, &prevOrigin); result = self->callbacks->paint(self->object, hwnd, self->style, hdc, &viewRect, erase); SetViewportOrgEx(hdc, prevOrigin.x, prevOrigin.y, NULL); } else result = FALSE; if (FALSE == result) { if (FALSE != erase) result = FillRect(hdc, paintRect, WIDGETSTYLE_BACK_BRUSH(self->style)); else result = TRUE; } return result; } static void Widget_FocusChanged(HWND hwnd, HWND focusWindow, BOOL focusReceived) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (NULL != self->callbacks && NULL != self->callbacks->focusChanged) { self->callbacks->focusChanged(self->object, hwnd, focusWindow, focusReceived); } } static LRESULT Widget_OnCreate(HWND hwnd, CREATESTRUCT *createStruct) { Widget *self; WidgetCreateParam *createParam; if (NULL == createStruct) return -1; createParam = (WidgetCreateParam*)createStruct->lpCreateParams; if (NULL == createParam) return -1; self = (Widget*)malloc(sizeof(Widget)); if (NULL == self) return -1; SetLastError(ERROR_SUCCESS); if (!SetWindowLongPtr(hwnd, 0, (LONGX86)self) && ERROR_SUCCESS != GetLastError()) return -1; memset(self, 0, sizeof(Widget)); if (WM_NULL == WINAMP_WM_DIRECT_MOUSE_WHEEL) WINAMP_WM_DIRECT_MOUSE_WHEEL = RegisterWindowMessageW(L"WINAMP_WM_DIRECT_MOUSE_WHEEL"); self->type = createParam->type; self->callbacks = createParam->callbacks; Widget_Freeze(self); if (NULL != createParam->text) SendMessage(hwnd, WM_SETTEXT, 0, (LPARAM)createParam->text); MLSkinWindow2(Plugin_GetLibraryWindow(), hwnd, SKINNEDWND_TYPE_SCROLLWND, SWS_USESKINFONT | SWS_USESKINCOLORS | SWS_USESKINCURSORS); if (NULL != self->callbacks->init && FALSE == self->callbacks->init(hwnd, &self->object, createParam->param)) { return -1; } Widget_Thaw(self); return 0; } static void Widget_OnDestroy(HWND hwnd) { Widget *self; self = WIDGET(hwnd); SetWindowLongPtr(hwnd, 0, 0); if (NULL == self) return; Widget_Freeze(self); if (NULL != self->callbacks->destroy) self->callbacks->destroy(self->object, hwnd); String_Free(self->text); free(self); } static void Widget_OnPaint(HWND hwnd) { PAINTSTRUCT ps; if (NULL != BeginPaint(hwnd, &ps)) { if (FALSE == Widget_Paint(hwnd, ps.hdc, &ps.rcPaint, ps.fErase)) { COLORREF backColor, prevBackColor; backColor = Graphics_GetSkinColor(WADLG_WNDBG); prevBackColor = SetBkColor(ps.hdc, backColor); ExtTextOut(ps.hdc, 0, 0, ETO_OPAQUE, &ps.rcPaint, NULL, 0, NULL); SetBkColor(ps.hdc, prevBackColor); } EndPaint(hwnd, &ps); } } static void Widget_OnPrintClient(HWND hwnd, HDC hdc, UINT options) { RECT clientRect; if (GetClientRect(hwnd, &clientRect)) { Widget_Paint(hwnd, hdc, &clientRect, TRUE); } } static void Widget_OnWindowPosChanged(HWND hwnd, WINDOWPOS *windowPos) { if ((SWP_NOSIZE | SWP_NOMOVE) != ((SWP_NOSIZE | SWP_NOMOVE | SWP_FRAMECHANGED) & windowPos->flags)) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (FALSE != WIDGET_IS_FROZEN(self)) return; Widget_Layout(hwnd, 0 == (SWP_NOREDRAW & windowPos->flags)); } } static LRESULT Widget_OnSetText(HWND hwnd, LPCWSTR text) { Widget *self; WIDGET_RET_VAL(self, hwnd, FALSE); String_Free(self->text); if (NULL == text) self->text = NULL; else if (FALSE != IS_INTRESOURCE(text)) { WCHAR buffer[4096] = {0}; ResourceString_CopyTo(buffer, text, ARRAYSIZE(buffer)); self->text = String_Duplicate(buffer); } else self->text = String_Duplicate(text); return TRUE; } static LRESULT Widget_OnGetText(HWND hwnd, LPWSTR buffer, size_t bufferMax) { Widget *self; WIDGET_RET_VAL(self, hwnd, 0); return String_CopyTo(buffer, self->text, bufferMax); } static LRESULT Widget_OnGetTextLength(HWND hwnd) { Widget *self; WIDGET_RET_VAL(self, hwnd, 0); return ( NULL != self->text) ? lstrlenW(self->text) : 0; } static void Widget_OnSetFont(HWND hwnd, HFONT font, BOOL redraw) { Widget *self; WIDGET_RET_VOID(self, hwnd); self->font = font; if (NULL != redraw) InvalidateRect(hwnd, NULL, TRUE); } static HFONT Widget_OnGetFont(HWND hwnd) { Widget *self; WIDGET_RET_VAL(self, hwnd, NULL); return self->font; } static void Widget_OnVertScroll(HWND hwnd, INT actionLayout, INT trackPosition, HWND scrollBar) { Widget *self; WIDGET_RET_VOID(self, hwnd); Widget_ScrollBarAction(hwnd, SB_VERT, actionLayout, self->style->unitSize.cy, TRUE); } static void Widget_OnHorzScroll(HWND hwnd, INT actionLayout, INT trackPosition, HWND scrollBar) { Widget *self; WIDGET_RET_VOID(self, hwnd); Widget_ScrollBarAction(hwnd, SB_HORZ, actionLayout, self->style->unitSize.cx, TRUE); } static void Widget_OnMouseWheel(HWND hwnd, INT virtualKeys, INT distance, LONG pointer_s) { Widget *self; UINT wheelScroll; INT scrollLines; UINT windowStyle; INT barType; WIDGET_RET_VOID(self, hwnd); windowStyle = GetWindowStyle(hwnd); if (0 != (WS_VSCROLL & windowStyle)) barType = SB_VERT; else if (0 != (WS_HSCROLL & windowStyle)) barType = SB_HORZ; else return; if (FALSE == SystemParametersInfo(SPI_GETWHEELSCROLLLINES, 0, &wheelScroll, 0)) wheelScroll = 3; if (0 == wheelScroll) return; if (WHEEL_PAGESCROLL == wheelScroll) { RECT clientRect; GetClientRect(hwnd, &clientRect); if (SB_VERT == barType) wheelScroll = RECTHEIGHT(clientRect)/self->style->unitSize.cy; else wheelScroll = RECTWIDTH(clientRect)/self->style->unitSize.cx; } distance += self->wheelCarryover; scrollLines = distance * (INT)wheelScroll / WHEEL_DELTA; self->wheelCarryover = distance - scrollLines * WHEEL_DELTA / (INT)wheelScroll; if (FALSE != Widget_ScrollWindow(hwnd, (SB_HORZ == barType) ? -(scrollLines * self->style->unitSize.cx) : 0, (SB_VERT == barType) ? -(scrollLines * self->style->unitSize.cy) : 0, TRUE)) { } } static void Widget_OnMouseMove(HWND hwnd, unsigned int vKeys, long cursor_s) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (NULL != self->callbacks && NULL != self->callbacks->mouseMove) { BOOL processed; POINT cursor; POINTSTOPOINT(cursor, cursor_s); processed = self->callbacks->mouseMove(self->object, hwnd, vKeys, &cursor); if (FALSE == WIDGET_IS_MOUSE_MOVE_TRACKED(self)) { TRACKMOUSEEVENT trackMouse; trackMouse.cbSize = sizeof(trackMouse); trackMouse.dwFlags = TME_LEAVE; trackMouse.hwndTrack = hwnd; if (FALSE != TrackMouseEvent(&trackMouse)) WIDGET_SET_MOUSE_MOVE_TRACK(self); } if (FALSE != processed) return; } DefWindowProc(hwnd, WM_MOUSEMOVE, (WPARAM)vKeys, (LPARAM)cursor_s); } static void Widget_OnMouseLeave(HWND hwnd) { Widget *self; WIDGET_RET_VOID(self, hwnd); WIDGET_UNSET_MOUSE_MOVE_TRACK(self); if (NULL != self->callbacks && NULL != self->callbacks->mouseMove) { POINT cursor; cursor.x = 0xEFFFFFFF; cursor.y = 0xEFFFFFFF; if (FALSE != self->callbacks->mouseMove(self->object, hwnd, 0, &cursor)) return; } DefWindowProc(hwnd, WM_MOUSELEAVE, 0, 0L); } static void Widget_OnLeftButtonDown(HWND hwnd, unsigned int vKeys, long cursor_s) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (NULL != self->callbacks && NULL != self->callbacks->leftButtonDown) { POINT cursor; POINTSTOPOINT(cursor, cursor_s); if (FALSE != self->callbacks->leftButtonDown(self->object, hwnd, vKeys, &cursor)) return; } DefWindowProc(hwnd, WM_LBUTTONDOWN, (WPARAM)vKeys, (LPARAM)cursor_s); } static void Widget_OnLeftButtonUp(HWND hwnd, unsigned int vKeys, long cursor_s) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (NULL != self->callbacks && NULL != self->callbacks->leftButtonUp) { POINT cursor; POINTSTOPOINT(cursor, cursor_s); if (FALSE != self->callbacks->leftButtonUp(self->object, hwnd, vKeys, &cursor)) return; } DefWindowProc(hwnd, WM_LBUTTONUP, (WPARAM)vKeys, (LPARAM)cursor_s); } static void Widget_OnLeftButtonDblClk(HWND hwnd, unsigned int vKeys, long cursor_s) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (NULL != self->callbacks && NULL != self->callbacks->leftButtonDblClk) { POINT cursor; POINTSTOPOINT(cursor, cursor_s); if (FALSE != self->callbacks->leftButtonDblClk(self->object, hwnd, vKeys, &cursor)) return; } DefWindowProc(hwnd, WM_LBUTTONDBLCLK, (WPARAM)vKeys, (LPARAM)cursor_s); } static void Widget_OnRightButtonDown(HWND hwnd, unsigned int vKeys, long cursor_s) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (NULL != self->callbacks && NULL != self->callbacks->rightButtonDown) { POINT cursor; POINTSTOPOINT(cursor, cursor_s); if (FALSE != self->callbacks->rightButtonDown(self->object, hwnd, vKeys, &cursor)) return; } DefWindowProc(hwnd, WM_RBUTTONDOWN, (WPARAM)vKeys, (LPARAM)cursor_s); } static void Widget_OnRightButtonUp(HWND hwnd, unsigned int vKeys, long cursor_s) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (NULL != self->callbacks && NULL != self->callbacks->rightButtonUp) { POINT cursor; POINTSTOPOINT(cursor, cursor_s); if (FALSE != self->callbacks->rightButtonUp(self->object, hwnd, vKeys, &cursor)) return; } DefWindowProc(hwnd, WM_RBUTTONUP, (WPARAM)vKeys, (LPARAM)cursor_s); } static void Widget_OnKeyDown(HWND hwnd, unsigned int vKey, unsigned int flags) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (NULL != self->callbacks && NULL != self->callbacks->keyDown && FALSE != self->callbacks->keyDown(self->object, hwnd, vKey, flags)) { return; } DefWindowProc(hwnd, WM_KEYDOWN, (WPARAM)vKey, (LPARAM)flags); } static void Widget_OnKeyUp(HWND hwnd, unsigned int vKey, unsigned int flags) { Widget *self; self = WIDGET(hwnd); if (NULL != self && NULL != self->callbacks && NULL != self->callbacks->keyUp && FALSE != self->callbacks->keyUp(self->object, hwnd, vKey, flags)) { return; } DefWindowProc(hwnd, WM_KEYUP, (WPARAM)vKey, (LPARAM)flags); } static void Widget_OnChar(HWND hwnd, unsigned int vKey, unsigned int flags) { Widget *self; self = WIDGET(hwnd); if (NULL != self && NULL != self->callbacks && NULL != self->callbacks->character && FALSE != self->callbacks->character(self->object, hwnd, vKey, flags)) { return; } DefWindowProc(hwnd, WM_CHAR, (WPARAM)vKey, (LPARAM)flags); } static unsigned int Widget_OnGetDlgCode(HWND hwnd, unsigned int vKey, MSG *message) { Widget *self; self = WIDGET(hwnd); if (NULL != self && NULL != self->callbacks && NULL != self->callbacks->inputRequest) { return self->callbacks->inputRequest(self->object, hwnd, vKey, message); } return (unsigned int)DefWindowProc(hwnd, WM_GETDLGCODE, (WPARAM)vKey, (LPARAM)message); } static void Widget_OnSetFocus(HWND hwnd, HWND focusWindow) { Widget_FocusChanged(hwnd, focusWindow, TRUE); } static void Widget_OnKillFocus(HWND hwnd, HWND focusWindow) { Widget_FocusChanged(hwnd, focusWindow, FALSE); } static void Widget_OnContextMenu(HWND hwnd, HWND targetWindow, long cursor_s) { BOOL processed; Widget *self; WIDGET_RET_VOID(self, hwnd); if (NULL != self->callbacks && NULL != self->callbacks->contextMenu) { POINT cursor; POINTSTOPOINT(cursor, cursor_s); processed = self->callbacks->contextMenu(self->object, hwnd, targetWindow, &cursor); } else processed = FALSE; if (FALSE == processed) Widget_DefWindowProc(hwnd, WM_CONTEXTMENU, (WPARAM)targetWindow, (LPARAM)cursor_s); } static unsigned int Widget_OnGetType(HWND hwnd) { Widget *self; WIDGET_RET_VAL(self, hwnd, WIDGET_TYPE_UNKNOWN); return self->type; } static void* Widget_OnGetSelf(HWND hwnd) { Widget *self; WIDGET_RET_VAL(self, hwnd, NULL); return self->object; } static BOOL Widget_OnSetStyle(HWND hwnd, WidgetStyle *style) { Widget *self; BOOL styleChanged; WIDGET_RET_VAL(self, hwnd, FALSE); styleChanged = (self->style != style); self->style = style; if (FALSE != styleChanged) { if (NULL != WIDGETCALLBACKS(self)) { if (NULL != WIDGETCALLBACKS(self)->styleColorChanged) WIDGETCALLBACKS(self)->styleColorChanged(WIDGETOBJECT(self), hwnd, WIDGETSTYLE(self)); if (NULL != WIDGETCALLBACKS(self)->styleFontChanged) WIDGETCALLBACKS(self)->styleFontChanged(WIDGETOBJECT(self), hwnd, WIDGETSTYLE(self)); } } return TRUE; } static WidgetStyle * Widget_OnGetStyle(HWND hwnd) { Widget *self; WIDGET_RET_VAL(self, hwnd, NULL); return self->style; } static void Widget_OnStyleColorChanged(HWND hwnd) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (FALSE != WIDGET_IS_FROZEN(self)) return; if (NULL != WIDGETCALLBACKS(self) && NULL != WIDGETCALLBACKS(self)->styleColorChanged) { WIDGETCALLBACKS(self)->styleColorChanged(WIDGETOBJECT(self), hwnd, WIDGETSTYLE(self)); } } static void Widget_OnStyleFontChanged(HWND hwnd) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (FALSE != WIDGET_IS_FROZEN(self)) return; if (NULL != WIDGETCALLBACKS(self) && NULL != WIDGETCALLBACKS(self)->styleFontChanged) { WIDGETCALLBACKS(self)->styleFontChanged(WIDGETOBJECT(self), hwnd, WIDGETSTYLE(self)); } } static void Widget_OnFreeze(HWND hwnd, BOOL freeze) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (FALSE == freeze) Widget_Thaw(self); else Widget_Freeze(self); } static BOOL Widget_OnScroll(HWND hwnd, int dx, int dy, BOOL redraw) { return Widget_ScrollWindow(hwnd, dx, dy, redraw); } static LRESULT Widget_OnSetScrollPos(HWND hwnd, int dx, int dy, BOOL redraw) { if (0 != dx) dx = Widget_ScrollBarOffsetPos(hwnd, SB_HORZ, dx, redraw); if (0 != dy) dy = Widget_ScrollBarOffsetPos(hwnd, SB_VERT, dy, redraw); return (LRESULT)MAKELONG(dx, dy); } static void Widget_OnZoomSliderPosChanging(HWND hwnd, NMTRBTHUMBPOSCHANGING *sliderInfo) { Widget *self; WIDGET_RET_VOID(self, hwnd); if (NULL != WIDGETCALLBACKS(self) && NULL != WIDGETCALLBACKS(self)->zoomChanging) { WIDGETCALLBACKS(self)->zoomChanging(WIDGETOBJECT(self), hwnd, sliderInfo); } } static LRESULT Widget_OnNotify(HWND hwnd, NMHDR *notification) { Widget *self; self = WIDGET(hwnd); if (NULL != self && NULL != self->callbacks && NULL != self->callbacks->notify) { LRESULT result; if (FALSE != self->callbacks->notify(WIDGETOBJECT(self), hwnd, notification, &result)) return result; } return Widget_DefWindowProc(hwnd, WM_NOTIFY, (WPARAM)notification->idFrom, (LPARAM)notification); } static BOOL Widget_OnEnableChildrenScroll(HWND hwnd, BOOL enable) { Widget *self; BOOL previous; WIDGET_RET_VAL(self, hwnd, FALSE); previous = (FALSE == WIDGET_IS_CHILDREN_SCROLL_DISABLED(self)); if (FALSE != enable) WIDGET_UNSET_DISABLE_CHILDREN_SCROLL(self); else WIDGET_SET_DISABLE_CHILDREN_SCROLL(self); return previous; } static BOOL Widget_OnGetChildrenScrollEnabled(HWND hwnd) { Widget *self; WIDGET_RET_VAL(self, hwnd, FALSE); return (FALSE == WIDGET_IS_CHILDREN_SCROLL_DISABLED(self)); } static LRESULT CALLBACK Widget_WindowProc(HWND hwnd, unsigned int uMsg, WPARAM wParam, LPARAM lParam) { switch(uMsg) { case WM_CREATE: return Widget_OnCreate(hwnd, (CREATESTRUCT*)lParam); case WM_DESTROY: Widget_OnDestroy(hwnd); return 0; case WM_PAINT: Widget_OnPaint(hwnd); return 0; case WM_PRINTCLIENT: Widget_OnPrintClient(hwnd, (HDC)wParam, (UINT)lParam); return 0; case WM_PRINT: return 0; case WM_ERASEBKGND: return 0; case WM_WINDOWPOSCHANGED: Widget_OnWindowPosChanged(hwnd, (WINDOWPOS*)lParam); return 0; case WM_SIZE: return 0; case WM_MOVE: return 0; case WM_SETTEXT: return Widget_OnSetText(hwnd, (LPCWSTR)lParam); case WM_GETTEXT: return Widget_OnGetText(hwnd, (LPWSTR)lParam, (INT)wParam); case WM_GETTEXTLENGTH: return Widget_OnGetTextLength(hwnd); case WM_SETFONT: Widget_OnSetFont(hwnd, (HFONT)wParam, (BOOL)LOWORD(lParam)); return 0; case WM_GETFONT: return (LRESULT)Widget_OnGetFont(hwnd); case WM_VSCROLL: Widget_OnVertScroll(hwnd, LOWORD(wParam), (short)HIWORD(wParam), (HWND)lParam); return 0; case WM_HSCROLL: Widget_OnHorzScroll(hwnd, LOWORD(wParam), (short)HIWORD(wParam), (HWND)lParam); return 0; case WM_MOUSEWHEEL: Widget_OnMouseWheel(hwnd, LOWORD(wParam), (short)HIWORD(wParam), (LONG)lParam); return 0; case WM_MOUSEMOVE: Widget_OnMouseMove(hwnd, (unsigned int)wParam, (long)lParam); return 0; case WM_MOUSELEAVE: Widget_OnMouseLeave(hwnd); return 0; case WM_LBUTTONDOWN: Widget_OnLeftButtonDown(hwnd, (unsigned int)wParam, (long)lParam); return 0; case WM_LBUTTONUP: Widget_OnLeftButtonUp(hwnd, (unsigned int)wParam, (long)lParam); return 0; case WM_LBUTTONDBLCLK: Widget_OnLeftButtonDblClk(hwnd, (unsigned int)wParam, (long)lParam); return 0; case WM_RBUTTONDOWN: Widget_OnRightButtonDown(hwnd, (unsigned int)wParam, (long)lParam); return 0; case WM_RBUTTONUP: Widget_OnRightButtonUp(hwnd, (unsigned int)wParam, (long)lParam); return 0; case WM_KEYDOWN: Widget_OnKeyDown(hwnd, (unsigned int)wParam, (unsigned int)lParam); return 0; case WM_KEYUP: Widget_OnKeyUp(hwnd, (unsigned int)wParam, (unsigned int)lParam); return 0; case WM_CHAR: Widget_OnChar(hwnd, (unsigned int)wParam, (unsigned int)lParam); return 0; case WM_GETDLGCODE: return Widget_OnGetDlgCode(hwnd, (unsigned int)wParam, (MSG*)lParam); case WM_SETFOCUS: Widget_OnSetFocus(hwnd, (HWND)wParam); return 0; case WM_KILLFOCUS: Widget_OnKillFocus(hwnd, (HWND)wParam); return 0; case WM_CONTEXTMENU: Widget_OnContextMenu(hwnd, (HWND)wParam, (long)lParam); return 0; case WM_NOTIFY: return Widget_OnNotify(hwnd, (NMHDR*)lParam); case WIDGET_WM_GET_TYPE: return (LRESULT)Widget_OnGetType(hwnd); case WIDGET_WM_GET_SELF: return (LRESULT)Widget_OnGetSelf(hwnd); case WIDGET_WM_SET_STYLE: return Widget_OnSetStyle(hwnd, (WidgetStyle*)lParam); case WIDGET_WM_GET_STYLE: return (LRESULT)Widget_OnGetStyle(hwnd); case WIDGET_WM_STYLE_COLOR_CHANGED: Widget_OnStyleColorChanged(hwnd); return 0; case WIDGET_WM_STYLE_FONT_CHANGED: Widget_OnStyleFontChanged(hwnd); return 0; case WIDGET_WM_FREEZE: Widget_OnFreeze(hwnd, (BOOL)wParam); return 0; case WIDGET_WM_SET_SCROLL_POS: return Widget_OnSetScrollPos(hwnd, (short)LOWORD(lParam), (short)HIWORD(lParam), (BOOL)wParam); case WIDGET_WM_SCROLL: return Widget_OnScroll(hwnd, (short)LOWORD(lParam), (short)HIWORD(lParam), (BOOL)wParam); case WIDGET_WM_ZOOM_SLIDER_POS_CHANGING: Widget_OnZoomSliderPosChanging(hwnd, (NMTRBTHUMBPOSCHANGING*)lParam); return 0; case WIDGET_WM_ENABLE_CHILDREN_SCROLL: return Widget_OnEnableChildrenScroll(hwnd, (BOOL)lParam); case WIDGET_WM_GET_CHILDREN_SCROLL_ENABLED: return Widget_OnGetChildrenScrollEnabled(hwnd); } if (WINAMP_WM_DIRECT_MOUSE_WHEEL == uMsg && WM_NULL != WINAMP_WM_DIRECT_MOUSE_WHEEL) { Widget_OnMouseWheel(hwnd, LOWORD(wParam), (SHORT)HIWORD(wParam), (LONG)lParam); return TRUE; } return Widget_DefWindowProc(hwnd, uMsg, wParam, lParam); }