mirror of
https://github.com/reactos/reactos.git
synced 2025-01-01 12:04:51 +00:00
470 lines
13 KiB
C
470 lines
13 KiB
C
|
/* ------------- listbox.c ------------ */
|
||
|
|
||
|
#include "dflat.h"
|
||
|
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
static int ExtendSelections(DFWINDOW, int, int);
|
||
|
static void TestExtended(DFWINDOW, PARAM);
|
||
|
static void ClearAllSelections(DFWINDOW);
|
||
|
static void SetSelection(DFWINDOW, int);
|
||
|
static void FlipSelection(DFWINDOW, int);
|
||
|
static void ClearSelection(DFWINDOW, int);
|
||
|
#else
|
||
|
#define TestExtended(w,p) /**/
|
||
|
#endif
|
||
|
static void ChangeSelection(DFWINDOW, int, int);
|
||
|
static void WriteSelection(DFWINDOW, int, int, DFRECT *);
|
||
|
static BOOL SelectionInWindow(DFWINDOW, int);
|
||
|
|
||
|
static int py = -1; /* the previous y mouse coordinate */
|
||
|
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
/* --------- SHIFT_F8 Key ------------ */
|
||
|
static void AddModeKey(DFWINDOW wnd)
|
||
|
{
|
||
|
if (isMultiLine(wnd)) {
|
||
|
wnd->AddMode ^= TRUE;
|
||
|
DfSendMessage(GetParent(wnd), ADDSTATUS,
|
||
|
wnd->AddMode ? ((PARAM) "Add Mode") : 0, 0);
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
/* --------- UP (Up Arrow) Key ------------ */
|
||
|
static void UpKey(DFWINDOW wnd, PARAM p2)
|
||
|
{
|
||
|
if (wnd->selection > 0) {
|
||
|
if (wnd->selection == wnd->wtop) {
|
||
|
BaseWndProc(LISTBOX, wnd, KEYBOARD, UP, p2);
|
||
|
DfPostMessage(wnd, LB_SELECTION, wnd->selection-1,
|
||
|
isMultiLine(wnd) ? p2 : FALSE);
|
||
|
}
|
||
|
else {
|
||
|
int newsel = wnd->selection-1;
|
||
|
if (wnd->wlines == ClientHeight(wnd))
|
||
|
while (*TextLine(wnd, newsel) == LINE)
|
||
|
--newsel;
|
||
|
DfPostMessage(wnd, LB_SELECTION, newsel,
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
isMultiLine(wnd) ? p2 :
|
||
|
#endif
|
||
|
FALSE);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* --------- DN (Down Arrow) Key ------------ */
|
||
|
static void DnKey(DFWINDOW wnd, PARAM p2)
|
||
|
{
|
||
|
if (wnd->selection < wnd->wlines-1) {
|
||
|
if (wnd->selection == wnd->wtop+ClientHeight(wnd)-1) {
|
||
|
BaseWndProc(LISTBOX, wnd, KEYBOARD, DN, p2);
|
||
|
DfPostMessage(wnd, LB_SELECTION, wnd->selection+1,
|
||
|
isMultiLine(wnd) ? p2 : FALSE);
|
||
|
}
|
||
|
else {
|
||
|
int newsel = wnd->selection+1;
|
||
|
if (wnd->wlines == ClientHeight(wnd))
|
||
|
while (*TextLine(wnd, newsel) == LINE)
|
||
|
newsel++;
|
||
|
DfPostMessage(wnd, LB_SELECTION, newsel,
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
isMultiLine(wnd) ? p2 :
|
||
|
#endif
|
||
|
FALSE);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* --------- HOME and PGUP Keys ------------ */
|
||
|
static void HomePgUpKey(DFWINDOW wnd, PARAM p1, PARAM p2)
|
||
|
{
|
||
|
BaseWndProc(LISTBOX, wnd, KEYBOARD, p1, p2);
|
||
|
DfPostMessage(wnd, LB_SELECTION, wnd->wtop,
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
isMultiLine(wnd) ? p2 :
|
||
|
#endif
|
||
|
FALSE);
|
||
|
}
|
||
|
|
||
|
/* --------- END and PGDN Keys ------------ */
|
||
|
static void EndPgDnKey(DFWINDOW wnd, PARAM p1, PARAM p2)
|
||
|
{
|
||
|
int bot;
|
||
|
BaseWndProc(LISTBOX, wnd, KEYBOARD, p1, p2);
|
||
|
bot = wnd->wtop+ClientHeight(wnd)-1;
|
||
|
if (bot > wnd->wlines-1)
|
||
|
bot = wnd->wlines-1;
|
||
|
DfPostMessage(wnd, LB_SELECTION, bot,
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
isMultiLine(wnd) ? p2 :
|
||
|
#endif
|
||
|
FALSE);
|
||
|
}
|
||
|
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
/* --------- Space Bar Key ------------ */
|
||
|
static void SpacebarKey(DFWINDOW wnd, PARAM p2)
|
||
|
{
|
||
|
if (isMultiLine(wnd)) {
|
||
|
int sel = DfSendMessage(wnd, LB_CURRENTSELECTION, 0, 0);
|
||
|
if (sel != -1) {
|
||
|
if (wnd->AddMode)
|
||
|
FlipSelection(wnd, sel);
|
||
|
if (ItemSelected(wnd, sel)) {
|
||
|
if (!((int) p2 & (LEFTSHIFT | RIGHTSHIFT)))
|
||
|
wnd->AnchorPoint = sel;
|
||
|
ExtendSelections(wnd, sel, (int) p2);
|
||
|
}
|
||
|
else
|
||
|
wnd->AnchorPoint = -1;
|
||
|
DfSendMessage(wnd, PAINT, 0, 0);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
/* --------- Enter ('\r') Key ------------ */
|
||
|
static void EnterKey(DFWINDOW wnd)
|
||
|
{
|
||
|
if (wnd->selection != -1) {
|
||
|
DfSendMessage(wnd, LB_SELECTION, wnd->selection, TRUE);
|
||
|
DfSendMessage(wnd, LB_CHOOSE, wnd->selection, 0);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* --------- All Other Key Presses ------------ */
|
||
|
static void KeyPress(DFWINDOW wnd, PARAM p1, PARAM p2)
|
||
|
{
|
||
|
int sel = wnd->selection+1;
|
||
|
while (sel < wnd->wlines) {
|
||
|
char *cp = TextLine(wnd, sel);
|
||
|
if (cp == NULL)
|
||
|
break;
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
if (isMultiLine(wnd))
|
||
|
cp++;
|
||
|
#endif
|
||
|
/* --- special for directory list box --- */
|
||
|
if (*cp == '[')
|
||
|
cp++;
|
||
|
if (tolower(*cp) == (int)p1) {
|
||
|
DfSendMessage(wnd, LB_SELECTION, sel,
|
||
|
isMultiLine(wnd) ? p2 : FALSE);
|
||
|
if (!SelectionInWindow(wnd, sel)) {
|
||
|
wnd->wtop = sel-ClientHeight(wnd)+1;
|
||
|
DfSendMessage(wnd, PAINT, 0, 0);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
sel++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* --------- KEYBOARD Message ------------ */
|
||
|
static int KeyboardMsg(DFWINDOW wnd, PARAM p1, PARAM p2)
|
||
|
{
|
||
|
switch ((int) p1) {
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
case SHIFT_F8:
|
||
|
AddModeKey(wnd);
|
||
|
return TRUE;
|
||
|
#endif
|
||
|
case UP:
|
||
|
TestExtended(wnd, p2);
|
||
|
UpKey(wnd, p2);
|
||
|
return TRUE;
|
||
|
case DN:
|
||
|
TestExtended(wnd, p2);
|
||
|
DnKey(wnd, p2);
|
||
|
return TRUE;
|
||
|
case PGUP:
|
||
|
case HOME:
|
||
|
TestExtended(wnd, p2);
|
||
|
HomePgUpKey(wnd, p1, p2);
|
||
|
return TRUE;
|
||
|
case PGDN:
|
||
|
case END:
|
||
|
TestExtended(wnd, p2);
|
||
|
EndPgDnKey(wnd, p1, p2);
|
||
|
return TRUE;
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
case ' ':
|
||
|
SpacebarKey(wnd, p2);
|
||
|
break;
|
||
|
#endif
|
||
|
case '\r':
|
||
|
EnterKey(wnd);
|
||
|
return TRUE;
|
||
|
default:
|
||
|
KeyPress(wnd, p1, p2);
|
||
|
break;
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
/* ------- LEFT_BUTTON Message -------- */
|
||
|
static int LeftButtonMsg(DFWINDOW wnd, PARAM p1, PARAM p2)
|
||
|
{
|
||
|
int my = (int) p2 - GetTop(wnd);
|
||
|
if (my >= wnd->wlines-wnd->wtop)
|
||
|
my = wnd->wlines - wnd->wtop;
|
||
|
|
||
|
if (!InsideRect(p1, p2, ClientRect(wnd)))
|
||
|
return FALSE;
|
||
|
if (wnd->wlines && my != py) {
|
||
|
int sel = wnd->wtop+my-1;
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
int sh = getshift();
|
||
|
if (!(sh & (LEFTSHIFT | RIGHTSHIFT))) {
|
||
|
if (!(sh & CTRLKEY))
|
||
|
ClearAllSelections(wnd);
|
||
|
wnd->AnchorPoint = sel;
|
||
|
DfSendMessage(wnd, PAINT, 0, 0);
|
||
|
}
|
||
|
#endif
|
||
|
DfSendMessage(wnd, LB_SELECTION, sel, TRUE);
|
||
|
py = my;
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/* ------------- DOUBLE_CLICK Message ------------ */
|
||
|
static int DoubleClickMsg(DFWINDOW wnd, PARAM p1, PARAM p2)
|
||
|
{
|
||
|
if (WindowMoving || WindowSizing)
|
||
|
return FALSE;
|
||
|
if (wnd->wlines) {
|
||
|
DFRECT rc = ClientRect(wnd);
|
||
|
BaseWndProc(LISTBOX, wnd, DOUBLE_CLICK, p1, p2);
|
||
|
if (InsideRect(p1, p2, rc))
|
||
|
DfSendMessage(wnd, LB_CHOOSE, wnd->selection, 0);
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/* ------------ ADDTEXT Message -------------- */
|
||
|
static int AddTextMsg(DFWINDOW wnd, PARAM p1, PARAM p2)
|
||
|
{
|
||
|
int rtn = BaseWndProc(LISTBOX, wnd, ADDTEXT, p1, p2);
|
||
|
if (wnd->selection == -1)
|
||
|
DfSendMessage(wnd, LB_SETSELECTION, 0, 0);
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
if (*(char *)p1 == LISTSELECTOR)
|
||
|
wnd->SelectCount++;
|
||
|
#endif
|
||
|
return rtn;
|
||
|
}
|
||
|
|
||
|
/* --------- GETTEXT Message ------------ */
|
||
|
static void GetTextMsg(DFWINDOW wnd, PARAM p1, PARAM p2)
|
||
|
{
|
||
|
if ((int)p2 != -1) {
|
||
|
char *cp1 = (char *)p1;
|
||
|
char *cp2 = TextLine(wnd, (int)p2);
|
||
|
while (cp2 && *cp2 && *cp2 != '\n')
|
||
|
*cp1++ = *cp2++;
|
||
|
*cp1 = '\0';
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* --------- LISTBOX Window Processing Module ------------ */
|
||
|
int ListBoxProc(DFWINDOW wnd, DFMESSAGE msg, PARAM p1, PARAM p2)
|
||
|
{
|
||
|
switch (msg) {
|
||
|
case CREATE_WINDOW:
|
||
|
BaseWndProc(LISTBOX, wnd, msg, p1, p2);
|
||
|
wnd->selection = -1;
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
wnd->AnchorPoint = -1;
|
||
|
#endif
|
||
|
return TRUE;
|
||
|
case KEYBOARD:
|
||
|
if (WindowMoving || WindowSizing)
|
||
|
break;
|
||
|
if (KeyboardMsg(wnd, p1, p2))
|
||
|
return TRUE;
|
||
|
break;
|
||
|
case LEFT_BUTTON:
|
||
|
if (LeftButtonMsg(wnd, p1, p2) == TRUE)
|
||
|
return TRUE;
|
||
|
break;
|
||
|
case DOUBLE_CLICK:
|
||
|
if (DoubleClickMsg(wnd, p1, p2))
|
||
|
return TRUE;
|
||
|
break;
|
||
|
case DFM_BUTTON_RELEASED:
|
||
|
if (WindowMoving || WindowSizing || VSliding)
|
||
|
break;
|
||
|
py = -1;
|
||
|
return TRUE;
|
||
|
case ADDTEXT:
|
||
|
return AddTextMsg(wnd, p1, p2);
|
||
|
case DFM_LB_GETTEXT:
|
||
|
GetTextMsg(wnd, p1, p2);
|
||
|
return TRUE;
|
||
|
case CLEARTEXT:
|
||
|
wnd->selection = -1;
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
wnd->AnchorPoint = -1;
|
||
|
#endif
|
||
|
wnd->SelectCount = 0;
|
||
|
break;
|
||
|
case PAINT:
|
||
|
BaseWndProc(LISTBOX, wnd, msg, p1, p2);
|
||
|
WriteSelection(wnd, wnd->selection, TRUE, (DFRECT *)p1);
|
||
|
return TRUE;
|
||
|
case SCROLL:
|
||
|
case HORIZSCROLL:
|
||
|
case SCROLLPAGE:
|
||
|
case HORIZPAGE:
|
||
|
case SCROLLDOC:
|
||
|
BaseWndProc(LISTBOX, wnd, msg, p1, p2);
|
||
|
WriteSelection(wnd,wnd->selection,TRUE,NULL);
|
||
|
return TRUE;
|
||
|
case LB_CHOOSE:
|
||
|
DfSendMessage(GetParent(wnd), LB_CHOOSE, p1, p2);
|
||
|
return TRUE;
|
||
|
case LB_SELECTION:
|
||
|
ChangeSelection(wnd, (int) p1, (int) p2);
|
||
|
DfSendMessage(GetParent(wnd), LB_SELECTION,
|
||
|
wnd->selection, 0);
|
||
|
return TRUE;
|
||
|
case LB_CURRENTSELECTION:
|
||
|
return wnd->selection;
|
||
|
case LB_SETSELECTION:
|
||
|
ChangeSelection(wnd, (int) p1, 0);
|
||
|
return TRUE;
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
case CLOSE_WINDOW:
|
||
|
if (isMultiLine(wnd) && wnd->AddMode) {
|
||
|
wnd->AddMode = FALSE;
|
||
|
DfSendMessage(GetParent(wnd), ADDSTATUS, 0, 0);
|
||
|
}
|
||
|
break;
|
||
|
#endif
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
return BaseWndProc(LISTBOX, wnd, msg, p1, p2);
|
||
|
}
|
||
|
|
||
|
static BOOL SelectionInWindow(DFWINDOW wnd, int sel)
|
||
|
{
|
||
|
return (wnd->wlines && sel >= wnd->wtop &&
|
||
|
sel < wnd->wtop+ClientHeight(wnd));
|
||
|
}
|
||
|
|
||
|
static void WriteSelection(DFWINDOW wnd, int sel,
|
||
|
int reverse, DFRECT *rc)
|
||
|
{
|
||
|
if (isVisible(wnd))
|
||
|
if (SelectionInWindow(wnd, sel))
|
||
|
WriteTextLine(wnd, rc, sel, reverse);
|
||
|
}
|
||
|
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
/* ----- Test for extended selections in the listbox ----- */
|
||
|
static void TestExtended(DFWINDOW wnd, PARAM p2)
|
||
|
{
|
||
|
if (isMultiLine(wnd) && !wnd->AddMode &&
|
||
|
!((int) p2 & (LEFTSHIFT | RIGHTSHIFT))) {
|
||
|
if (wnd->SelectCount > 1) {
|
||
|
ClearAllSelections(wnd);
|
||
|
DfSendMessage(wnd, PAINT, 0, 0);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* ----- Clear selections in the listbox ----- */
|
||
|
static void ClearAllSelections(DFWINDOW wnd)
|
||
|
{
|
||
|
if (isMultiLine(wnd) && wnd->SelectCount > 0) {
|
||
|
int sel;
|
||
|
for (sel = 0; sel < wnd->wlines; sel++)
|
||
|
ClearSelection(wnd, sel);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* ----- Invert a selection in the listbox ----- */
|
||
|
static void FlipSelection(DFWINDOW wnd, int sel)
|
||
|
{
|
||
|
if (isMultiLine(wnd)) {
|
||
|
if (ItemSelected(wnd, sel))
|
||
|
ClearSelection(wnd, sel);
|
||
|
else
|
||
|
SetSelection(wnd, sel);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static int ExtendSelections(DFWINDOW wnd, int sel, int shift)
|
||
|
{
|
||
|
if (shift & (LEFTSHIFT | RIGHTSHIFT) &&
|
||
|
wnd->AnchorPoint != -1) {
|
||
|
int i = sel;
|
||
|
int j = wnd->AnchorPoint;
|
||
|
int rtn;
|
||
|
if (j > i)
|
||
|
swap(i,j);
|
||
|
rtn = i - j;
|
||
|
while (j <= i)
|
||
|
SetSelection(wnd, j++);
|
||
|
return rtn;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static void SetSelection(DFWINDOW wnd, int sel)
|
||
|
{
|
||
|
if (isMultiLine(wnd) && !ItemSelected(wnd, sel)) {
|
||
|
char *lp = TextLine(wnd, sel);
|
||
|
*lp = LISTSELECTOR;
|
||
|
wnd->SelectCount++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void ClearSelection(DFWINDOW wnd, int sel)
|
||
|
{
|
||
|
if (isMultiLine(wnd) && ItemSelected(wnd, sel)) {
|
||
|
char *lp = TextLine(wnd, sel);
|
||
|
*lp = ' ';
|
||
|
--wnd->SelectCount;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
BOOL ItemSelected(DFWINDOW wnd, int sel)
|
||
|
{
|
||
|
if (sel != -1 && isMultiLine(wnd) && sel < wnd->wlines) {
|
||
|
char *cp = TextLine(wnd, sel);
|
||
|
return (int)((*cp) & 255) == LISTSELECTOR;
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static void ChangeSelection(DFWINDOW wnd,int sel,int shift)
|
||
|
{
|
||
|
if (sel != wnd->selection) {
|
||
|
#ifdef INCLUDE_EXTENDEDSELECTIONS
|
||
|
if (isMultiLine(wnd)) {
|
||
|
int sels;
|
||
|
if (!wnd->AddMode)
|
||
|
ClearAllSelections(wnd);
|
||
|
sels = ExtendSelections(wnd, sel, shift);
|
||
|
if (sels > 1)
|
||
|
DfSendMessage(wnd, PAINT, 0, 0);
|
||
|
if (sels == 0 && !wnd->AddMode) {
|
||
|
ClearSelection(wnd, wnd->selection);
|
||
|
SetSelection(wnd, sel);
|
||
|
wnd->AnchorPoint = sel;
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
WriteSelection(wnd, wnd->selection, FALSE, NULL);
|
||
|
wnd->selection = sel;
|
||
|
WriteSelection(wnd, sel, TRUE, NULL);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* EOF */
|