aboutsummaryrefslogtreecommitdiff
path: root/Src/Plugins/Library/ml_devices/widget.cpp
diff options
context:
space:
mode:
authorJef <jef@targetspot.com>2024-09-24 08:54:57 -0400
committerJef <jef@targetspot.com>2024-09-24 08:54:57 -0400
commit20d28e80a5c861a9d5f449ea911ab75b4f37ad0d (patch)
tree12f17f78986871dd2cfb0a56e5e93b545c1ae0d0 /Src/Plugins/Library/ml_devices/widget.cpp
parent537bcbc86291b32fc04ae4133ce4d7cac8ebe9a7 (diff)
downloadwinamp-20d28e80a5c861a9d5f449ea911ab75b4f37ad0d.tar.gz
Initial community commit
Diffstat (limited to 'Src/Plugins/Library/ml_devices/widget.cpp')
-rw-r--r--Src/Plugins/Library/ml_devices/widget.cpp1284
1 files changed, 1284 insertions, 0 deletions
diff --git a/Src/Plugins/Library/ml_devices/widget.cpp b/Src/Plugins/Library/ml_devices/widget.cpp
new file mode 100644
index 00000000..c5e99c6e
--- /dev/null
+++ b/Src/Plugins/Library/ml_devices/widget.cpp
@@ -0,0 +1,1284 @@
+#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);
+}
+