mirror of
https://github.com/WinampDesktop/winamp.git
synced 2024-12-20 19:36:00 +01:00
2151 lines
50 KiB
C++
2151 lines
50 KiB
C++
|
#pragma warning(disable:4355)
|
||
|
|
||
|
#include <precomp.h>
|
||
|
|
||
|
#include "listwnd.h"
|
||
|
|
||
|
#include <bfc/wasabi_std.h>
|
||
|
#include <api/wnd/usermsg.h>
|
||
|
#include <bfc/ptrlist.h>
|
||
|
#include <tataki/color/skinclr.h>
|
||
|
#include <api/wnd/notifmsg.h>
|
||
|
|
||
|
#include <bfc/assert.h>
|
||
|
|
||
|
#include <api/locales/xlatstr.h>
|
||
|
#include <api/wnd/accessible.h>
|
||
|
#include <bfc/string/StringW.h>
|
||
|
#include <new>
|
||
|
#define DEF_TEXT_SIZE 14 // Default text size
|
||
|
|
||
|
#define ITEMLIST_INC 4092
|
||
|
#define LISTWND_DRAG_TIMERID 0x8972
|
||
|
#define LISTWND_DRAG_MARGIN 12
|
||
|
#define LISTWND_DRAG_TIMERDELAY 100
|
||
|
#define DRAGSKIP_START 5
|
||
|
|
||
|
#define X_SHIFT 2
|
||
|
#define Y_SHIFT 2
|
||
|
#define DRAG_THRESHOLD 4
|
||
|
#define COLUMNS_DEFAULT_HEIGHT 12
|
||
|
#define COLUMNS_DEFAULT_WIDTH 96
|
||
|
#define COLUMNS_MARGIN 2
|
||
|
#define COLUMNS_RESIZE_THRESHOLD 4
|
||
|
#define COLUMNS_MIN_WIDTH 1
|
||
|
#define COLSEPHEIGHT 1
|
||
|
|
||
|
static SkinColor textColor(L"studio.list.text");// todo: have own color in skin.cpp
|
||
|
static SkinColor bgcolor(L"studio.list.background");// todo: have own color in skin.cpp
|
||
|
static SkinColor color_item_selected_fg(L"studio.list.item.selected.fg"); // RGB(0, 0, 128)
|
||
|
static SkinColor color_item_selected(L"studio.list.item.selected"); // RGB(0, 0, 128)
|
||
|
static SkinColor color_item_focused(L"studio.list.item.focused");// RGB(0, 128, 128)
|
||
|
static SkinColor color_item_focusrect(L"studio.list.item.focused");// RGB(0, 128, 128)
|
||
|
static SkinColor color_headers(L"studio.list.column.background");//RGB(0, 152, 208)
|
||
|
static SkinColor columnTextColor(L"studio.list.column.text");
|
||
|
static SkinColor columnSepColor(L"studio.list.column.separator");
|
||
|
|
||
|
typedef struct
|
||
|
{
|
||
|
wchar_t *label;
|
||
|
int column;
|
||
|
} listSubitemStruct;
|
||
|
|
||
|
class listItem
|
||
|
{
|
||
|
friend ListWnd;
|
||
|
public:
|
||
|
ListWnd *getList() const { return listwnd; }
|
||
|
protected:
|
||
|
listItem()
|
||
|
{
|
||
|
data = 0;
|
||
|
subitems = NULL;
|
||
|
listwnd = NULL;
|
||
|
}
|
||
|
~listItem()
|
||
|
{
|
||
|
if (subitems != NULL)
|
||
|
{
|
||
|
for (int i=0;i<subitems->getNumItems();i++)
|
||
|
{
|
||
|
listSubitemStruct *subitem = subitems->enumItem(i);
|
||
|
if (subitem->label)
|
||
|
FREE(subitem->label);
|
||
|
}
|
||
|
subitems->freeAll();
|
||
|
delete subitems;
|
||
|
}
|
||
|
}
|
||
|
void setList(ListWnd *newlist) { listwnd=newlist; }
|
||
|
|
||
|
StringW label;
|
||
|
LPARAM data;
|
||
|
PtrList<listSubitemStruct> *subitems;
|
||
|
ListWnd *listwnd;
|
||
|
AutoSkinBitmap icon;
|
||
|
};
|
||
|
|
||
|
class CompareListItem {
|
||
|
public:
|
||
|
static int compareItem(listItem *p1, listItem *p2);
|
||
|
};
|
||
|
|
||
|
int CompareListItem::compareItem(listItem *p1, listItem *p2) {
|
||
|
return p1->getList()->sortCompareItem(p1, p2);
|
||
|
}
|
||
|
|
||
|
|
||
|
XMLParamPair ListWnd::params[] = {
|
||
|
{LIST_ANTIALIAS, L"ANTIALIAS"},
|
||
|
{LIST_BACKGROUND, L"BACKGROUND"},
|
||
|
{LIST_TILE, L"TILE"},
|
||
|
{LIST_NOCOLHEADER, L"NOCOLHEADER"},
|
||
|
};
|
||
|
|
||
|
ListWnd::ListWnd()
|
||
|
: selItemList(this)
|
||
|
{
|
||
|
xuihandle = newXuiHandle();
|
||
|
CreateXMLParameters(xuihandle);
|
||
|
|
||
|
if (WASABI_API_SKIN->skin_getVersion() >= 1) {
|
||
|
textColor.setElementName(L"wasabi.list.text");
|
||
|
textColor.setColorGroup(L"");
|
||
|
bgcolor.setElementName(L"wasabi.list.background");
|
||
|
bgcolor.setColorGroup(L"");
|
||
|
color_item_selected_fg.setElementName(L"wasabi.list.text.selected");
|
||
|
color_item_selected_fg.setColorGroup(L"");
|
||
|
color_item_selected.setElementName(L"wasabi.list.text.selected.background");
|
||
|
color_item_selected.setColorGroup(L"");
|
||
|
color_headers.setElementName(L"wasabi.list.column.background");
|
||
|
color_headers.setColorGroup(L"");
|
||
|
columnTextColor.setElementName(L"wasabi.list.column.text");
|
||
|
columnTextColor.setColorGroup(L"");
|
||
|
columnSepColor.setElementName(L"wasabi.list.column.frame.bottom");
|
||
|
columnSepColor.setColorGroup(L"");
|
||
|
color_item_focused.setColorGroup(L"");
|
||
|
color_item_focusrect.setColorGroup(L"");
|
||
|
}
|
||
|
selectonupdown = 1;
|
||
|
setAutoSort(FALSE);
|
||
|
setOwnerDraw(FALSE);
|
||
|
setSortDirection(0);
|
||
|
setSortColumn(0);
|
||
|
lastcolsort=-1;
|
||
|
dragtimeron = 0;
|
||
|
dragskip = DRAGSKIP_START;
|
||
|
dragskipcount = 0;
|
||
|
item_invalidate_border = 0;
|
||
|
|
||
|
metrics_ok = FALSE;
|
||
|
setFontSize(DEF_TEXT_SIZE);
|
||
|
redraw = TRUE;
|
||
|
columnsHeight = COLUMNS_DEFAULT_HEIGHT;
|
||
|
lastItemFocused = NULL;
|
||
|
lastItemFocusedPos = -1;
|
||
|
lastAddedItem = NULL;
|
||
|
selectionStart = -1;
|
||
|
colresize = -1;
|
||
|
resizing_col = FALSE;
|
||
|
processbup = FALSE;
|
||
|
bdown = FALSE;
|
||
|
nodrag = FALSE;
|
||
|
showColumnsHeaders = TRUE;
|
||
|
rectselecting=0;
|
||
|
preventMultipleSelection = 0;
|
||
|
//CUT autoresizecol0 = 0;
|
||
|
wantautodeselect = 1;
|
||
|
registerAcceleratorSection(L"listwnd");
|
||
|
hoverselect = 0;
|
||
|
firstItemVisible = -1;
|
||
|
lastItemVisible = -1;
|
||
|
showicons = 0;
|
||
|
iconWidth = -1; // If it's still negative use itemHeight instead -- better user getIconWidth()
|
||
|
iconHeight = -1;
|
||
|
antialias=1;
|
||
|
hasUserBg = false;
|
||
|
}
|
||
|
|
||
|
void ListWnd::CreateXMLParameters(int master_handle)
|
||
|
{
|
||
|
//LISTWND_PARENT::CreateXMLParameters(master_handle);
|
||
|
int numParams = sizeof(params) / sizeof(params[0]);
|
||
|
hintNumberOfParams(xuihandle, numParams);
|
||
|
for (int i = 0;i < numParams;i++)
|
||
|
addParam(xuihandle, params[i], XUI_ATTRIBUTE_IMPLIED);
|
||
|
}
|
||
|
|
||
|
ListWnd::~ListWnd() {
|
||
|
deleteAllItems();
|
||
|
columnsList.deleteAll();
|
||
|
nodrag=FALSE;
|
||
|
}
|
||
|
|
||
|
int ListWnd::setXuiParam(int _xuihandle, int xmlattributeid, const wchar_t *xmlattributename, const wchar_t *value)
|
||
|
{
|
||
|
if (_xuihandle != xuihandle)
|
||
|
return ScrlBkgWnd::setXuiParam(_xuihandle, xmlattributeid, xmlattributename, value);
|
||
|
|
||
|
switch (xmlattributeid)
|
||
|
{
|
||
|
case LIST_ANTIALIAS:
|
||
|
antialias = WTOI(value);
|
||
|
break;
|
||
|
case LIST_BACKGROUND:
|
||
|
{
|
||
|
SkinBitmap __bmp(value);
|
||
|
if (!__bmp.isInvalid())
|
||
|
{
|
||
|
this->setBgBitmap(value);
|
||
|
hasUserBg = true;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
this->setBgBitmap(L"wasabi.list.background");
|
||
|
hasUserBg = false;
|
||
|
}
|
||
|
}
|
||
|
this->invalidate();
|
||
|
break;
|
||
|
case LIST_TILE:
|
||
|
this->wantTileBg = WTOI(value)?1:0;
|
||
|
this->invalidate();
|
||
|
break;
|
||
|
case LIST_NOCOLHEADER:
|
||
|
setShowColumnsHeaders(!WTOI(value));
|
||
|
break;
|
||
|
default:
|
||
|
return 0;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onInit() {
|
||
|
|
||
|
LISTWND_PARENT::onInit();
|
||
|
|
||
|
if (!hasUserBg) setBgBitmap(L"wasabi.list.background");
|
||
|
setLineHeight(getItemHeight());
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onPostOnInit() {
|
||
|
LISTWND_PARENT::onPostOnInit();
|
||
|
recalcHeaders();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onPaint(Canvas *canvas)
|
||
|
{
|
||
|
|
||
|
PaintCanvas paintcanvas;
|
||
|
PaintBltCanvas paintbcanvas;
|
||
|
|
||
|
if (canvas == NULL) {
|
||
|
if (needDoubleBuffer()) {
|
||
|
if (!paintbcanvas.beginPaintNC(this)) return 0;
|
||
|
canvas = &paintbcanvas;
|
||
|
} else {
|
||
|
if (!paintcanvas.beginPaint(this)) return 0;
|
||
|
canvas = &paintcanvas;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
LISTWND_PARENT::onPaint(canvas);
|
||
|
|
||
|
RegionI clip;
|
||
|
canvas->getClipRgn(&clip);
|
||
|
api_region *orig;
|
||
|
orig = clip.clone();
|
||
|
|
||
|
drawColumnHeaders(canvas);
|
||
|
|
||
|
if (getColumnsHeight() > 0) {
|
||
|
RECT cr;
|
||
|
getClientRect(&cr);
|
||
|
cr.bottom = cr.top;
|
||
|
cr.top -= getColumnsHeight();
|
||
|
clip.subtractRect(&cr);
|
||
|
canvas->selectClipRgn(&clip);
|
||
|
}
|
||
|
|
||
|
firstItemVisible = -1;
|
||
|
lastItemVisible = -1;
|
||
|
|
||
|
//drawSubItems(canvas, x, &y, items, r.top, r.bottom, 0);
|
||
|
drawItems(canvas);
|
||
|
|
||
|
canvas->selectClipRgn(orig); // reset cliping region;
|
||
|
|
||
|
clip.disposeClone(orig);
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onResize() {
|
||
|
LISTWND_PARENT::onResize();
|
||
|
recalcHeaders();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getNumItems(void) {
|
||
|
return itemList.getNumItems();
|
||
|
}
|
||
|
|
||
|
void ListWnd::drawItems(Canvas *canvas) {
|
||
|
|
||
|
RECT r, c;
|
||
|
getClientRect(&r);
|
||
|
|
||
|
RegionI clip;
|
||
|
if (!canvas->getClipRgn(&clip))
|
||
|
{
|
||
|
clip.setRect(&r);
|
||
|
}
|
||
|
|
||
|
if (GetClipBox(canvas->getHDC(), &c) == NULLREGION) {
|
||
|
getClientRect(&c);
|
||
|
}
|
||
|
|
||
|
// DebugString("%d %d %d %d\n", c.left, c.top, c.right, c.bottom);
|
||
|
|
||
|
// float f,l;
|
||
|
calcBounds();
|
||
|
|
||
|
//int first, last;
|
||
|
|
||
|
/* RECT s=c;
|
||
|
|
||
|
s.bottom = min(s.bottom, r.bottom);
|
||
|
s.top = max(s.top, getLabelHeight()+(showColumnsHeaders ? getColumnsHeight() : 0));
|
||
|
|
||
|
f = ((float)(s.top - getLabelHeight() - (showColumnsHeaders ? getColumnsHeight() : 0) - getYShift() + getScrollY())) / (float)itemHeight;
|
||
|
l = ((float)(s.bottom - getLabelHeight() - (showColumnsHeaders ? getColumnsHeight() : 0) - getYShift() + getScrollY())) / (float)itemHeight;
|
||
|
first = (int)f;
|
||
|
first = max(0,first);
|
||
|
last = (int)l;
|
||
|
if ((float)((int)l) != l) {
|
||
|
last++;
|
||
|
}*/
|
||
|
|
||
|
int g=0;
|
||
|
|
||
|
for (int i=firstItemVisible;i<=lastItemVisible && i<itemList.getNumItems();i++) {
|
||
|
RECT ir={0,0,0,0};
|
||
|
getItemRect(i, &ir);
|
||
|
int a=ir.right;
|
||
|
ir.right = r.right;
|
||
|
if (!clip.doesIntersectRect(&ir))
|
||
|
continue;
|
||
|
ir.right=a;
|
||
|
g++;
|
||
|
|
||
|
LPARAM itemdata = getItemData(i);
|
||
|
int itemselected = getItemSelected(i);
|
||
|
int itemfocused = getItemFocused(i);
|
||
|
|
||
|
onPreItemDraw(canvas, i, &ir, itemdata, itemselected, itemfocused);
|
||
|
|
||
|
int sel = getItemSelected(i);
|
||
|
canvas->pushPen(PS_SOLID, 1, getColumnSepColor());
|
||
|
|
||
|
if (!ownerDraw(canvas, i, &ir, itemdata, itemselected, itemfocused))
|
||
|
{
|
||
|
Wasabi::FontInfo fontInfo;
|
||
|
fontInfo.antialias = getTextAntialias(itemdata);
|
||
|
fontInfo.bold = getTextBold(itemdata);
|
||
|
fontInfo.italic = !!getTextItalic(itemdata);
|
||
|
fontInfo.opaque = false;
|
||
|
fontInfo.color = sel ? getSelFgColor(itemdata) : getTextColor(itemdata);
|
||
|
fontInfo.pointSize = getFontSize();
|
||
|
int x = -getScrollX();
|
||
|
if (sel)
|
||
|
canvas->fillRect(&ir, getSelBgColor(itemdata));
|
||
|
if (getItemFocused(i))
|
||
|
canvas->fillRect(&ir, getFocusColor(itemdata));
|
||
|
if (needFocusRect(itemdata))
|
||
|
canvas->drawRect(&ir, 1, getFocusRectColor(itemdata));
|
||
|
if (showicons)
|
||
|
{
|
||
|
SkinBitmap *icon = getItemIcon(i);
|
||
|
if (icon)
|
||
|
{
|
||
|
RECT dst={x+X_SHIFT+r.left+1, ir.top+1, x+X_SHIFT+r.left+getIconWidth()+1, ir.top+getIconHeight()+1};
|
||
|
icon->stretchToRect(canvas, &dst);
|
||
|
}
|
||
|
x += getIconWidth()+1;
|
||
|
}
|
||
|
int xsep = wantColSepOnItems()?COLSEPHEIGHT:0;
|
||
|
for (int j=0;j<columnsList.getNumItems();j++) {
|
||
|
ListColumn *col = columnsList[j];
|
||
|
RECT cr=ir;
|
||
|
cr.left = x+X_SHIFT+r.left;
|
||
|
cr.right = cr.left + col->getWidth()-X_SHIFT*2+xsep;
|
||
|
if (j > 0 && wantColSepOnItems()) {
|
||
|
canvas->moveTo(x, ir.top);
|
||
|
canvas->lineTo(x, ir.top+getItemHeight());
|
||
|
}
|
||
|
switch (col->getAlignment()) {
|
||
|
case COL_LEFTALIGN:
|
||
|
canvas->textOutEllipsed(cr.left+2, cr.top, cr.right-cr.left-4, cr.bottom-cr.top, getSubitemText(i, j), &fontInfo);
|
||
|
break;
|
||
|
case COL_CENTERALIGN: {
|
||
|
RECT _cr = {cr.left+2, cr.top, cr.right-4, cr.bottom};
|
||
|
canvas->textOutCentered(&_cr, getSubitemText(i, j), &fontInfo);
|
||
|
break;
|
||
|
}
|
||
|
case COL_RIGHTALIGN: {
|
||
|
const wchar_t *txt = getSubitemText(i, j);
|
||
|
int __x = cr.left;
|
||
|
int __y = cr.top;
|
||
|
int fw = canvas->getTextWidth(txt, &fontInfo);
|
||
|
int aw = cr.right-cr.left-4;
|
||
|
__x -= fw-aw;
|
||
|
canvas->textOut(__x, __y, txt, &fontInfo);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
x += col->getWidth()+xsep;
|
||
|
}
|
||
|
if (wantColSepOnItems()) {
|
||
|
canvas->moveTo(x, ir.top);
|
||
|
canvas->lineTo(x, ir.top+getItemHeight());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
canvas->popPen();
|
||
|
onPostItemDraw(canvas, i, &ir, itemdata, itemselected, itemfocused);
|
||
|
}
|
||
|
/*
|
||
|
OutputDebugString("%d items draw\n", g);
|
||
|
*/
|
||
|
}
|
||
|
|
||
|
int ListWnd::wantColSepOnItems() {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getXShift() {
|
||
|
if (wantColSepOnItems()) return X_SHIFT; else return 0;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getFirstItemSelected() {
|
||
|
return getNextItemSelected(-1);
|
||
|
}
|
||
|
|
||
|
int ListWnd::getNextItemSelected(int lastpos) {
|
||
|
if (lastpos < -1) lastpos = -1;
|
||
|
for (int i=lastpos+1;i<itemList.getNumItems();i++)
|
||
|
if (getItemSelected(i))
|
||
|
return i;
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
void ListWnd::calcBounds() {
|
||
|
lastComplete = TRUE;
|
||
|
firstComplete = TRUE;
|
||
|
float f,l;
|
||
|
RECT r;
|
||
|
getClientRect(&r);
|
||
|
|
||
|
f = ((float)(getScrollY()-Y_SHIFT) / getItemHeight());
|
||
|
l = ((float)(getScrollY()-Y_SHIFT+(r.bottom-r.top)) / getItemHeight()) - 1.0f;
|
||
|
|
||
|
firstItemVisible = (int)f;
|
||
|
lastItemVisible = (int)l;
|
||
|
|
||
|
if ((float)((int)l) != l) {
|
||
|
lastItemVisible++;
|
||
|
lastComplete = FALSE;
|
||
|
}
|
||
|
if ((float)((int)f) != f && f >= 0) {
|
||
|
firstComplete = FALSE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Draws tiled background
|
||
|
void ListWnd::drawBackground(Canvas *canvas)
|
||
|
{
|
||
|
LISTWND_PARENT::drawBackground(canvas);
|
||
|
drawColumnHeaders(canvas);
|
||
|
}
|
||
|
|
||
|
void ListWnd::drawColumnHeaders(Canvas *c)
|
||
|
{
|
||
|
if (columnsList.getNumItems() == 0 || !showColumnsHeaders) return;
|
||
|
|
||
|
RECT r;
|
||
|
getClientRect(&r);
|
||
|
r.top -= getColumnsHeight();
|
||
|
r.bottom = r.top + getColumnsHeight();
|
||
|
if (renderRatioActive())
|
||
|
r.left -= (int)((double)getScrollX()*getRenderRatio());
|
||
|
else
|
||
|
r.left-=getScrollX();
|
||
|
|
||
|
c->fillRect(&r, color_headers);
|
||
|
int x = r.left + X_SHIFT/*+ 1*/;
|
||
|
|
||
|
if (showicons)
|
||
|
x += getIconWidth()+1 + 2;
|
||
|
|
||
|
Wasabi::FontInfo fontInfo;
|
||
|
fontInfo.color = columnTextColor;
|
||
|
fontInfo.opaque = false;
|
||
|
fontInfo.pointSize = getColumnsHeight();
|
||
|
c->pushPen(PS_SOLID, 1, getColumnSepColor());
|
||
|
|
||
|
for (int i=0;i<columnsList.getNumItems();i++)
|
||
|
{
|
||
|
ListColumn *col = columnsList[i];
|
||
|
int width = col->getWidth();
|
||
|
if (i > 0) {
|
||
|
c->moveTo(x, r.top);
|
||
|
c->lineTo(x, r.top+getColumnsHeight());
|
||
|
}
|
||
|
RECT ch;
|
||
|
ch.left = x+COLUMNS_MARGIN-((i==0)?X_SHIFT:0);
|
||
|
ch.top = r.top;
|
||
|
ch.right = ch.left + col->getWidth()-1-COLUMNS_MARGIN*2+((i==0)?X_SHIFT:0);
|
||
|
ch.bottom = ch.top + getColumnsHeight()-1;
|
||
|
col->customDrawHeader(c, &ch, &fontInfo);
|
||
|
x+=width/*+1*/;
|
||
|
}
|
||
|
c->moveTo(x, r.top);
|
||
|
c->lineTo(x, r.top+getColumnsHeight());
|
||
|
c->popPen();
|
||
|
|
||
|
}
|
||
|
|
||
|
ARGB32 ListWnd::getColumnSepColor() {
|
||
|
return columnSepColor;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getHeaderHeight() {
|
||
|
return (showColumnsHeaders && columnsList.getNumItems() > 0) ? getColumnsHeight() : 0;
|
||
|
}
|
||
|
|
||
|
// Returns the current tree width in pixels
|
||
|
int ListWnd::getContentsWidth() {
|
||
|
return getColumnsWidth()+X_SHIFT;
|
||
|
}
|
||
|
|
||
|
// Returns the current tree height in pixels
|
||
|
int ListWnd::getContentsHeight() {
|
||
|
return itemList.getNumItems()*(getItemHeight()+(wantColSepOnItems()?COLSEPHEIGHT:0))+Y_SHIFT*2;
|
||
|
}
|
||
|
|
||
|
void ListWnd::setAutoSort(bool dosort) {
|
||
|
autosort = dosort;
|
||
|
itemList.setAutoSort(dosort);
|
||
|
}
|
||
|
|
||
|
void ListWnd::setOwnerDraw(bool doownerdraw) {
|
||
|
ownerdraw = doownerdraw;
|
||
|
}
|
||
|
|
||
|
int ListWnd::setFontSize(int size)
|
||
|
{
|
||
|
LISTWND_PARENT::setFontSize(size);
|
||
|
if (size >= 0) textsize = size;
|
||
|
TextInfoCanvas c(this);
|
||
|
Wasabi::FontInfo fontInfo;
|
||
|
fontInfo.pointSize = getFontSize();
|
||
|
setItemHeight(c.getTextHeight(&fontInfo)+2, false);
|
||
|
redraw = TRUE;
|
||
|
metrics_ok = FALSE;
|
||
|
invalidate();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getFontSize() {
|
||
|
#ifndef WASABINOMAINAPI
|
||
|
return textsize+api->metrics_getDelta();
|
||
|
#else
|
||
|
//MULTIAPI-FIXME: not handling delta
|
||
|
return textsize;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
void ListWnd::onSetVisible(int show) {
|
||
|
LISTWND_PARENT::onSetVisible(show);
|
||
|
if (show) invalidate();
|
||
|
}
|
||
|
|
||
|
int ListWnd::fullyVisible(int pos) {
|
||
|
return (((lastComplete && pos <= lastItemVisible) || (!lastComplete && pos < lastItemVisible)) && ((firstComplete && pos >= firstItemVisible) || (!firstComplete && pos > firstItemVisible)));
|
||
|
}
|
||
|
|
||
|
void ListWnd::ensureItemVisible(int pos) {
|
||
|
if (pos >= itemList.getNumItems()) pos = itemList.getNumItems()-1;
|
||
|
if (pos < 0) pos = 0;
|
||
|
if (fullyVisible(pos)) return;
|
||
|
|
||
|
RECT c;
|
||
|
int y=pos*getItemHeight();
|
||
|
getClientRect(&c);
|
||
|
int showing_height = c.bottom - c.top;
|
||
|
|
||
|
if (getScrollY() < y) // scrolling up
|
||
|
y = (y - showing_height) + getItemHeight()*3;
|
||
|
else
|
||
|
y -= getItemHeight()*2;
|
||
|
|
||
|
if (y < 0) y = 0;
|
||
|
else if (y + showing_height > getContentsHeight()) {
|
||
|
// just show bottom pane
|
||
|
y = getContentsHeight()-showing_height;
|
||
|
}
|
||
|
scrollToY(y);
|
||
|
}
|
||
|
|
||
|
void ListWnd::scrollToItem(int pos) {
|
||
|
if (!isInited()) return;
|
||
|
scrollToY(Y_SHIFT+pos*(getItemHeight()+(wantColSepOnItems()?COLSEPHEIGHT:0)));
|
||
|
}
|
||
|
|
||
|
int ListWnd::getNumColumns() {
|
||
|
return columnsList.getNumItems();
|
||
|
}
|
||
|
|
||
|
ListColumn *ListWnd::getColumn(int n) {
|
||
|
return columnsList.enumItem(n);
|
||
|
}
|
||
|
|
||
|
int ListWnd::getColumnWidth(int c) {
|
||
|
ListColumn *col = columnsList[c];
|
||
|
ASSERT(col != NULL);
|
||
|
return col->getWidth();
|
||
|
}
|
||
|
|
||
|
int ListWnd::selectAll(int cb) {
|
||
|
|
||
|
int i;
|
||
|
|
||
|
if (preventMultipleSelection) return 1;
|
||
|
|
||
|
for (i=0;i<itemList.getNumItems();i++) {
|
||
|
selItemList.setSelected(i, TRUE, cb);
|
||
|
}
|
||
|
|
||
|
if (cb)
|
||
|
notifySelChanged();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::deselectAll(int cb) {
|
||
|
|
||
|
int lif = getItemFocused();
|
||
|
if (lif != -1)
|
||
|
invalidateItem(lif);
|
||
|
lastItemFocused = NULL;
|
||
|
|
||
|
for (int i = 0; i < itemList.getNumItems(); i++) {
|
||
|
selItemList.setSelected(i, FALSE, cb);
|
||
|
}
|
||
|
|
||
|
if (cb)
|
||
|
notifySelChanged();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
void ListWnd::notifySelChanged(int item, int sel) {
|
||
|
if (!getRedraw()) return;
|
||
|
if (item == -1)
|
||
|
notifyParent(ChildNotify::LISTWND_SELCHANGED);
|
||
|
else
|
||
|
notifyParent(ChildNotify::LISTWND_ITEMSELCHANGED, item, sel);
|
||
|
}
|
||
|
|
||
|
int ListWnd::invertSelection(int cb) {
|
||
|
if (preventMultipleSelection) return 1;
|
||
|
for (int i = 0; i < itemList.getNumItems(); i++) {
|
||
|
toggleSelection(i, FALSE, cb);
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::invalidateItem(int pos) {
|
||
|
RECT r;
|
||
|
if (!isInited()) return 0;
|
||
|
int rv = getItemRect(pos, &r);
|
||
|
r.top -= item_invalidate_border;
|
||
|
r.bottom += item_invalidate_border;
|
||
|
if (rv)
|
||
|
invalidateRect(&r);
|
||
|
return rv;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onLeftButtonDblClk(int x, int y) {
|
||
|
// check for column dblclick
|
||
|
int colhit;
|
||
|
if ((colhit = hitTestColumns(Wasabi::Std::makePoint(x, y))) >= 0) {
|
||
|
return onColumnDblClick(colhit, x, y);
|
||
|
}
|
||
|
|
||
|
if (itemList.getNumItems() == 0) return 0;
|
||
|
POINT p={x,y};
|
||
|
int i = hitTest(p);
|
||
|
if (i > -1) {
|
||
|
notifyParent(ChildNotify::LISTWND_DBLCLK, i, 0);
|
||
|
onDoubleClick(i);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onRightButtonDown(int x, int y) {
|
||
|
nodrag=TRUE;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onRightButtonUp(int x, int y) {
|
||
|
nodrag=FALSE;
|
||
|
int i = hitTest(x,y);
|
||
|
if (i >= 0) { // did hit something
|
||
|
setItemFocused(i); // it always gets the focus
|
||
|
if (!getItemSelected(i)) { // reselect the item out of the cur selection
|
||
|
ListWnd::onLeftButtonDown(x,y); // don't call inherited!
|
||
|
ListWnd::onLeftButtonUp(x,y); // don't call inherited!
|
||
|
}
|
||
|
onRightClick(i);
|
||
|
} else {
|
||
|
if (wantAutoDeselect())
|
||
|
deselectAll();
|
||
|
}
|
||
|
onContextMenu(x,y);
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onRightClick(int itemnum) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
int ListWnd::onLeftButtonDown(int x, int y) {
|
||
|
if (colresize != -1) {
|
||
|
resizing_col = TRUE;
|
||
|
drawXorLine(colresizept.x);
|
||
|
}
|
||
|
|
||
|
processbup = FALSE;
|
||
|
bdown = TRUE;
|
||
|
bdownx = x;
|
||
|
bdowny = y;
|
||
|
|
||
|
if (!resizing_col) {
|
||
|
|
||
|
POINT p={x,y};
|
||
|
int i = hitTest(p);
|
||
|
if (i >= 0) {
|
||
|
if (Std::keyDown(VK_SHIFT)) {
|
||
|
if (getItemSelected(i))
|
||
|
processbup=TRUE;
|
||
|
else
|
||
|
setSelectionEnd(i);
|
||
|
} else
|
||
|
if (Std::keyDown(VK_CONTROL)) {
|
||
|
if (getItemSelected(i))
|
||
|
processbup=TRUE;
|
||
|
else
|
||
|
toggleSelection(i);
|
||
|
} else {
|
||
|
if (getItemSelected(i))
|
||
|
processbup = TRUE;
|
||
|
else
|
||
|
setSelectionStart(i);
|
||
|
}
|
||
|
} else {
|
||
|
if (wantAutoDeselect())
|
||
|
deselectAll();
|
||
|
/*rectselecting=1;
|
||
|
selectStart.x = x;
|
||
|
selectStart.y = y;
|
||
|
selectLast.x = x;
|
||
|
selectLast.y = y;
|
||
|
drawRect(selectStart.x, selectStart.y, x, y);
|
||
|
beginCapture();*/
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onLeftButtonUp(int x, int y) {
|
||
|
|
||
|
bdown = FALSE;
|
||
|
|
||
|
if (resizing_col) {
|
||
|
resizing_col = FALSE;
|
||
|
drawXorLine(colresizept.x);
|
||
|
calcNewColWidth(colresize, colresizept.x);
|
||
|
recalcHeaders();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
// check for column label click
|
||
|
int colhit;
|
||
|
if ((colhit = hitTestColumnsLabel(Wasabi::Std::makePoint(x, y))) >= 0) {
|
||
|
ListColumn *lc = getColumn(colhit);
|
||
|
int ret = lc->onHeaderClick();
|
||
|
if (!ret) ret = onColumnLabelClick(colhit, x, y);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
POINT p={x,y};
|
||
|
int i = hitTest(p);
|
||
|
|
||
|
if (rectselecting || (processbup && !resizing_col) || hoverselect) {
|
||
|
if (i >= 0) {
|
||
|
if (Std::keyDown(VK_SHIFT)) {
|
||
|
if (getItemSelected(i))
|
||
|
setSelectionStart(i);
|
||
|
else
|
||
|
setSelectionEnd(i);
|
||
|
} else {
|
||
|
if (Std::keyDown(VK_CONTROL) || !wantAutoDeselect()) {
|
||
|
toggleSelection(i);
|
||
|
selectionStart = i;
|
||
|
} else {
|
||
|
setSelectionStart(i);
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
/* if (rectselecting) {
|
||
|
drawRect(selectStart.x, selectStart.y, selectLast.x, selectLast.y);
|
||
|
endCapture();
|
||
|
rectselecting=0;
|
||
|
selectRect(selectStart.x, selectStart.y, x, y);
|
||
|
} else*/
|
||
|
if (wantAutoDeselect())
|
||
|
deselectAll();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (i >= 0) {
|
||
|
int cn = hitTestColumnClient(x);
|
||
|
int r = 0;
|
||
|
if (cn >= 0) {
|
||
|
ListColumn *lc = getColumn(cn);
|
||
|
ASSERT(lc != NULL);
|
||
|
r = lc->onColumnLeftClick(i);
|
||
|
}
|
||
|
if (!r)
|
||
|
{
|
||
|
// Add 1px clickable border to our icon
|
||
|
if (x < getIconWidth()+2) r = onIconLeftClick(i,x,y-(i*getItemHeight()) - getHeaderHeight());
|
||
|
if (!r) onLeftClick(i);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onMouseMove(int x, int y) {
|
||
|
LISTWND_PARENT::onMouseMove(x,y);
|
||
|
|
||
|
if (!bdown && (Std::keyDown(MK_RBUTTON) || Std::keyDown(MK_RBUTTON))) {
|
||
|
bdown = TRUE;
|
||
|
processbup = TRUE;
|
||
|
_enterCapture();
|
||
|
}
|
||
|
|
||
|
if (!rectselecting && bdown && !resizing_col && !nodrag && (ABS(x-bdownx) >= 4 || ABS(y-bdowny) >= 4)) {
|
||
|
processbup = FALSE;
|
||
|
bdown = FALSE;
|
||
|
int i = hitTest(x, y);
|
||
|
if (i != -1) {
|
||
|
onBeginDrag(i);
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* if (rectselecting) {
|
||
|
drawRect(selectStart.x, selectStart.y, selectLast.x, selectLast.y);
|
||
|
selectLast.x = x;
|
||
|
selectLast.y = y;
|
||
|
drawRect(selectStart.x, selectStart.y, selectLast.x, selectLast.y);
|
||
|
return 1;
|
||
|
}*/
|
||
|
|
||
|
POINT p={x,y};
|
||
|
if (wantResizeCols()) {
|
||
|
if (!resizing_col) {
|
||
|
int c = hitTestColumns(p, &colresizeo);
|
||
|
if (c != -1) {
|
||
|
if (colresize != c) {
|
||
|
SetCursor(LoadCursor(NULL, IDC_SIZEWE)); // NONPORTABLE
|
||
|
if (!getCapture())
|
||
|
beginCapture();
|
||
|
colresize = c;
|
||
|
colresizept = p;
|
||
|
}
|
||
|
} else {
|
||
|
if (colresize != -1) {
|
||
|
SetCursor(LoadCursor(NULL, IDC_ARROW)); // NONPORTABLE
|
||
|
endCapture();
|
||
|
colresize = -1;
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
if (p.x + getScrollX() < colresizeo + COLUMNS_MIN_WIDTH) {
|
||
|
p.x = colresizeo + COLUMNS_MIN_WIDTH - getScrollX();
|
||
|
}
|
||
|
drawXorLine(colresizept.x);
|
||
|
colresizept = p;
|
||
|
drawXorLine(colresizept.x);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (hoverselect) {
|
||
|
int i = hitTest(x, y);
|
||
|
if (i >= 0 && !getItemSelected(i)) {
|
||
|
deselectAll(0);
|
||
|
setSelected(i, 1, 0);
|
||
|
wchar_t t[256]=L"";
|
||
|
getItemLabel(i, 0, t, 255);
|
||
|
foreach(tempselectnotifies)
|
||
|
sendAction(tempselectnotifies.getfor(), L"tempselectnotify", t);
|
||
|
endfor;
|
||
|
}
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
// NONPORTABLE / Todo: implement necessary stuff in canvas
|
||
|
void ListWnd::drawXorLine(int x) {
|
||
|
HDC dc = GetDC(gethWnd());
|
||
|
HBRUSH brush = CreateSolidBrush(0xFFFFFF);
|
||
|
HPEN pen = CreatePen(PS_SOLID,0,0xFFFFFF);
|
||
|
HBRUSH oldB = (HBRUSH)SelectObject(dc, brush);
|
||
|
HPEN oldP = (HPEN)SelectObject(dc, pen);
|
||
|
int mix = SetROP2(dc,R2_XORPEN);
|
||
|
|
||
|
RECT r;
|
||
|
getClientRect(&r);
|
||
|
r.top -= getColumnsHeight();
|
||
|
r.bottom = r.top + getColumnsHeight();
|
||
|
|
||
|
if (renderRatioActive()) {
|
||
|
multRatio(&x);
|
||
|
multRatio(&r);
|
||
|
}
|
||
|
|
||
|
MoveToEx(dc, x, r.top, NULL);
|
||
|
LineTo(dc, x, r.bottom);
|
||
|
|
||
|
SelectObject(dc, oldB);
|
||
|
SelectObject(dc, oldP);
|
||
|
SetROP2(dc, mix);
|
||
|
DeleteObject(pen);
|
||
|
DeleteObject(brush);
|
||
|
ReleaseDC(gethWnd(), dc);
|
||
|
}
|
||
|
|
||
|
void ListWnd::calcNewColWidth(int c, int px) {
|
||
|
RECT r;
|
||
|
getClientRect(&r);
|
||
|
r.top -= getColumnsHeight();
|
||
|
r.bottom = r.top + getColumnsHeight();
|
||
|
px += getScrollX();
|
||
|
int x = r.left+X_SHIFT;
|
||
|
for (int i=0;i<columnsList.getNumItems();i++) {
|
||
|
ListColumn *col = columnsList[i];
|
||
|
if (col->getIndex() == c) {
|
||
|
int w = px - x;
|
||
|
col->setWidth(w);
|
||
|
setSlidersPosition();
|
||
|
return;
|
||
|
}
|
||
|
x += col->getWidth();
|
||
|
}
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
int ListWnd::hitTestColumns(POINT p, int *origin) {
|
||
|
RECT r;
|
||
|
if (!showColumnsHeaders) return -1;
|
||
|
int best=-1, besto = 0, bestd=9999;
|
||
|
getClientRect(&r);
|
||
|
r.top -= getColumnsHeight();
|
||
|
r.bottom = r.top + getColumnsHeight();
|
||
|
p.x += getScrollX();
|
||
|
if (p.y > r.top && p.y < r.top + getColumnsHeight()) {
|
||
|
int x = r.left+X_SHIFT;
|
||
|
for (int i=0;i<columnsList.getNumItems();i++) {
|
||
|
ListColumn *col = columnsList[i];
|
||
|
x += col->getWidth();
|
||
|
if (p.x > x-COLUMNS_RESIZE_THRESHOLD && p.x < x+COLUMNS_RESIZE_THRESHOLD) {
|
||
|
int d = ABS(p.x-x);
|
||
|
if (d < bestd) {
|
||
|
bestd = d;
|
||
|
besto = x - col->getWidth();
|
||
|
best = col->getIndex();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (best != -1)
|
||
|
if (origin != NULL) *origin = besto;
|
||
|
return best;
|
||
|
}
|
||
|
|
||
|
int ListWnd::hitTestColumnClient(int x) {
|
||
|
RECT cr = clientRect();
|
||
|
int x1 = cr.left+X_SHIFT;
|
||
|
foreach(columnsList)
|
||
|
int x2 = x1 + columnsList.getfor()->getWidth();
|
||
|
if (x >= x1 && x <= x2) return foreach_index;
|
||
|
x1 = x2;
|
||
|
endfor
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
void ListWnd::invalidateColumns() {
|
||
|
RECT r;
|
||
|
getClientRect(&r);
|
||
|
r.top -= getColumnsHeight();
|
||
|
r.bottom = r.top + getColumnsHeight();
|
||
|
invalidateRect(&r);
|
||
|
}
|
||
|
|
||
|
int ListWnd::hitTestColumnsLabel(POINT p) {
|
||
|
RECT r;
|
||
|
if (!showColumnsHeaders) return -1;
|
||
|
getClientRect(&r);
|
||
|
r.top -= getColumnsHeight();
|
||
|
r.bottom = r.top + getColumnsHeight();
|
||
|
p.x += getScrollX();
|
||
|
if (p.y > r.top && p.y < r.top + getColumnsHeight()) {
|
||
|
int x = X_SHIFT;
|
||
|
for (int i=0;i<columnsList.getNumItems();i++) {
|
||
|
ListColumn *col = columnsList[i];
|
||
|
if (p.x >= x && p.x <= x+col->getWidth())
|
||
|
return i;
|
||
|
x += col->getWidth();
|
||
|
}
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
void ListWnd::setSelectionStart(int pos, int wantcb) {
|
||
|
if (wantAutoDeselect())
|
||
|
deselectAll(wantcb);
|
||
|
if (!selItemList.isSelected(pos)) {
|
||
|
selItemList.setSelected(pos, TRUE, wantcb);
|
||
|
lastItemFocused = itemList[pos];
|
||
|
lastItemFocusedPos = pos;
|
||
|
invalidateItem(pos);
|
||
|
repaint();
|
||
|
}
|
||
|
|
||
|
selectionStart = pos;
|
||
|
notifySelChanged();
|
||
|
}
|
||
|
|
||
|
void ListWnd::setSelectionEnd(int pos) {
|
||
|
|
||
|
if (itemList.getNumItems() == 0) return;
|
||
|
if (selectionStart == -1) selectionStart = 0;
|
||
|
|
||
|
if (wantAutoDeselect())
|
||
|
deselectAll();
|
||
|
|
||
|
int inc = (selectionStart > pos) ? -1 : 1;
|
||
|
int i = selectionStart;
|
||
|
|
||
|
while (1) {
|
||
|
if (!selItemList.isSelected(i)) {
|
||
|
selItemList.setSelected(i, TRUE);
|
||
|
lastItemFocused = itemList[i];
|
||
|
lastItemFocusedPos = i;
|
||
|
invalidateItem(i);
|
||
|
}
|
||
|
if (i == pos) break;
|
||
|
i += inc;
|
||
|
}
|
||
|
notifySelChanged();
|
||
|
}
|
||
|
|
||
|
void ListWnd::setSelected(int pos, int selected, int cb) {
|
||
|
selItemList.setSelected(pos, selected, cb);
|
||
|
}
|
||
|
|
||
|
void ListWnd::toggleSelection(int pos, int setfocus, int cb) {
|
||
|
if (!selItemList.isSelected(pos)) {
|
||
|
selItemList.setSelected(pos, TRUE, cb);
|
||
|
if (setfocus) {
|
||
|
if (selItemList.getNumSelected() > 1) {
|
||
|
for (int i=0;i<itemList.getNumItems();i++)
|
||
|
if (selItemList.isSelected(i))
|
||
|
invalidateItem(i);
|
||
|
}
|
||
|
lastItemFocused = itemList[pos];
|
||
|
lastItemFocusedPos = pos;
|
||
|
}
|
||
|
} else {
|
||
|
selItemList.setSelected(pos, FALSE, cb);
|
||
|
if (setfocus) {
|
||
|
lastItemFocused = NULL;
|
||
|
lastItemFocusedPos = -1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
invalidateItem(pos);
|
||
|
if (cb)
|
||
|
notifySelChanged();
|
||
|
}
|
||
|
|
||
|
int ListWnd::onBeginDrag(int iItem) {
|
||
|
// nothing by default
|
||
|
lastItemFocused = NULL;
|
||
|
lastItemFocusedPos = -1;
|
||
|
invalidateItem(iItem);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int ListWnd::dragOver(int x, int y, ifc_window *sourceWnd) {
|
||
|
int rt = LISTWND_PARENT::dragOver(x, y, sourceWnd);
|
||
|
if (dragtimeron) return rt;
|
||
|
|
||
|
POINT pos={x,y};
|
||
|
screenToClient(&pos);
|
||
|
|
||
|
int item = hitTest(pos);
|
||
|
if (item == LW_HT_BELOW || item == LW_HT_ABOVE) {
|
||
|
dragtimeron = 1;
|
||
|
dragskip = DRAGSKIP_START;
|
||
|
dragskipcount = DRAGSKIP_START-1; // start right away
|
||
|
setTimer(LISTWND_DRAG_TIMERID, LISTWND_DRAG_TIMERDELAY);
|
||
|
}
|
||
|
return rt;
|
||
|
}
|
||
|
|
||
|
void ListWnd::onDragTimer() {
|
||
|
POINT pos;
|
||
|
Wasabi::Std::getMousePos(&pos);
|
||
|
screenToClient(&pos);
|
||
|
int item = hitTest(pos);
|
||
|
if (item == LW_HT_BELOW || item == LW_HT_ABOVE) {
|
||
|
dragskipcount++;
|
||
|
if (dragskipcount >= dragskip) {
|
||
|
switch (item) {
|
||
|
case LW_HT_BELOW:
|
||
|
scrollDown();
|
||
|
break;
|
||
|
case LW_HT_ABOVE:
|
||
|
scrollUp();
|
||
|
break;
|
||
|
}
|
||
|
dragskipcount = 0;
|
||
|
if (dragskip > 0) dragskip--;
|
||
|
}
|
||
|
} else {
|
||
|
killTimer(LISTWND_DRAG_TIMERID);
|
||
|
dragtimeron = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ListWnd::timerCallback(int id) {
|
||
|
switch (id) {
|
||
|
case LISTWND_DRAG_TIMERID:
|
||
|
onDragTimer();
|
||
|
return;
|
||
|
}
|
||
|
LISTWND_PARENT::timerCallback(id);
|
||
|
}
|
||
|
|
||
|
void ListWnd::onSelectAll() {
|
||
|
}
|
||
|
|
||
|
void ListWnd::onDelete() {
|
||
|
// do nothing by default
|
||
|
}
|
||
|
|
||
|
void ListWnd::onDoubleClick(int itemnum) {
|
||
|
// do nothing by default
|
||
|
}
|
||
|
|
||
|
void ListWnd::onLeftClick(int itemnum) {
|
||
|
// do nothing by default
|
||
|
}
|
||
|
|
||
|
int ListWnd::onIconLeftClick (int itemnum, int x, int y)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void ListWnd::onSecondLeftClick(int itemnum) {
|
||
|
// do nothing by default
|
||
|
}
|
||
|
|
||
|
int ListWnd::scrollAbsolute(int x) {
|
||
|
scrollToX(x);
|
||
|
return getScrollX();
|
||
|
}
|
||
|
|
||
|
int ListWnd::scrollRelative(int x) {
|
||
|
scrollToX(getScrollX() + x);
|
||
|
return getScrollX();
|
||
|
}
|
||
|
|
||
|
int ListWnd::getItemFocused() {
|
||
|
if (lastItemFocused == NULL) return -1;
|
||
|
if (itemList[lastItemFocusedPos] == lastItemFocused)
|
||
|
return lastItemFocusedPos;
|
||
|
else {
|
||
|
lastItemFocusedPos = itemList.searchItem(lastItemFocused);
|
||
|
return lastItemFocusedPos;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ListWnd::setItemFocused(int pos, int ensure_visible) {
|
||
|
invalidateItem(lastItemFocusedPos);
|
||
|
lastItemFocused = itemList[pos];
|
||
|
lastItemFocusedPos = -1;
|
||
|
if (lastItemFocused != NULL) lastItemFocusedPos = pos;
|
||
|
invalidateItem(lastItemFocusedPos);
|
||
|
if (ensure_visible) ensureItemVisible(pos);
|
||
|
}
|
||
|
|
||
|
int ListWnd::getItemRect(int pos, RECT *r) {
|
||
|
MEMSET(r, 0, sizeof(RECT));
|
||
|
if (pos < 0 || pos >= itemList.getNumItems()) return 0;
|
||
|
RECT cr={0,0,0,0};
|
||
|
getClientRect(&cr);
|
||
|
r->left = -getScrollX() + X_SHIFT + cr.left;
|
||
|
r->right = cr.left + getColumnsWidth();
|
||
|
if (showicons) r->right += getItemHeight();
|
||
|
r->top = -getScrollY() + pos * (getItemHeight() + (wantColSepOnItems() ? COLSEPHEIGHT : 0)) + Y_SHIFT + cr.top;
|
||
|
r->bottom = r->top + getItemHeight();
|
||
|
|
||
|
// clip!
|
||
|
if (r->top > cr.bottom || r->bottom < cr.top) return 0;
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::locateData(LPARAM data) { // linear search
|
||
|
for (int i=0;i<itemList.getNumItems();i++) {
|
||
|
if (itemList[i]->data == data)
|
||
|
return i;
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getItemFocused(int pos) {
|
||
|
if (pos >= itemList.getNumItems()) return 0;
|
||
|
return getItemFocused() == pos;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getItemSelected(int pos) {
|
||
|
listItem *item = itemList[pos];
|
||
|
return (item && selItemList.isSelected(pos));
|
||
|
}
|
||
|
|
||
|
int ListWnd::hitTest(POINT pos, int drag) {
|
||
|
RECT r;
|
||
|
getClientRect(&r);
|
||
|
if (pos.y < r.top) return LW_HT_ABOVE;
|
||
|
if (getScrollY() > 0 && drag && pos.y < r.top + LISTWND_DRAG_MARGIN) return LW_HT_ABOVE;
|
||
|
if (pos.y > r.bottom) return LW_HT_BELOW;
|
||
|
if (getContentsHeight() > (getScrollY() + r.bottom-r.top) && drag && pos.y > r.bottom - LISTWND_DRAG_MARGIN) return LW_HT_BELOW;
|
||
|
|
||
|
if (pos.x > r.left + getColumnsWidth() - getScrollX()) return LW_HT_DONTKNOW;
|
||
|
if (pos.x < r.left + getScrollX()) return LW_HT_DONTKNOW;
|
||
|
|
||
|
int i = (pos.y - r.top + getScrollY() - Y_SHIFT) / (getItemHeight() + (wantColSepOnItems() ? COLSEPHEIGHT : 0));
|
||
|
if (i >= itemList.getNumItems()) return LW_HT_DONTKNOW;
|
||
|
return i;
|
||
|
}
|
||
|
|
||
|
int ListWnd::hitTest(int x, int y, int drag) {
|
||
|
POINT pt={x, y};
|
||
|
return hitTest(pt, drag);
|
||
|
}
|
||
|
|
||
|
void ListWnd::selectRect(int x1, int y1, int x2, int y2) {
|
||
|
|
||
|
}
|
||
|
|
||
|
void ListWnd::drawRect(int x1, int y1, int x2, int y2) {
|
||
|
HDC dc = GetDC(gethWnd());
|
||
|
RECT r={x1,y1,x2,y2};
|
||
|
DrawFocusRect(dc, &r);
|
||
|
ReleaseDC(gethWnd(), dc);
|
||
|
}
|
||
|
|
||
|
LPARAM ListWnd::getItemData(int pos) {
|
||
|
if (pos >= itemList.getNumItems()) return 0;
|
||
|
listItem *item = itemList[pos];
|
||
|
if (item) return item->data;
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getItemLabel(int pos, int subpos, wchar_t *txt, int textmax)
|
||
|
{
|
||
|
const wchar_t *t = getSubitemText(pos, subpos);
|
||
|
if (t)
|
||
|
{
|
||
|
WCSCPYN(txt, t, textmax);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void ListWnd::setItemLabel(int pos, const wchar_t *text)
|
||
|
{
|
||
|
if (pos >= itemList.getNumItems()) return;
|
||
|
listItem *item = itemList[pos];
|
||
|
if (!item) return;
|
||
|
item->label = text;
|
||
|
invalidateItem(pos);
|
||
|
}
|
||
|
|
||
|
void ListWnd::resort() {
|
||
|
itemList.sort(TRUE);
|
||
|
invalidate();
|
||
|
}
|
||
|
|
||
|
int ListWnd::getSortDirection() {
|
||
|
return sortdir;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getSortColumn() {
|
||
|
return sortcol;
|
||
|
}
|
||
|
|
||
|
void ListWnd::setSortDirection(int dir) {
|
||
|
sortdir=dir;
|
||
|
}
|
||
|
|
||
|
void ListWnd::setSortColumn(int col) {
|
||
|
sortcol=col;
|
||
|
}
|
||
|
|
||
|
int ListWnd::sortCompareItem(listItem *p1, listItem *p2)
|
||
|
{
|
||
|
const wchar_t *l1=p1->label;
|
||
|
const wchar_t *l2=p2->label;
|
||
|
|
||
|
int i;
|
||
|
|
||
|
if(sortcol!=0)
|
||
|
{
|
||
|
if (p1->subitems != NULL)
|
||
|
{
|
||
|
for (i=0;i<p1->subitems->getNumItems();i++)
|
||
|
{
|
||
|
listSubitemStruct *subitem = p1->subitems->enumItem(i);
|
||
|
if (subitem->column == sortcol)
|
||
|
{
|
||
|
l1=subitem->label;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
l1 = L"";
|
||
|
}
|
||
|
if (p2->subitems != NULL) {
|
||
|
for (i=0;i<p2->subitems->getNumItems();i++) {
|
||
|
listSubitemStruct *subitem = p2->subitems->enumItem(i);
|
||
|
if (subitem->column == sortcol) {
|
||
|
l2=subitem->label;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
l2 = L"";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if(!columnsList.enumItem(sortcol)->getNumeric()) {
|
||
|
if(!sortdir) return(WCSICMP(l1, l2));
|
||
|
else return(WCSICMP(l2, l1));
|
||
|
} else {
|
||
|
int a=WTOI(l1),b=WTOI(l2);
|
||
|
if(!sortdir) return a-b;
|
||
|
else return b-a;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
int ListWnd::findItemByParam(LPARAM param) {
|
||
|
for (int i=0;i<itemList.getNumItems();i++) {
|
||
|
if (itemList[i]->data == param)
|
||
|
return i;
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
void ListWnd::setItemParam(int pos, LPARAM param) {
|
||
|
if (pos >= itemList.getNumItems()) return;
|
||
|
itemList[pos]->data = param;
|
||
|
invalidateItem(pos);
|
||
|
}
|
||
|
|
||
|
int ListWnd::deleteByPos(int pos) {
|
||
|
listItem *item = itemList[pos];
|
||
|
if (item == NULL) return 0;
|
||
|
|
||
|
// selItemList.setSelected(pos, FALSE); // If you do not delete the item from the selItemList, you corrupt it for all item positions downstream of this one.
|
||
|
selItemList.deleteByPos(pos);
|
||
|
|
||
|
itemList.removeByPos(pos);
|
||
|
|
||
|
onItemDelete(item->data);
|
||
|
|
||
|
deleteListItem(item); item=NULL;
|
||
|
|
||
|
if (redraw) {
|
||
|
invalidate();
|
||
|
setSlidersPosition();
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
void ListWnd::deleteAllItems() {
|
||
|
bool sav = getRedraw();
|
||
|
deselectAll();
|
||
|
setRedraw(FALSE);
|
||
|
selItemList.deselectAll(); //force desel so as to avoid the MEMCPY
|
||
|
while (itemList.getNumItems()) {
|
||
|
deleteByPos(0);
|
||
|
}
|
||
|
setRedraw(sav);
|
||
|
// setSlidersPosition();
|
||
|
}
|
||
|
|
||
|
void ListWnd::setSubItem(int pos, int subpos, const wchar_t *txt)
|
||
|
{
|
||
|
if (pos >= itemList.getNumItems()) return;
|
||
|
listItem *item = itemList[pos];
|
||
|
if (!item) return;
|
||
|
if (!item->subitems)
|
||
|
item->subitems = new PtrList<listSubitemStruct>;
|
||
|
for (int i=0;i<item->subitems->getNumItems();i++) {
|
||
|
listSubitemStruct *subitem = item->subitems->enumItem(i);
|
||
|
if (subitem->column == subpos) {
|
||
|
if (subitem->label) {
|
||
|
FREE(subitem->label);
|
||
|
subitem->label = NULL;
|
||
|
}
|
||
|
if (txt)
|
||
|
subitem->label = WCSDUP(txt);
|
||
|
invalidateItem(pos);
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
listSubitemStruct *subitem = (listSubitemStruct *) MALLOC(sizeof(listSubitemStruct));
|
||
|
item->subitems->addItem(subitem);
|
||
|
subitem->label = WCSDUP(txt);
|
||
|
subitem->column = subpos;
|
||
|
invalidateItem(pos);
|
||
|
}
|
||
|
|
||
|
SkinBitmap *ListWnd::getItemIcon(int pos)
|
||
|
{
|
||
|
if (pos >= itemList.getNumItems()) return NULL;
|
||
|
listItem *item = itemList[pos];
|
||
|
return item->icon;
|
||
|
}
|
||
|
|
||
|
void ListWnd::setItemIcon(int pos, const wchar_t *bitmapid) {
|
||
|
if (pos >= itemList.getNumItems()) return;
|
||
|
listItem *item = itemList[pos];
|
||
|
item->icon = bitmapid;
|
||
|
invalidateItem(pos);
|
||
|
}
|
||
|
|
||
|
const wchar_t *ListWnd::getSubitemText(int pos, int subpos) {
|
||
|
if (pos >= itemList.getNumItems()) return NULL;
|
||
|
listItem *item = itemList[pos];
|
||
|
if (!item) return NULL;
|
||
|
if (subpos == 0) {
|
||
|
return item->label;
|
||
|
}
|
||
|
if (!item->subitems) return NULL;
|
||
|
for (int i=0;i<item->subitems->getNumItems();i++) {
|
||
|
listSubitemStruct *subitem = item->subitems->enumItem(i);
|
||
|
if (subitem->column == subpos) {
|
||
|
return subitem->label;
|
||
|
}
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
listItem *ListWnd::createListItem() {
|
||
|
listItem *item = listItem_freelist.getRecord();
|
||
|
new(item) listItem();
|
||
|
item->setList(this);
|
||
|
return item;
|
||
|
}
|
||
|
|
||
|
void ListWnd::deleteListItem(listItem *item) {
|
||
|
if (item == NULL) return;
|
||
|
item->~listItem();
|
||
|
listItem_freelist.freeRecord(item);
|
||
|
}
|
||
|
|
||
|
ListColumn *ListWnd::enumListColumn(int pos) {
|
||
|
return columnsList[pos];
|
||
|
}
|
||
|
|
||
|
int ListWnd::getColumnPosByName(const wchar_t *name)
|
||
|
{
|
||
|
for (int i = 0; i < columnsList.getNumItems(); i++)
|
||
|
{
|
||
|
const wchar_t *name2 = columnsList[i]->getName();
|
||
|
if (name2 == NULL) continue;
|
||
|
if (!wcscmp(name, name2)) return i;
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::delColumnByPos(int pos) {
|
||
|
if (pos < 0 || pos >= columnsList.getNumItems()) return 0;
|
||
|
delete columnsList[pos];
|
||
|
columnsList.removeByPos(pos);
|
||
|
recalcHeaders();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
void ListWnd::recalcHeaders() {
|
||
|
if (getNumColumns() <= 0) return;
|
||
|
|
||
|
int wid = 0, ndynamic = 0;
|
||
|
for (int i = 0; i < getNumColumns(); i++) {
|
||
|
ListColumn *col = enumListColumn(i);
|
||
|
if (!col->isDynamic()) wid += col->getWidth()+COLUMNS_MARGIN-1;
|
||
|
else ndynamic++;
|
||
|
}
|
||
|
if (ndynamic == 0) return;
|
||
|
|
||
|
RECT r = clientRect();
|
||
|
int wwidth = (r.right - r.left) - 2;
|
||
|
|
||
|
int leftover = wwidth - wid;
|
||
|
if (leftover <= 1) return;
|
||
|
|
||
|
leftover--;
|
||
|
leftover /= ndynamic;
|
||
|
|
||
|
for (int i = 0; i < getNumColumns(); i++) {
|
||
|
ListColumn *col = enumListColumn(i);
|
||
|
if (col->isDynamic()) col->setWidth(leftover);
|
||
|
}
|
||
|
|
||
|
//BU note: we could probably find a way to not invalidate everything
|
||
|
invalidate();
|
||
|
}
|
||
|
|
||
|
void ListWnd::itemSelection(int itemnum, int selected) {
|
||
|
notifySelChanged(itemnum, selected);
|
||
|
onItemSelection(itemnum, selected);
|
||
|
}
|
||
|
|
||
|
void ListWnd::onItemSelection(int itemnum, int selected) {
|
||
|
if (selected) {
|
||
|
Accessible *a = getAccessibleObject();
|
||
|
if (a != NULL)
|
||
|
a->onGetFocus(itemnum);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int ListWnd::doAddItem(const wchar_t *label, LPARAM lParam, int pos)
|
||
|
{
|
||
|
listItem *item = createListItem();
|
||
|
item->label = label;
|
||
|
item->data = lParam;
|
||
|
|
||
|
itemList.addItem(item, pos, ITEMLIST_INC);
|
||
|
lastAddedItem = item;
|
||
|
|
||
|
if (redraw)
|
||
|
{
|
||
|
invalidate(); // todo: optimize to invalidate only if necessary
|
||
|
setSlidersPosition();
|
||
|
}
|
||
|
|
||
|
if (isInited()) calcBounds();
|
||
|
int p = (pos == POS_LAST) ? itemList.getNumItems()-1 : pos;
|
||
|
if (p <= selectionStart) selectionStart++;
|
||
|
if (autosort) {
|
||
|
itemList.sort();
|
||
|
p = itemList.searchItem(item);
|
||
|
}
|
||
|
return p;
|
||
|
}
|
||
|
|
||
|
void ListWnd::setMinimumSize(int size) {
|
||
|
if (size > 0) itemList.setMinimumSize(size);
|
||
|
}
|
||
|
|
||
|
int ListWnd::addItem(const wchar_t *label, LPARAM lParam)
|
||
|
{
|
||
|
return doAddItem(label, lParam, POS_LAST);
|
||
|
}
|
||
|
|
||
|
int ListWnd::insertItem(int pos, const wchar_t *label, LPARAM lParam)
|
||
|
{
|
||
|
return doAddItem(label, lParam, pos);
|
||
|
}
|
||
|
|
||
|
int ListWnd::getLastAddedItemPos() {
|
||
|
if (lastAddedItem == NULL) return -1;
|
||
|
return itemList.searchItem(lastAddedItem);
|
||
|
}
|
||
|
|
||
|
int ListWnd::getColumnsHeight() {
|
||
|
return columnsHeight;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getColumnsWidth() {
|
||
|
int i, x=0;
|
||
|
for (i=0;i<columnsList.getNumItems();i++) x+= columnsList[i]->getWidth();
|
||
|
return x+1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::insertColumn(ListColumn *col, int pos, int alignment)
|
||
|
{
|
||
|
ASSERT(col != NULL);
|
||
|
ASSERT(pos >= -1);
|
||
|
if (pos < 0) col->setIndex(columnsList.getNumItems());
|
||
|
else col->setIndex(pos);
|
||
|
col->setList(this);
|
||
|
col->setAlignment(alignment);
|
||
|
columnsList.addItem(col);
|
||
|
if (pos >= 0) {
|
||
|
columnsList.moveItem(columnsList.getNumItems()-1, pos);
|
||
|
}
|
||
|
if (redraw && isInited()) {
|
||
|
invalidate();
|
||
|
setSlidersPosition();
|
||
|
}
|
||
|
recalcHeaders();
|
||
|
return columnsList.getNumItems();
|
||
|
}
|
||
|
|
||
|
void ListWnd::deleteAllColumns() {
|
||
|
columnsList.deleteAll();
|
||
|
if (redraw && isInited()) {
|
||
|
invalidate();
|
||
|
setSlidersPosition();
|
||
|
}
|
||
|
recalcHeaders();
|
||
|
}
|
||
|
|
||
|
int ListWnd::addColumn(const wchar_t *name, int width, int numeric, int align)
|
||
|
{
|
||
|
ListColumn *col = new ListColumn();
|
||
|
col->setWidth(width);
|
||
|
col->setLabel(name);
|
||
|
col->setNumeric(numeric);
|
||
|
col->setAlignment(align);
|
||
|
return insertColumn(col, -1, align);
|
||
|
}
|
||
|
|
||
|
bool ListWnd::setRedraw(bool _redraw) {
|
||
|
bool prev = redraw;
|
||
|
if (!redraw && _redraw) {
|
||
|
invalidate();
|
||
|
setSlidersPosition();
|
||
|
notifySelChanged();
|
||
|
}
|
||
|
redraw = _redraw;
|
||
|
return prev;
|
||
|
}
|
||
|
|
||
|
bool ListWnd::getRedraw() {
|
||
|
return redraw;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onChar(unsigned int c) {
|
||
|
//CT> Commented this out so shortcuts work in playlist editor
|
||
|
/*char b = TOUPPER(c);
|
||
|
if (b >= 'A' && b <= 'Z') {
|
||
|
jumpToNext(b);
|
||
|
return 1;
|
||
|
}*/
|
||
|
|
||
|
return LISTWND_PARENT::onChar(c);
|
||
|
}
|
||
|
|
||
|
int ListWnd::onKeyDown(int keyCode) {
|
||
|
switch (keyCode) {
|
||
|
case VK_DOWN:
|
||
|
next(selectonupdown);
|
||
|
return 1;
|
||
|
case VK_UP:
|
||
|
previous(selectonupdown);
|
||
|
return 1;
|
||
|
case VK_PRIOR:
|
||
|
pageup(selectonupdown);
|
||
|
return 1;
|
||
|
case VK_NEXT:
|
||
|
pagedown(selectonupdown);
|
||
|
return 1;
|
||
|
case VK_HOME:
|
||
|
home(selectonupdown);
|
||
|
return 1;
|
||
|
case VK_END:
|
||
|
end(selectonupdown);
|
||
|
return 1;
|
||
|
case VK_DELETE:
|
||
|
onDelete();
|
||
|
return 1;
|
||
|
case VK_RETURN: {
|
||
|
int i=getItemFocused();
|
||
|
//setSelected(i, 0, 0);
|
||
|
//setSelected(i, 1, 1);
|
||
|
if(i!=-1)
|
||
|
onDoubleClick(i);
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
return LISTWND_PARENT::onKeyDown(keyCode);
|
||
|
}
|
||
|
|
||
|
void ListWnd::next(int wantcb) {
|
||
|
int from=getItemFocused();
|
||
|
/* if (selItemList.getNumItems() > 0)
|
||
|
for (int i=0;i<itemList.getNumItems();i++)
|
||
|
if (getItemSelected(i)) {
|
||
|
from = i;
|
||
|
break;
|
||
|
}*/
|
||
|
int to = from + 1;
|
||
|
if (to < itemList.getNumItems() && to >= 0) {
|
||
|
setSelectionStart(to, wantcb);
|
||
|
if (!fullyVisible(to)) {
|
||
|
RECT c;
|
||
|
getClientRect(&c);
|
||
|
scrollToY((Y_SHIFT*2+(to+1)*(getItemHeight()+(wantColSepOnItems()?COLSEPHEIGHT:0)))-(c.bottom-c.top));
|
||
|
}
|
||
|
if (!wantcb)
|
||
|
{
|
||
|
wchar_t t[256]=L"";
|
||
|
getItemLabel(to, 0, t, 255);
|
||
|
foreach(tempselectnotifies)
|
||
|
sendAction(tempselectnotifies.getfor(), L"tempselectnotify", t);
|
||
|
endfor;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ListWnd::selectCurrent() {
|
||
|
int from=getItemFocused();
|
||
|
int to = from;
|
||
|
if (to < itemList.getNumItems() && to >= 0) {
|
||
|
setSelectionStart(to);
|
||
|
if (!fullyVisible(to)) {
|
||
|
RECT c;
|
||
|
getClientRect(&c);
|
||
|
scrollToY((Y_SHIFT*2+(to+1)*(getItemHeight()+(wantColSepOnItems()?COLSEPHEIGHT:0)))-(c.bottom-c.top));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ListWnd::selectFirstEntry(int wantcb) {
|
||
|
setSelectionStart(0, wantcb);
|
||
|
ensureItemVisible(0);
|
||
|
}
|
||
|
|
||
|
void ListWnd::previous(int wantcb) {
|
||
|
int from=0;
|
||
|
/* if (selItemList.getNumItems() > 0)
|
||
|
for (int i=0;i<itemList.getNumItems();i++)
|
||
|
if (getItemSelected(i)) {
|
||
|
from = i;
|
||
|
break;
|
||
|
}*/
|
||
|
from = getItemFocused();
|
||
|
int to = from - 1;
|
||
|
if (to < itemList.getNumItems() && to >= 0) {
|
||
|
setSelectionStart(to, wantcb);
|
||
|
ensureItemVisible(to);
|
||
|
if (!wantcb) {
|
||
|
wchar_t t[256]=L"";
|
||
|
getItemLabel(to, 0, t, 255);
|
||
|
foreach(tempselectnotifies)
|
||
|
sendAction(tempselectnotifies.getfor(), L"tempselectnotify", t);
|
||
|
endfor;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ListWnd::pagedown(int wantcb) {
|
||
|
int from=-1,to;
|
||
|
if (selItemList.getNumSelected())
|
||
|
for (int i=0;i<itemList.getNumItems();i++)
|
||
|
if (getItemSelected(i)) {
|
||
|
from = i;
|
||
|
break;
|
||
|
}
|
||
|
if(from==-1) to = 0;
|
||
|
else to = from + getLinesPerPage();
|
||
|
to=MIN(to,itemList.getNumItems()-1);
|
||
|
if(to>=0) {
|
||
|
setSelectionStart(to, wantcb);
|
||
|
if (!fullyVisible(to)) {
|
||
|
RECT c;
|
||
|
getClientRect(&c);
|
||
|
scrollToY((Y_SHIFT*2+(to+1)*(getItemHeight()+(wantColSepOnItems()?COLSEPHEIGHT:0)))-(c.bottom-c.top));
|
||
|
}
|
||
|
if (!wantcb) {
|
||
|
wchar_t t[256]=L"";
|
||
|
getItemLabel(to, 0, t, 255);
|
||
|
foreach(tempselectnotifies)
|
||
|
sendAction(tempselectnotifies.getfor(), L"tempselectnotify", t);
|
||
|
endfor;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ListWnd::pageup(int wantcb) {
|
||
|
int from=-1,to;
|
||
|
if (selItemList.getNumSelected())
|
||
|
for (int i=0;i<itemList.getNumItems();i++)
|
||
|
if (getItemSelected(i)) {
|
||
|
from = i;
|
||
|
break;
|
||
|
}
|
||
|
if(from==-1) to = 0;
|
||
|
else to = from - getLinesPerPage();
|
||
|
to=MAX(to,0);
|
||
|
to=MIN(to,itemList.getNumItems()-1);
|
||
|
if(to>=0) {
|
||
|
setSelectionStart(to, wantcb);
|
||
|
ensureItemVisible(to);
|
||
|
if (!wantcb) {
|
||
|
wchar_t t[256]=L"";
|
||
|
getItemLabel(to, 0, t, 255);
|
||
|
foreach(tempselectnotifies)
|
||
|
sendAction(tempselectnotifies.getfor(), L"tempselectnotify", t);
|
||
|
endfor;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ListWnd::home(int wantcb) {
|
||
|
if(!itemList.getNumItems()) return;
|
||
|
setSelectionStart(0, wantcb);
|
||
|
ensureItemVisible(0);
|
||
|
if (!wantcb) {
|
||
|
wchar_t t[256]=L"";
|
||
|
getItemLabel(0, 0, t, 255);
|
||
|
foreach(tempselectnotifies)
|
||
|
sendAction(tempselectnotifies.getfor(), L"tempselectnotify", t);
|
||
|
endfor;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ListWnd::end(int wantcb) {
|
||
|
if(!itemList.getNumItems()) return;
|
||
|
int i=itemList.getNumItems()-1;
|
||
|
setSelectionStart(i, wantcb);
|
||
|
if (!fullyVisible(i)) {
|
||
|
RECT c;
|
||
|
getClientRect(&c);
|
||
|
scrollToY((Y_SHIFT*2+(i+1)*(getItemHeight()+(wantColSepOnItems()?COLSEPHEIGHT:0)))-(c.bottom-c.top));
|
||
|
}
|
||
|
if (!wantcb) {
|
||
|
wchar_t t[256]=L"";
|
||
|
getItemLabel(i, 0, t, 255);
|
||
|
foreach(tempselectnotifies)
|
||
|
sendAction(tempselectnotifies.getfor(), L"tempselectnotify", t);
|
||
|
endfor;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ListWnd::jumpToNext(wchar_t c)
|
||
|
{
|
||
|
if (doJumpToNext(c, FALSE)) return;
|
||
|
doJumpToNext(c, TRUE);
|
||
|
}
|
||
|
|
||
|
int ListWnd::doJumpToNext(wchar_t c, bool fromtop)
|
||
|
{
|
||
|
int from = 0;
|
||
|
if (!fromtop && selItemList.getNumSelected())
|
||
|
{
|
||
|
for (int i=0;i<itemList.getNumItems();i++)
|
||
|
if (getItemSelected(i))
|
||
|
{
|
||
|
from = i+1;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
for (int j=from;j<itemList.getNumItems();j++)
|
||
|
{
|
||
|
listItem *item = itemList[j];
|
||
|
if (item->label != NULL) {
|
||
|
wchar_t z = TOUPPERW(*(item->label));
|
||
|
if (z == c)
|
||
|
{
|
||
|
setSelectionStart(j);
|
||
|
ensureItemVisible(j);
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void ListWnd::reset() {
|
||
|
columnsList.deleteAll();
|
||
|
deleteAllItems();
|
||
|
}
|
||
|
|
||
|
void ListWnd::setShowColumnsHeaders(int show) {
|
||
|
int prev = showColumnsHeaders;
|
||
|
showColumnsHeaders = !!show;
|
||
|
if (prev != show) {
|
||
|
invalidate();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int ListWnd::onContextMenu (int x, int y) {
|
||
|
return notifyParent(ChildNotify::LISTWND_POPUPMENU, x, y);
|
||
|
}
|
||
|
|
||
|
void ListWnd::scrollUp(int lines) {
|
||
|
scrollToY(MAX(0, getScrollY()-getItemHeight()*lines));
|
||
|
}
|
||
|
|
||
|
void ListWnd::scrollLeft(int lines) {
|
||
|
scrollToX(MAX(0, getScrollX()-getItemHeight()*lines));
|
||
|
}
|
||
|
|
||
|
void ListWnd::scrollDown(int lines) {
|
||
|
scrollToY(MIN(getMaxScrollY(), getScrollY()+getItemHeight()*lines));
|
||
|
}
|
||
|
|
||
|
void ListWnd::scrollRight(int lines) {
|
||
|
scrollToX(MIN(getMaxScrollX(), getScrollX()+getItemHeight()*lines));
|
||
|
}
|
||
|
|
||
|
int ListWnd::onMouseWheelUp(int clicked, int lines) {
|
||
|
lines *= Wasabi::Std::osparam_getScrollLines();
|
||
|
if (!clicked)
|
||
|
scrollUp(lines);
|
||
|
else
|
||
|
scrollLeft(lines);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onMouseWheelDown(int clicked, int lines) {
|
||
|
lines *= Wasabi::Std::osparam_getScrollLines();
|
||
|
if (!clicked)
|
||
|
scrollDown(lines);
|
||
|
else
|
||
|
scrollRight(lines);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onColumnLabelClick(int col, int x, int y)
|
||
|
{
|
||
|
if(lastcolsort==col) {
|
||
|
setSortDirection(1);
|
||
|
lastcolsort=-1;
|
||
|
} else {
|
||
|
setSortDirection(0);
|
||
|
lastcolsort=col;
|
||
|
}
|
||
|
setSortColumn(col);
|
||
|
resort();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
ARGB32 ListWnd::getTextColor(LPARAM lParam) {
|
||
|
return textColor;
|
||
|
}
|
||
|
|
||
|
ARGB32 ListWnd::getSelBgColor(LPARAM LParam) {
|
||
|
return color_item_selected;
|
||
|
}
|
||
|
|
||
|
ARGB32 ListWnd::getSelFgColor(LPARAM LParam) {
|
||
|
ARGB32 r = color_item_selected_fg;
|
||
|
if (r == 0xFFFFFFFF)
|
||
|
return color_item_selected_fg;
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
ARGB32 ListWnd::getBgColor() {
|
||
|
return bgcolor;
|
||
|
}
|
||
|
|
||
|
ARGB32 ListWnd::getFocusColor(LPARAM LParam) {
|
||
|
return color_item_focused;
|
||
|
}
|
||
|
|
||
|
ARGB32 ListWnd::getFocusRectColor(LPARAM lParam) {
|
||
|
return color_item_focusrect;
|
||
|
}
|
||
|
|
||
|
|
||
|
void ListWnd::moveItem(int from, int to) {
|
||
|
itemList.moveItem(from,to);
|
||
|
invalidate();
|
||
|
}
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
// ListColumn
|
||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
|
||
|
ListColumn::ListColumn(const wchar_t *name, int isdynamic)
|
||
|
: NamedW(name), dynamic(isdynamic)
|
||
|
{
|
||
|
align=COL_LEFTALIGN;
|
||
|
index = -1;
|
||
|
width = COLUMNS_DEFAULT_WIDTH;
|
||
|
list = NULL;
|
||
|
numeric = 0;
|
||
|
}
|
||
|
|
||
|
void ListColumn::setLabel(const wchar_t *newlabel) {
|
||
|
setName(newlabel);
|
||
|
}
|
||
|
|
||
|
const wchar_t *ListColumn::getLabel() {
|
||
|
return getName();
|
||
|
}
|
||
|
|
||
|
void ListColumn::setIndex(int newindex) {
|
||
|
index = newindex;
|
||
|
}
|
||
|
|
||
|
int ListColumn::getIndex() {
|
||
|
return index;
|
||
|
}
|
||
|
|
||
|
void ListColumn::setWidth(int _width) {
|
||
|
width = _width;
|
||
|
if (list && list->getRedraw()) {
|
||
|
list->invalidate();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int ListColumn::getWidth() {
|
||
|
return width;
|
||
|
}
|
||
|
|
||
|
int ListColumn::customDrawHeader(Canvas *c, RECT *r, const Wasabi::FontInfo *fontInfo)
|
||
|
{
|
||
|
int y = (r->bottom-r->top-c->getTextHeight(fontInfo)) / 2;
|
||
|
c->textOutEllipsed(r->left, y, r->right-r->left, c->getTextHeight(fontInfo), _(getName()), fontInfo);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
void ListColumn::setDynamic(int isdynamic)
|
||
|
{
|
||
|
int prev = dynamic;
|
||
|
dynamic = !!isdynamic;
|
||
|
if (prev != dynamic && dynamic && list != NULL)
|
||
|
list->recalcHeaders();
|
||
|
}
|
||
|
|
||
|
void ListColumn::setList(ListWnd *_list) {
|
||
|
list = _list;
|
||
|
}
|
||
|
|
||
|
ListWnd *ListColumn::getList() {
|
||
|
return list;
|
||
|
}
|
||
|
|
||
|
int ListWnd::wantAutoContextMenu() {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onAcceleratorEvent(const wchar_t *name) {
|
||
|
if(!_wcsicmp(name, L"selectall")) {
|
||
|
selectAll();
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int ListWnd::onAction(const wchar_t *action, const wchar_t *param, int x, int y, intptr_t p1, intptr_t p2, void *data, size_t datalen, ifc_window *source) {
|
||
|
int r = LISTWND_PARENT::onAction(action, param, x, y, p1, p2, data, datalen, source);
|
||
|
if (WCSCASEEQLSAFE(action, L"register_tempselectnotify")) {
|
||
|
tempselectnotifies.addItem(source);
|
||
|
}
|
||
|
else if (WCSCASEEQLSAFE(action, L"up")) {
|
||
|
previous((int)p1);
|
||
|
}
|
||
|
else if (WCSCASEEQLSAFE(action, L"down")) {
|
||
|
next((int)p1);
|
||
|
}
|
||
|
else if (WCSCASEEQLSAFE(action, L"home")) {
|
||
|
home((int)p1);
|
||
|
}
|
||
|
else if (WCSCASEEQLSAFE(action, L"end")) {
|
||
|
end((int)p1);
|
||
|
}
|
||
|
else if (WCSCASEEQLSAFE(action, L"pageup")) {
|
||
|
pageup((int)p1);
|
||
|
}
|
||
|
else if (WCSCASEEQLSAFE(action, L"pagedown")) {
|
||
|
pagedown((int)p1);
|
||
|
}
|
||
|
else if (WCSCASEEQLSAFE(action, L"select_current")) {
|
||
|
selectCurrent();
|
||
|
}
|
||
|
else if (WCSCASEEQLSAFE(action, L"doubleclick")) {
|
||
|
int pos = getItemFocused();
|
||
|
if (pos >= 0) onDoubleClick(pos);
|
||
|
return 1;
|
||
|
}
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
void ListWnd::setShowIcons(int icons)
|
||
|
{
|
||
|
showicons = icons;
|
||
|
invalidate();
|
||
|
}
|
||
|
|
||
|
int ListWnd::getShowIcons ()
|
||
|
{
|
||
|
return showicons;
|
||
|
}
|
||
|
|
||
|
int ListWnd::getIconWidth ()
|
||
|
{
|
||
|
// The old behaviour used the itemheight as value, so we return this for backwards compatibility if iconWidth is negative
|
||
|
return (iconWidth < 0 ? itemHeight-2 : iconWidth);
|
||
|
}
|
||
|
|
||
|
void ListWnd::setIconWidth (int width)
|
||
|
{
|
||
|
iconWidth = width;
|
||
|
invalidate();
|
||
|
}
|
||
|
|
||
|
int ListWnd::getIconHeight ()
|
||
|
{
|
||
|
// The old behaviour used the itemheight as value, so we return this for backwards compatibility if iconWidth is negative
|
||
|
return (iconHeight < 0 ? itemHeight-2 : iconHeight);
|
||
|
}
|
||
|
|
||
|
void ListWnd::setIconHeight (int height)
|
||
|
{
|
||
|
iconHeight = height;
|
||
|
invalidate();
|
||
|
}
|
||
|
|
||
|
int ListWnd::getItemHeight ()
|
||
|
{
|
||
|
return (itemHeight < getIconHeight()+2) ? getIconHeight()+2 : itemHeight;
|
||
|
}
|
||
|
|
||
|
void ListWnd::setItemHeight (int height, bool forceInvalidate /*true*/)
|
||
|
{
|
||
|
itemHeight = height;
|
||
|
if (forceInvalidate) invalidate();
|
||
|
}
|