andlabs-ui/wintable/select.h

267 lines
8.4 KiB
C
Raw Normal View History

2014-12-13 12:56:31 -06:00
// 13 december 2014
// damn winsock
static void doselect(struct table *t, intptr_t row, intptr_t column)
{
RECT r, client;
intptr_t oldrow;
LONG width, height;
struct rowcol rc;
BOOL dovscroll;
intptr_t i;
intptr_t xpos;
LONG clientWidth;
// check existing selection to see if it's valid
if (t->selectedRow == -1 && t->selectedColumn != -1)
2014-12-16 10:50:53 -06:00
panic("sanity check failure: old Table selection invalid (row == -1, column != -1)");
if (t->selectedRow != -1 && t->selectedColumn == -1)
2014-12-16 10:50:53 -06:00
panic("sanity check failure: old Table selection invalid (row != -1, column == -1)");
if (t->selectedRow >= t->count)
panic("sanity check failure: old Table selection invalid (row out of range)");
if (t->selectedColumn >= t->nColumns)
2014-12-16 10:50:53 -06:00
panic("sanity check failure: old Table selection invalid (column out of range)");
oldrow = t->selectedRow;
2014-12-13 12:56:31 -06:00
t->selectedRow = row;
t->selectedColumn = column;
// check new selection to see if it's valid
if (t->selectedRow == -1 && t->selectedColumn != -1)
2014-12-16 10:50:53 -06:00
panic("sanity check failure: new Table selection invalid (row == -1, column != -1)");
if (t->selectedRow != -1 && t->selectedColumn == -1)
2014-12-16 10:50:53 -06:00
panic("sanity check failure: new Table selection invalid (row != -1, column == -1)");
if (t->selectedRow >= t->count)
panic("sanity check failure: new Table selection invalid (row out of range)");
if (t->selectedColumn >= t->nColumns)
2014-12-16 10:50:53 -06:00
panic("sanity check failure: new Table selection invalid (column out of range)");
2014-12-16 19:19:06 -06:00
// do this even if we don't scroll before; noScroll depends on it
if (GetClientRect(t->hwnd, &client) == 0)
panic("error getting Table client rect in doselect()");
client.top += t->headerHeight;
height = rowht(t);
2014-12-16 19:19:06 -06:00
// only scroll if we selected something
if (t->selectedRow == -1 || t->selectedColumn == -1)
goto noScroll;
// first vertically scroll to the new row to make it fully visible (or as visible as possible)
if (t->selectedRow < t->vscrollpos)
vscrollto(t, t->selectedRow);
else {
rc.row = t->selectedRow;
rc.column = t->selectedColumn;
// first assume entirely outside the client area
dovscroll = TRUE;
if (rowColumnToClientRect(t, rc, &r))
// partially outside the client area?
if (r.bottom <= client.bottom) // <= here since we are comparing bottoms (which are the first pixels outside the rectangle)
dovscroll = FALSE;
if (dovscroll)
vscrollto(t, t->selectedRow - t->vpagesize + 1); // + 1 because apparently just t->selectedRow - t->vpagesize results in no scrolling (t->selectedRow - t->vpagesize == t->vscrollpos)...
}
// now see if the cell we want is to the left of offscreen, in which case scroll to its x-position
xpos = 0;
for (i = 0; i < t->selectedColumn; i++)
xpos += columnWidth(t, i);
if (xpos < t->hscrollpos)
hscrollto(t, xpos);
else {
// if the full cell is not visible, scroll to the right just enough to make it fully visible (or as visible as possible)
width = columnWidth(t, t->selectedColumn);
clientWidth = client.right - client.left;
if (xpos + width > t->hscrollpos + clientWidth) // > because both sides deal with the first pixel outside
// if the column is too wide, then just make it occupy the whole visible area (left-aligned)
2014-12-14 16:29:32 -06:00
if (width > clientWidth) // TODO >= ?
hscrollto(t, xpos);
else
2014-12-14 17:06:08 -06:00
// TODO don't use t->hpagesize here? depends if other code uses it
hscrollto(t, (xpos + width) - t->hpagesize);
}
noScroll:
// now redraw the old and new /rows/
// we do this after scrolling so the rectangles to be invalidated make sense
r.left = client.left;
r.right = client.right;
if (oldrow != -1 && oldrow >= t->vscrollpos) {
r.top = client.top + ((oldrow - t->vscrollpos) * height);
r.bottom = r.top + height;
if (InvalidateRect(t->hwnd, &r, TRUE) == 0)
panic("error queueing previously selected row for redraw in doselect()");
}
// t->selectedRow must be visible by this point; we scrolled to it
if (t->selectedRow != -1 && t->selectedRow != oldrow) {
r.top = client.top + ((t->selectedRow - t->vscrollpos) * height);
r.bottom = r.top + height;
if (InvalidateRect(t->hwnd, &r, TRUE) == 0)
panic("error queueing newly selected row for redraw in doselect()");
}
2015-02-17 11:56:46 -06:00
// TODO what about deselect/defocus?
// TODO notify on the old row too?
NotifyWinEvent(EVENT_OBJECT_SELECTION, t->hwnd, OBJID_CLIENT, t->selectedRow);
// TODO send EVENT_OBJECT_STATECHANGED too?
2015-02-17 11:56:46 -06:00
// TODO send EVENT_OBJECT_FOCUS
2014-12-13 12:56:31 -06:00
}
// TODO make this needless
HANDLER(checkboxMouseDownHandler);
2014-12-13 12:56:31 -06:00
// TODO which WM_xBUTTONDOWNs?
HANDLER(mouseDownSelectHandler)
{
struct rowcol rc;
// TODO separate this from here
// TODO other mouse buttons?
// don't check SetFocus()'s error (http://stackoverflow.com/questions/24073695/winapi-can-setfocus-return-null-without-an-error-because-thats-what-im-see)
2015-02-17 11:59:28 -06:00
// TODO NotifyWinEvent() here?
SetFocus(t->hwnd);
2014-12-13 12:56:31 -06:00
rc = lParamToRowColumn(t, lParam);
// don't check if lParamToRowColumn() returned row -1 or column -1; we want deselection behavior
doselect(t, rc.row, rc.column);
// TODO separate this from here
checkboxMouseDownHandler(t, uMsg, wParam, lParam, lResult);
2014-12-13 12:56:31 -06:00
*lResult = 0;
return TRUE;
}
/*
the routine below is intended to simulate the comctl32.dll listview keyboard navigation rules, at least as far as vertical navigation is concerned.
2014-12-16 01:58:31 -06:00
horizontal scrolling is different because unlike the comctl32 listview, we say that a single column in each row has the keyboard focus, so left and right navigate between columns here, instead of scrolling left/right by pixels.
TODO provide an override for scrolling by pixels?
TODO any other keyboard shortcuts?
TODO browser keys
TODO media navigation keys
TODO XBUTTON1/2?
TODO clear keys?
keyboard selection behaviors of the windows 7 listview:
with 100 items (0-99), the window currently shows items 30 through 47 as well as having item 48 partially visible
- item 30:
- page up -> item 13
- page down -> item 47
- item 31:
- page up -> item 30
- page down -> item 47
- item 42:
- page up -> item 30
- page down -> item 47
- item 46:
- page up -> item 30
- page down -> item 47
- item 47:
- page up: -> item 30
- page down: -> item 64
2014-12-15 19:03:46 -06:00
when nothing is selected:
- down selects item 0 regardless of scroll
- up selects nothing regardless of scroll
2014-12-15 22:48:09 -06:00
- page down selects the last fully visible item depending on scroll
- so with the above configuration:
- item 0 -> item 17
- item 30 -> item 47
- item 80 -> item 97
- page up selects item 0 regardless of scroll
- home selects item 0 regardless of scroll
- end selects the last item regardless of scroll
for left and right we will simulate up and down, respectively (so right selects row 0 column 0); remember that you can't have either row or column be -1 but not both
2014-12-17 13:58:39 -06:00
TODO what happens if page up and page down are pressed with an item selected and the scroll in a different position?
*/
HANDLER(keyDownSelectHandler)
{
intptr_t row;
intptr_t column;
if (t->count == 0 || t->nColumns == 0) // no items to select
return FALSE;
row = t->selectedRow;
column = t->selectedColumn;
switch (wParam) {
case VK_UP:
if (row == -1)
return FALSE;
row--;
if (row < 0)
row = 0;
break;
case VK_DOWN:
if (row == -1) {
row = 0;
column = 0;
} else {
row++;
if (row >= t->count)
row = t->count - 1;
}
break;
case VK_LEFT:
if (column == -1)
return FALSE;
column--;
if (column < 0)
column = 0;
break;
case VK_RIGHT:
if (column == -1) {
row = 0;
column = 0;
} else {
column++;
if (column >= t->nColumns)
column = t->nColumns - 1;
}
break;
case VK_HOME:
row = 0;
if (column == -1)
column = 0;
break;
case VK_END:
row = t->count - 1;
if (column == -1)
column = 0;
break;
case VK_PRIOR:
if (row == -1) {
row = 0;
column = 0;
} else {
row = t->vscrollpos;
if (row == t->selectedRow)
// TODO investigate why the - 1 is needed here and below
// TODO if this is a misunderstanding of how t->vpagesize works, figure out what happens if there is no partially visible row, and what is supposed to happen
row -= t->vpagesize - 1;
if (row < 0)
row = 0;
}
break;
case VK_NEXT:
if (row == -1) {
2014-12-17 13:58:39 -06:00
row = t->vscrollpos + t->vpagesize - 1;
// TODO ensusre this is the case with the real list view
if (row >= t->count)
row = t->count - 1;
column = 0;
} else {
row = t->vscrollpos + t->vpagesize - 1;
if (row == t->selectedRow)
row += t->vpagesize - 1;
if (row >= t->count)
row = t->count - 1;
}
break;
default:
return FALSE;
}
doselect(t, row, column);
*lResult = 0;
return TRUE;
}