2014-10-19 13:44:27 -05:00
|
|
|
// 19 october 2014
|
|
|
|
#define UNICODE
|
|
|
|
#define _UNICODE
|
|
|
|
#define STRICT
|
|
|
|
#define STRICT_TYPED_ITEMIDS
|
|
|
|
// get Windows version right; right now Windows XP
|
|
|
|
#define WINVER 0x0501
|
|
|
|
#define _WIN32_WINNT 0x0501
|
|
|
|
#define _WIN32_WINDOWS 0x0501 /* according to Microsoft's winperf.h */
|
|
|
|
#define _WIN32_IE 0x0600 /* according to Microsoft's sdkddkver.h */
|
|
|
|
#define NTDDI_VERSION 0x05010000 /* according to Microsoft's sdkddkver.h */
|
|
|
|
#include <windows.h>
|
|
|
|
#include <commctrl.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <uxtheme.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <wchar.h>
|
|
|
|
#include <windowsx.h>
|
|
|
|
#include <vsstyle.h>
|
|
|
|
#include <vssym32.h>
|
|
|
|
|
|
|
|
// #qo LIBS: user32 kernel32 gdi32
|
|
|
|
|
|
|
|
#define tableWindowClass L"gouitable"
|
|
|
|
|
|
|
|
struct table {
|
|
|
|
HWND hwnd;
|
|
|
|
HFONT defaultFont;
|
|
|
|
HFONT font;
|
|
|
|
intptr_t selected;
|
2014-10-19 18:40:23 -05:00
|
|
|
intptr_t count;
|
2014-10-19 21:20:53 -05:00
|
|
|
intptr_t firstVisible;
|
2014-10-19 22:33:08 -05:00
|
|
|
intptr_t pagesize;
|
2014-10-19 13:44:27 -05:00
|
|
|
};
|
|
|
|
|
2014-10-19 22:33:08 -05:00
|
|
|
static void vscroll(struct table *t, WPARAM wParam)
|
|
|
|
{
|
2014-10-19 22:48:25 -05:00
|
|
|
HFONT thisfont, prevfont;
|
|
|
|
TEXTMETRICW tm;
|
|
|
|
HDC dc;
|
2014-10-19 22:33:08 -05:00
|
|
|
SCROLLINFO si;
|
|
|
|
intptr_t newpos;
|
|
|
|
|
2014-10-19 22:48:25 -05:00
|
|
|
// TODO split into a function
|
|
|
|
dc = GetDC(t->hwnd);
|
|
|
|
if (dc == NULL)
|
|
|
|
abort();
|
|
|
|
thisfont = t->font; // in case WM_SETFONT happens before we return
|
|
|
|
prevfont = (HFONT) SelectObject(dc, thisfont);
|
|
|
|
if (prevfont == NULL)
|
|
|
|
abort();
|
|
|
|
if (GetTextMetricsW(dc, &tm) == 0)
|
|
|
|
abort();
|
|
|
|
if (SelectObject(dc, prevfont) != (HGDIOBJ) (thisfont))
|
|
|
|
abort();
|
|
|
|
if (ReleaseDC(t->hwnd, dc) == 0)
|
|
|
|
abort();
|
|
|
|
|
2014-10-19 22:33:08 -05:00
|
|
|
ZeroMemory(&si, sizeof (SCROLLINFO));
|
|
|
|
si.cbSize = sizeof (SCROLLINFO);
|
|
|
|
si.fMask = SIF_POS | SIF_TRACKPOS;
|
|
|
|
if (GetScrollInfo(t->hwnd, SB_VERT, &si) == 0)
|
|
|
|
abort();
|
|
|
|
|
|
|
|
newpos = t->firstVisible;
|
|
|
|
switch (LOWORD(wParam)) {
|
|
|
|
case SB_TOP:
|
|
|
|
newpos = 0;
|
|
|
|
break;
|
|
|
|
case SB_BOTTOM:
|
|
|
|
newpos = t->count - t->pagesize;
|
|
|
|
break;
|
|
|
|
case SB_LINEUP:
|
|
|
|
newpos--;
|
|
|
|
break;
|
|
|
|
case SB_LINEDOWN:
|
|
|
|
newpos++;
|
|
|
|
break;
|
|
|
|
case SB_PAGEUP:
|
|
|
|
newpos -= t->pagesize;
|
|
|
|
break;
|
|
|
|
case SB_PAGEDOWN:
|
|
|
|
newpos += t->pagesize;
|
|
|
|
break;
|
|
|
|
case SB_THUMBPOSITION:
|
|
|
|
newpos = (intptr_t) (si.nPos);
|
|
|
|
break;
|
|
|
|
case SB_THUMBTRACK:
|
|
|
|
newpos = (intptr_t) (si.nTrackPos);
|
|
|
|
}
|
|
|
|
if (newpos < 0)
|
|
|
|
newpos = 0;
|
|
|
|
if (newpos > (t->count - t->pagesize))
|
|
|
|
newpos = (t->count - t->pagesize);
|
|
|
|
|
|
|
|
// negative because ScrollWindowEx() is "backwards"
|
2014-10-19 22:48:25 -05:00
|
|
|
if (ScrollWindowEx(t->hwnd, 0, (-(newpos - t->firstVisible)) * tm.tmHeight,
|
2014-10-19 22:33:08 -05:00
|
|
|
NULL, NULL, NULL, NULL,
|
|
|
|
SW_ERASE | SW_INVALIDATE) == ERROR)
|
|
|
|
abort();
|
|
|
|
t->firstVisible = newpos;
|
|
|
|
|
|
|
|
ZeroMemory(&si, sizeof (SCROLLINFO));
|
|
|
|
si.cbSize = sizeof (SCROLLINFO);
|
|
|
|
si.fMask = SIF_PAGE | SIF_POS | SIF_RANGE;
|
|
|
|
si.nPage = t->pagesize;
|
|
|
|
si.nMin = 0;
|
|
|
|
si.nMax = t->count - 1; // nMax is inclusive
|
|
|
|
si.nPos = t->firstVisible;
|
|
|
|
SetScrollInfo(t->hwnd, SB_VERT, &si, TRUE);
|
|
|
|
}
|
|
|
|
|
2014-10-19 22:48:25 -05:00
|
|
|
static void resize(struct table *t)
|
|
|
|
{
|
|
|
|
HFONT thisfont, prevfont;
|
|
|
|
TEXTMETRICW tm;
|
|
|
|
HDC dc;
|
|
|
|
RECT r;
|
|
|
|
SCROLLINFO si;
|
|
|
|
|
|
|
|
if (GetClientRect(t->hwnd, &r) == 0)
|
|
|
|
abort();
|
|
|
|
dc = GetDC(t->hwnd);
|
|
|
|
if (dc == NULL)
|
|
|
|
abort();
|
|
|
|
thisfont = t->font; // in case WM_SETFONT happens before we return
|
|
|
|
prevfont = (HFONT) SelectObject(dc, thisfont);
|
|
|
|
if (prevfont == NULL)
|
|
|
|
abort();
|
|
|
|
if (GetTextMetricsW(dc, &tm) == 0)
|
|
|
|
abort();
|
|
|
|
if (SelectObject(dc, prevfont) != (HGDIOBJ) (thisfont))
|
|
|
|
abort();
|
|
|
|
if (ReleaseDC(t->hwnd, dc) == 0)
|
|
|
|
abort();
|
|
|
|
t->pagesize = (r.bottom - r.top) / tm.tmHeight;
|
|
|
|
ZeroMemory(&si, sizeof (SCROLLINFO));
|
|
|
|
si.cbSize = sizeof (SCROLLINFO);
|
|
|
|
si.fMask = SIF_RANGE | SIF_PAGE;
|
|
|
|
si.nMin = 0;
|
|
|
|
si.nMax = t->count - 1;
|
|
|
|
si.nPage = t->pagesize;
|
|
|
|
SetScrollInfo(t->hwnd, SB_VERT, &si, TRUE);
|
|
|
|
}
|
|
|
|
|
2014-10-19 20:49:27 -05:00
|
|
|
static void drawItems(struct table *t, HDC dc, RECT cliprect)
|
2014-10-19 13:44:27 -05:00
|
|
|
{
|
2014-10-19 19:01:01 -05:00
|
|
|
HFONT thisfont, prevfont;
|
2014-10-19 13:44:27 -05:00
|
|
|
TEXTMETRICW tm;
|
|
|
|
LONG y;
|
|
|
|
intptr_t i;
|
|
|
|
RECT r;
|
2014-10-19 20:49:27 -05:00
|
|
|
intptr_t first, last;
|
2014-10-19 21:20:53 -05:00
|
|
|
POINT prevOrigin;
|
2014-10-19 13:44:27 -05:00
|
|
|
|
2014-10-19 20:49:27 -05:00
|
|
|
// TODO eliminate the need (only use cliprect)
|
2014-10-19 13:44:27 -05:00
|
|
|
if (GetClientRect(t->hwnd, &r) == 0)
|
|
|
|
abort();
|
2014-10-19 20:49:27 -05:00
|
|
|
|
2014-10-19 19:01:01 -05:00
|
|
|
thisfont = t->font; // in case WM_SETFONT happens before we return
|
|
|
|
prevfont = (HFONT) SelectObject(dc, thisfont);
|
2014-10-19 13:44:27 -05:00
|
|
|
if (prevfont == NULL)
|
|
|
|
abort();
|
|
|
|
if (GetTextMetricsW(dc, &tm) == 0)
|
|
|
|
abort();
|
2014-10-19 20:49:27 -05:00
|
|
|
|
2014-10-19 21:20:53 -05:00
|
|
|
// adjust the clip rect and the viewport so that (0, 0) is always the first item
|
|
|
|
if (OffsetRect(&cliprect, 0, t->firstVisible * tm.tmHeight) == 0)
|
|
|
|
abort();
|
|
|
|
if (GetWindowOrgEx(dc, &prevOrigin) == 0)
|
|
|
|
abort();
|
|
|
|
if (SetWindowOrgEx(dc, prevOrigin.x, prevOrigin.y + (t->firstVisible * tm.tmHeight), NULL) == 0)
|
|
|
|
abort();
|
|
|
|
|
2014-10-19 20:49:27 -05:00
|
|
|
// see http://blogs.msdn.com/b/oldnewthing/archive/2003/07/29/54591.aspx and http://blogs.msdn.com/b/oldnewthing/archive/2003/07/30/54600.aspx
|
|
|
|
first = cliprect.top / tm.tmHeight;
|
|
|
|
if (first < 0)
|
|
|
|
first = 0;
|
|
|
|
last = (cliprect.bottom + tm.tmHeight - 1) / tm.tmHeight;
|
|
|
|
if (last >= t->count)
|
|
|
|
last = t->count;
|
|
|
|
|
2014-10-19 22:51:42 -05:00
|
|
|
y = first * tm.tmHeight;
|
2014-10-19 20:49:27 -05:00
|
|
|
for (i = first; i < last; i++) {
|
2014-10-19 18:02:18 -05:00
|
|
|
RECT rsel;
|
|
|
|
HBRUSH background;
|
|
|
|
|
2014-10-19 13:44:27 -05:00
|
|
|
// TODO check errors
|
|
|
|
// TODO verify correct colors
|
2014-10-19 18:02:18 -05:00
|
|
|
rsel.left = r.left;
|
|
|
|
rsel.top = y;
|
|
|
|
rsel.right = r.right - r.left;
|
|
|
|
rsel.bottom = y + tm.tmHeight;
|
|
|
|
background = (HBRUSH) (COLOR_WINDOW + 1);
|
2014-10-19 13:44:27 -05:00
|
|
|
if (t->selected == i) {
|
2014-10-19 18:02:18 -05:00
|
|
|
background = (HBRUSH) (COLOR_HIGHLIGHT + 1);
|
2014-10-19 13:44:27 -05:00
|
|
|
SetTextColor(dc, GetSysColor(COLOR_HIGHLIGHTTEXT));
|
2014-10-19 18:02:18 -05:00
|
|
|
} else
|
2014-10-19 13:44:27 -05:00
|
|
|
SetTextColor(dc, GetSysColor(COLOR_WINDOWTEXT));
|
2014-10-19 18:02:18 -05:00
|
|
|
FillRect(dc, &rsel, background);
|
2014-10-19 13:44:27 -05:00
|
|
|
SetBkMode(dc, TRANSPARENT);
|
|
|
|
TextOutW(dc, r.left, y, L"Item", 4);
|
|
|
|
y += tm.tmHeight;
|
|
|
|
}
|
2014-10-19 20:49:27 -05:00
|
|
|
|
2014-10-19 21:20:53 -05:00
|
|
|
// reset everything
|
|
|
|
if (SetWindowOrgEx(dc, prevOrigin.x, prevOrigin.y, NULL) == 0)
|
|
|
|
abort();
|
2014-10-19 19:01:01 -05:00
|
|
|
if (SelectObject(dc, prevfont) != (HGDIOBJ) (thisfont))
|
2014-10-19 13:44:27 -05:00
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
static LRESULT CALLBACK tableWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
struct table *t;
|
|
|
|
HDC dc;
|
|
|
|
PAINTSTRUCT ps;
|
|
|
|
|
|
|
|
t = (struct table *) GetWindowLongPtrW(hwnd, GWLP_USERDATA);
|
|
|
|
if (t == NULL) {
|
|
|
|
t = (struct table *) malloc(sizeof (struct table));
|
|
|
|
if (t == NULL)
|
|
|
|
abort();
|
|
|
|
ZeroMemory(t, sizeof (struct table));
|
|
|
|
t->hwnd = hwnd;
|
|
|
|
// TODO this should be a global
|
|
|
|
t->defaultFont = (HFONT) GetStockObject(SYSTEM_FONT);
|
|
|
|
if (t->defaultFont == NULL)
|
|
|
|
abort();
|
|
|
|
t->font = t->defaultFont;
|
2014-10-19 18:40:23 -05:00
|
|
|
t->selected = 5;t->count=100;//TODO
|
2014-10-19 13:44:27 -05:00
|
|
|
SetWindowLongPtrW(hwnd, GWLP_USERDATA, (LONG_PTR) t);
|
|
|
|
}
|
|
|
|
switch (uMsg) {
|
|
|
|
case WM_PAINT:
|
|
|
|
dc = BeginPaint(hwnd, &ps);
|
|
|
|
if (dc == NULL)
|
|
|
|
abort();
|
2014-10-19 20:49:27 -05:00
|
|
|
drawItems(t, dc, ps.rcPaint);
|
2014-10-19 13:44:27 -05:00
|
|
|
EndPaint(hwnd, &ps);
|
|
|
|
return 0;
|
|
|
|
case WM_SETFONT:
|
|
|
|
t->font = (HFONT) wParam;
|
|
|
|
if (t->font == NULL)
|
|
|
|
t->font = t->defaultFont;
|
|
|
|
if (LOWORD(lParam) != FALSE)
|
|
|
|
; // TODO
|
|
|
|
return 0;
|
|
|
|
case WM_GETFONT:
|
|
|
|
return (LRESULT) t->font;
|
2014-10-19 22:33:08 -05:00
|
|
|
case WM_VSCROLL:
|
|
|
|
vscroll(t, wParam);
|
|
|
|
return 0;
|
|
|
|
case WM_SIZE:
|
2014-10-19 22:48:25 -05:00
|
|
|
resize(t);
|
2014-10-19 22:33:08 -05:00
|
|
|
return 0;
|
2014-10-19 13:44:27 -05:00
|
|
|
default:
|
|
|
|
return DefWindowProcW(hwnd, uMsg, wParam, lParam);
|
|
|
|
}
|
|
|
|
abort();
|
|
|
|
return 0; // unreached
|
|
|
|
}
|
|
|
|
|
|
|
|
void makeTableWindowClass(void)
|
|
|
|
{
|
|
|
|
WNDCLASSW wc;
|
|
|
|
|
|
|
|
ZeroMemory(&wc, sizeof (WNDCLASSW));
|
|
|
|
wc.lpszClassName = tableWindowClass;
|
|
|
|
wc.lpfnWndProc = tableWndProc;
|
|
|
|
wc.hCursor = LoadCursorW(NULL, IDC_ARROW);
|
|
|
|
wc.hIcon = LoadIconW(NULL, IDI_APPLICATION);
|
|
|
|
wc.hbrBackground = (HBRUSH) (COLOR_WINDOW + 1); // TODO correct?
|
2014-10-19 20:49:27 -05:00
|
|
|
wc.style = CS_HREDRAW | CS_VREDRAW;
|
2014-10-19 13:44:27 -05:00
|
|
|
wc.hInstance = GetModuleHandle(NULL);
|
|
|
|
if (RegisterClassW(&wc) == 0)
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
HWND mainwin;
|
|
|
|
MSG msg;
|
|
|
|
|
|
|
|
makeTableWindowClass();
|
|
|
|
mainwin = CreateWindowExW(0,
|
|
|
|
tableWindowClass, L"Main Window",
|
|
|
|
WS_OVERLAPPEDWINDOW | WS_HSCROLL | WS_VSCROLL,
|
|
|
|
CW_USEDEFAULT, CW_USEDEFAULT,
|
|
|
|
400, 400,
|
|
|
|
NULL, NULL, GetModuleHandle(NULL), NULL);
|
|
|
|
if (mainwin == NULL)
|
|
|
|
abort();
|
|
|
|
ShowWindow(mainwin, SW_SHOWDEFAULT);
|
|
|
|
if (UpdateWindow(mainwin) == 0)
|
|
|
|
abort();
|
|
|
|
while (GetMessageW(&msg, NULL, 0, 0) > 0) {
|
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessageW(&msg);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|