This repository has been archived on 2024-04-08. You can view files and clone it, but cannot push or open issues or pull requests.

1008 lines
23 KiB
C
Raw Normal View History

#include <cdk_int.h>
#include <scroller.h>
/*
* $Author: tom $
* $Date: 2016/11/20 19:25:35 $
* $Revision: 1.156 $
*/
/*
* Declare file local prototypes.
*/
static int createList (CDKSELECTION *selection, CDK_CSTRING2 list, int listSize);
static void drawCDKSelectionList (CDKSELECTION *selection, boolean Box);
static void setViewSize (CDKSELECTION *scrollp, int listSize);
static int maxViewSize (CDKSELECTION *scrollp);
/* Determine how many characters we can shift to the right */
/* before all the items have been scrolled off the screen. */
#define AvailableWidth(w) ((w)->boxWidth - 2*BorderOf(w) - (w)->maxchoicelen)
#define updateViewWidth(w, widest) \
(w)->maxLeftChar = (((w)->boxWidth > widest) \
? 0 \
: (widest - AvailableWidth(w)))
#define WidestItem(w) ((w)->maxLeftChar + AvailableWidth(w))
#define SCREENPOS(w,n) (w)->itemPos[n] - (w)->leftChar + scrollbarAdj /* + BorderOf(w) */
DeclareCDKObjects (SELECTION, Selection, setCdk, Int);
/*
* This function creates a selection widget.
*/
CDKSELECTION *newCDKSelection (CDKSCREEN *cdkscreen,
int xplace,
int yplace,
int splace,
int height,
int width,
const char *title,
CDK_CSTRING2 list,
int listSize,
CDK_CSTRING2 choices,
int choiceCount,
chtype highlight,
boolean Box,
boolean shadow)
{
/* *INDENT-EQLS* */
CDKSELECTION *selection = 0;
int widestItem = -1;
int parentWidth = getmaxx (cdkscreen->window);
int parentHeight = getmaxy (cdkscreen->window);
int boxWidth;
int boxHeight;
int xpos = xplace;
int ypos = yplace;
int j = 0;
int junk2;
/* *INDENT-OFF* */
static const struct { int from; int to; } bindings[] = {
{ CDK_BACKCHAR, KEY_PPAGE },
{ CDK_FORCHAR, KEY_NPAGE },
{ 'g', KEY_HOME },
{ '1', KEY_HOME },
{ 'G', KEY_END },
{ '<', KEY_HOME },
{ '>', KEY_END },
};
/* *INDENT-ON* */
if (choiceCount <= 0
|| (selection = newCDKObject (CDKSELECTION, &my_funcs)) == 0
|| (selection->choice = typeCallocN (chtype *, choiceCount + 1)) == 0
|| (selection->choicelen = typeCallocN (int, choiceCount + 1)) == 0)
{
destroyCDKObject (selection);
return (0);
}
setCDKSelectionBox (selection, Box);
/*
* If the height is a negative value, the height will
* be ROWS-height, otherwise, the height will be the
* given height.
*/
boxHeight = setWidgetDimension (parentHeight, height, 0);
/*
* If the width is a negative value, the width will
* be COLS-width, otherwise, the width will be the
* given width.
*/
boxWidth = setWidgetDimension (parentWidth, width, 0);
boxWidth = setCdkTitle (ObjOf (selection), title, boxWidth);
/* Set the box height. */
if (TitleLinesOf (selection) > boxHeight)
{
boxHeight = TitleLinesOf (selection)
+ MINIMUM (listSize, 8)
+ 2 * BorderOf (selection);
}
selection->maxchoicelen = 0;
/* Adjust the box width if there is a scroll bar. */
if (splace == LEFT || splace == RIGHT)
{
boxWidth++;
selection->scrollbar = TRUE;
}
else
{
selection->scrollbar = FALSE;
}
/*
* Make sure we didn't extend beyond the dimensions of the window.
*/
selection->boxWidth = (boxWidth > parentWidth ? parentWidth : boxWidth);
selection->boxHeight = (boxHeight > parentHeight ? parentHeight : boxHeight);
setViewSize (selection, listSize);
/* Rejustify the x and y positions if we need to. */
alignxy (cdkscreen->window, &xpos, &ypos, selection->boxWidth, selection->boxHeight);
/* Make the selection window */
selection->win = newwin (selection->boxHeight, selection->boxWidth, ypos, xpos);
/* Is the window null?? */
if (selection->win == 0)
{
destroyCDKObject (selection);
return (0);
}
/* Turn the keypad on for this window. */
keypad (selection->win, TRUE);
/* Create the scrollbar window. */
if (splace == RIGHT)
{
selection->scrollbarWin = subwin (selection->win,
maxViewSize (selection), 1,
SCREEN_YPOS (selection, ypos),
(xpos
+ selection->boxWidth
- BorderOf (selection)
- 1));
}
else if (splace == LEFT)
{
selection->scrollbarWin = subwin (selection->win,
maxViewSize (selection), 1,
SCREEN_YPOS (selection, ypos),
SCREEN_XPOS (selection, xpos));
}
else
{
selection->scrollbarWin = 0;
}
/* *INDENT-EQLS* Set the rest of the variables */
ScreenOf (selection) = cdkscreen;
selection->parent = cdkscreen->window;
selection->scrollbarPlacement = splace;
selection->maxLeftChar = 0;
selection->leftChar = 0;
selection->highlight = highlight;
selection->choiceCount = choiceCount;
initExitType (selection);
ObjOf (selection)->acceptsFocus = TRUE;
ObjOf (selection)->inputWindow = selection->win;
selection->shadow = shadow;
setCDKSelectionCurrent (selection, 0);
/* Each choice has to be converted from char * to chtype * */
for (j = 0; j < choiceCount; j++)
{
selection->choice[j] = char2Chtype (choices[j],
&selection->choicelen[j],
&junk2);
selection->maxchoicelen = MAXIMUM (selection->maxchoicelen,
selection->choicelen[j]);
}
/* Each item in the needs to be converted to chtype * */
widestItem = createList (selection, list, listSize);
if (widestItem > 0)
{
updateViewWidth (selection, widestItem);
}
else if (listSize)
{
destroyCDKObject (selection);
return (0);
}
/* Do we need to create a shadow. */
if (shadow)
{
selection->shadowWin = newwin (boxHeight, boxWidth, ypos + 1, xpos + 1);
}
/* Setup the key bindings. */
for (j = 0; j < (int)SIZEOF (bindings); ++j)
bindCDKObject (vSELECTION,
selection,
(chtype)bindings[j].from,
getcCDKBind,
(void *)(long)bindings[j].to);
/* Register this baby. */
registerCDKObject (cdkscreen, vSELECTION, selection);
/* Return the selection list */
return (selection);
}
/*
* Put the cursor on the currently-selected item.
*/
static void fixCursorPosition (CDKSELECTION *selection)
{
scroller_FixCursorPosition ((CDKSCROLLER *)selection);
}
/*
* This actually manages the selection widget...
*/
int activateCDKSelection (CDKSELECTION *selection, chtype *actions)
{
/* Draw the selection list */
drawCDKSelection (selection, ObjOf (selection)->box);
if (actions == 0)
{
chtype input;
boolean functionKey;
for (;;)
{
int ret;
fixCursorPosition (selection);
input = (chtype)getchCDKObject (ObjOf (selection), &functionKey);
/* Inject the character into the widget. */
ret = injectCDKSelection (selection, input);
if (selection->exitType != vEARLY_EXIT)
{
return ret;
}
}
}
else
{
int length = chlen (actions);
int j = 0;
/* Inject each character one at a time. */
for (j = 0; j < length; j++)
{
int ret = injectCDKSelection (selection, actions[j]);
if (selection->exitType != vEARLY_EXIT)
{
return ret;
}
}
}
/* Set the exit type and return. */
setExitType (selection, 0);
return 0;
}
/*
* This injects a single character into the widget.
*/
static int _injectCDKSelection (CDKOBJS *object, chtype input)
{
CDKSELECTION *selection = (CDKSELECTION *)object;
CDKSCROLLER *widget = (CDKSCROLLER *)object;
int ppReturn = 1;
int ret = unknownInt;
bool complete = FALSE;
/* Set the exit type. */
setExitType (widget, 0);
/* Draw the widget list */
drawCDKSelectionList (selection, ObjOf (widget)->box);
/* Check if there is a pre-process function to be called. */
if (PreProcessFuncOf (widget) != 0)
{
/* Call the pre-process function. */
ppReturn = PreProcessFuncOf (widget) (vSELECTION,
widget,
PreProcessDataOf (widget),
input);
}
/* Should we continue? */
if (ppReturn != 0)
{
/* Check for a predefined binding. */
if (checkCDKObjectBind (vSELECTION, widget, input) != 0)
{
checkEarlyExit (widget);
complete = TRUE;
}
else
{
switch (input)
{
case KEY_UP:
scroller_KEY_UP (widget);
break;
case KEY_DOWN:
scroller_KEY_DOWN (widget);
break;
case KEY_RIGHT:
scroller_KEY_RIGHT (widget);
break;
case KEY_LEFT:
scroller_KEY_LEFT (widget);
break;
case KEY_PPAGE:
scroller_KEY_PPAGE (widget);
break;
case KEY_NPAGE:
scroller_KEY_NPAGE (widget);
break;
case KEY_HOME:
scroller_KEY_HOME (widget);
break;
case KEY_END:
scroller_KEY_END (widget);
break;
case '$':
widget->leftChar = widget->maxLeftChar;
break;
case '|':
widget->leftChar = 0;
break;
case SPACE:
if (selection->mode[widget->currentItem] == 0)
{
if (selection->selections[widget->currentItem]
== (selection->choiceCount - 1))
{
selection->selections[widget->currentItem] = 0;
}
else
{
selection->selections[widget->currentItem]++;
}
}
else
{
Beep ();
}
break;
case KEY_ESC:
setExitType (widget, input);
complete = TRUE;
break;
case KEY_ERROR:
setExitType (widget, input);
complete = TRUE;
break;
case KEY_TAB:
case KEY_ENTER:
setExitType (widget, input);
ret = 1;
complete = TRUE;
break;
case CDK_REFRESH:
eraseCDKScreen (ScreenOf (widget));
refreshCDKScreen (ScreenOf (widget));
break;
default:
break;
}
}
/* Should we call a post-process? */
if (!complete && (PostProcessFuncOf (widget) != 0))
{
PostProcessFuncOf (widget) (vSELECTION,
widget,
PostProcessDataOf (widget),
input);
}
}
if (!complete)
{
drawCDKSelectionList (selection, ObjOf (widget)->box);
setExitType (widget, 0);
}
ResultOf (widget).valueInt = ret;
fixCursorPosition (selection);
return (ret != unknownInt);
}
/*
* This moves the selection field to the given location.
*/
static void _moveCDKSelection (CDKOBJS *object,
int xplace,
int yplace,
boolean relative,
boolean refresh_flag)
{
CDKSELECTION *selection = (CDKSELECTION *)object;
/* *INDENT-EQLS* */
int currentX = getbegx (selection->win);
int currentY = getbegy (selection->win);
int xpos = xplace;
int ypos = yplace;
int xdiff = 0;
int ydiff = 0;
/*
* If this is a relative move, then we will adjust where we want
* to move to.
*/
if (relative)
{
xpos = getbegx (selection->win) + xplace;
ypos = getbegy (selection->win) + yplace;
}
/* Adjust the window if we need to. */
alignxy (WindowOf (selection), &xpos, &ypos, selection->boxWidth, selection->boxHeight);
/* Get the difference. */
xdiff = currentX - xpos;
ydiff = currentY - ypos;
/* Move the window to the new location. */
moveCursesWindow (selection->win, -xdiff, -ydiff);
moveCursesWindow (selection->scrollbarWin, -xdiff, -ydiff);
moveCursesWindow (selection->shadowWin, -xdiff, -ydiff);
/* Touch the windows so they 'move'. */
refreshCDKWindow (WindowOf (selection));
/* Redraw the window, if they asked for it. */
if (refresh_flag)
{
drawCDKSelection (selection, ObjOf (selection)->box);
}
}
/*
* This function draws the selection list.
*/
static void _drawCDKSelection (CDKOBJS *object, boolean Box)
{
CDKSELECTION *selection = (CDKSELECTION *)object;
/* Draw in the shadow if we need to. */
if (selection->shadowWin != 0)
{
drawShadow (selection->shadowWin);
}
drawCdkTitle (selection->win, object);
/* Redraw the list */
drawCDKSelectionList (selection, Box);
}
static int maxViewSize (CDKSELECTION *widget)
{
return scroller_MaxViewSize ((CDKSCROLLER *)widget);
}
/*
* Set variables that depend upon the list-size.
*/
static void setViewSize (CDKSELECTION *widget, int listSize)
{
scroller_SetViewSize ((CDKSCROLLER *)widget, listSize);
}
/*
* This function draws the selection list window.
*/
static void drawCDKSelectionList (CDKSELECTION *selection, boolean Box GCC_UNUSED)
{
/* *INDENT-EQLS* */
int scrollbarAdj = (selection->scrollbarPlacement == LEFT) ? 1 : 0;
int screenPos = 0;
int j;
int selItem = -1;
/* If there is to be a highlight, assign it now */
if (ObjOf (selection)->hasFocus)
selItem = selection->currentItem;
/* draw the list... */
for (j = 0; j < selection->viewSize; j++)
{
int xpos = SCREEN_XPOS (selection, 0);
int ypos = SCREEN_YPOS (selection, j);
int k;
/* Draw the empty line. */
writeBlanks (selection->win, xpos, ypos,
HORIZONTAL, 0, getmaxx (selection->win));
k = j + selection->currentTop;
/* Draw the element in the selection list. */
if (k < selection->listSize)
{
screenPos = SCREENPOS (selection, k);
/* Draw the selection item. */
writeChtypeAttrib (selection->win,
(screenPos >= 0) ? screenPos : 1,
ypos,
selection->item[k],
((k == selItem)
? selection->highlight
: A_NORMAL),
HORIZONTAL,
(screenPos >= 0) ? 0 : (1 - screenPos),
selection->itemLen[k]);
/* Draw the choice value. */
writeChtype (selection->win,
xpos + scrollbarAdj,
ypos,
selection->choice[selection->selections[k]],
HORIZONTAL,
0,
selection->choicelen[selection->selections[k]]);
}
}
/* Determine where the toggle is supposed to be. */
if (selection->scrollbar)
{
selection->togglePos = floorCDK (selection->currentItem * (double)selection->step);
selection->togglePos = MINIMUM (selection->togglePos,
(getmaxy (selection->scrollbarWin) - 1));
(void)mvwvline (selection->scrollbarWin, 0, 0, ACS_CKBOARD, getmaxy (selection->scrollbarWin));
(void)mvwvline (selection->scrollbarWin,
selection->togglePos,
0,
' ' | A_REVERSE, selection->toggleSize);
}
/* Box it if needed */
if (ObjOf (selection)->box)
{
drawObjBox (selection->win, ObjOf (selection));
}
fixCursorPosition (selection);
}
/*
* This sets the background attribute of the widget.
*/
static void _setBKattrSelection (CDKOBJS *object, chtype attrib)
{
if (object != 0)
{
CDKSELECTION *widget = (CDKSELECTION *)object;
wbkgd (widget->win, attrib);
if (widget->scrollbarWin != 0)
{
wbkgd (widget->scrollbarWin, attrib);
}
}
}
static void destroyInfo (CDKSELECTION *widget)
{
CDKfreeChtypes (widget->item);
widget->item = 0;
freeAndNull (widget->itemPos);
freeAndNull (widget->itemLen);
freeAndNull (widget->selections);
freeAndNull (widget->mode);
}
/*
* This function destroys the selection list.
*/
static void _destroyCDKSelection (CDKOBJS *object)
{
if (object != 0)
{
CDKSELECTION *selection = (CDKSELECTION *)object;
cleanCdkTitle (object);
CDKfreeChtypes (selection->choice);
freeChecked (selection->choicelen);
destroyInfo (selection);
/* Clean up the windows. */
deleteCursesWindow (selection->scrollbarWin);
deleteCursesWindow (selection->shadowWin);
deleteCursesWindow (selection->win);
/* Clean the key bindings. */
cleanCDKObjectBindings (vSELECTION, selection);
/* Unregister this object. */
unregisterCDKObject (vSELECTION, selection);
}
}
/*
* This function erases the selection list from the screen.
*/
static void _eraseCDKSelection (CDKOBJS *object)
{
if (validCDKObject (object))
{
CDKSELECTION *selection = (CDKSELECTION *)object;
eraseCursesWindow (selection->win);
eraseCursesWindow (selection->shadowWin);
}
}
/*
* This function sets a couple of the selection list attributes.
*/
void setCDKSelection (CDKSELECTION *selection,
chtype highlight,
int *choices,
boolean Box)
{
setCDKSelectionChoices (selection, choices);
setCDKSelectionHighlight (selection, highlight);
setCDKSelectionBox (selection, Box);
}
/*
* This sets the selection list items.
*/
void setCDKSelectionItems (CDKSELECTION *selection, CDK_CSTRING2 list, int listSize)
{
int widestItem;
int j = 0;
widestItem = createList (selection, list, listSize);
if (widestItem <= 0)
return;
/* Clean up the display. */
for (j = 0; j < selection->viewSize; j++)
{
writeBlanks (selection->win,
SCREEN_XPOS (selection, 0),
SCREEN_YPOS (selection, j),
HORIZONTAL, 0, getmaxx (selection->win));
}
setViewSize (selection, listSize);
setCDKSelectionCurrent (selection, 0);
updateViewWidth (selection, widestItem);
}
int getCDKSelectionItems (CDKSELECTION *selection, char **list)
{
if (list != 0)
{
int j;
for (j = 0; j < selection->listSize; j++)
{
list[j] = chtype2Char (selection->item[j]);
}
}
return selection->listSize;
}
/*
*
*/
void setCDKSelectionTitle (CDKSELECTION *selection, const char *title)
{
/* Make sure the title isn't null. */
if (title == 0)
{
return;
}
(void)setCdkTitle (ObjOf (selection), title, -(selection->boxWidth + 1));
setViewSize (selection, selection->listSize);
}
char *getCDKSelectionTitle (CDKSELECTION *selection GCC_UNUSED)
{
return chtype2Char (*TitleOf (selection));
}
/*
* This sets the highlight bar.
*/
void setCDKSelectionHighlight (CDKSELECTION *selection, chtype highlight)
{
selection->highlight = highlight;
}
chtype getCDKSelectionHighlight (CDKSELECTION *selection)
{
return selection->highlight;
}
/*
* This sets the default choices for the selection list.
*/
void setCDKSelectionChoices (CDKSELECTION *selection, int *choices)
{
int j;
/* Set the choice values in the selection list. */
for (j = 0; j < selection->listSize; j++)
{
if (choices[j] < 0)
{
selection->selections[j] = 0;
}
else if (choices[j] > selection->choiceCount)
{
selection->selections[j] = selection->choiceCount - 1;
}
else
{
selection->selections[j] = choices[j];
}
}
}
int *getCDKSelectionChoices (CDKSELECTION *selection)
{
return selection->selections;
}
/*
* This sets a single item's choice value.
*/
void setCDKSelectionChoice (CDKSELECTION *selection, int Index, int choice)
{
int correctChoice = choice;
int correctIndex = Index;
/* Verify that the choice value is in range. */
if (choice < 0)
{
correctChoice = 0;
}
else if (choice > selection->choiceCount)
{
correctChoice = selection->choiceCount - 1;
}
/* Make sure the index isn't out of range. */
if (correctIndex < 0)
{
correctIndex = 0;
}
else if (correctIndex > selection->listSize)
{
correctIndex = selection->listSize - 1;
}
/* Set the choice value. */
selection->selections[correctIndex] = correctChoice;
}
int getCDKSelectionChoice (CDKSELECTION *selection, int Index)
{
/* Make sure the index isn't out of range. */
if (Index < 0)
{
return selection->selections[0];
}
else if (Index > selection->listSize)
{
return selection->selections[selection->listSize - 1];
}
else
{
return selection->selections[Index];
}
}
/*
* This sets the modes of the items in the selection list. Currently
* there are only two: editable=0 and read-only=1
*/
void setCDKSelectionModes (CDKSELECTION *selection, int *modes)
{
int j;
/* Make sure the widget pointer is not null. */
if (selection == 0)
{
return;
}
/* Set the modes. */
for (j = 0; j < selection->listSize; j++)
{
selection->mode[j] = modes[j];
}
}
int *getCDKSelectionModes (CDKSELECTION *selection)
{
return selection->mode;
}
/*
* This sets a single mode of an item in the selection list.
*/
void setCDKSelectionMode (CDKSELECTION *selection, int Index, int mode)
{
/* Make sure the widget pointer is not null. */
if (selection == 0)
{
return;
}
/* Make sure the index isn't out of range. */
if (Index < 0)
{
selection->mode[0] = mode;
}
else if (Index > selection->listSize)
{
selection->mode[selection->listSize - 1] = mode;
}
else
{
selection->mode[Index] = mode;
}
}
int getCDKSelectionMode (CDKSELECTION *selection, int Index)
{
/* Make sure the index isn't out of range. */
if (Index < 0)
{
return selection->mode[0];
}
else if (Index > selection->listSize)
{
return selection->mode[selection->listSize - 1];
}
else
{
return selection->mode[Index];
}
}
/*
* This sets the box attribute of the widget.
*/
void setCDKSelectionBox (CDKSELECTION *selection, boolean Box)
{
ObjOf (selection)->box = Box;
ObjOf (selection)->borderSize = Box ? 1 : 0;
}
boolean getCDKSelectionBox (CDKSELECTION *selection)
{
return ObjOf (selection)->box;
}
/*
* set/get the current item index
*/
void setCDKSelectionCurrent (CDKSELECTION *selection, int item)
{
scroller_SetPosition ((CDKSCROLLER *)selection, item);
}
int getCDKSelectionCurrent (CDKSELECTION *selection)
{
return selection->currentItem;
}
/*
* methods for generic type methods
*/
static void _focusCDKSelection (CDKOBJS *object)
{
CDKSELECTION *selection = (CDKSELECTION *)object;
drawCDKSelectionList (selection, ObjOf (selection)->box);
}
static void _unfocusCDKSelection (CDKOBJS *object)
{
CDKSELECTION *selection = (CDKSELECTION *)object;
drawCDKSelectionList (selection, ObjOf (selection)->box);
}
static int createList (CDKSELECTION *selection, CDK_CSTRING2 list, int listSize)
{
int status = 0;
int widestItem = 0;
if (listSize >= 0)
{
/* *INDENT-EQLS* */
chtype **newList = typeCallocN (chtype *, listSize + 1);
int *newLen = typeCallocN (int, listSize + 1);
int *newPos = typeCallocN (int, listSize + 1);
int *newSel = typeCallocN (int, listSize + 1);
int *newMode = typeCallocN (int, listSize + 1);
if (newList != 0
&& newLen != 0
&& newPos != 0
&& newSel != 0
&& newMode != 0)
{
int boxWidth = AvailableWidth (selection);
int adjust = selection->maxchoicelen + BorderOf (selection);
int j;
status = 1;
for (j = 0; j < listSize; j++)
{
newList[j] = char2Chtype (list[j], &newLen[j], &newPos[j]);
if (newList[j] == 0)
{
status = 0;
break;
}
newPos[j] = justifyString (boxWidth, newLen[j], newPos[j]) + adjust;
widestItem = MAXIMUM (widestItem, newLen[j]);
}
if (status)
{
destroyInfo (selection);
/* *INDENT-EQLS* */
selection->item = newList;
selection->itemPos = newPos;
selection->itemLen = newLen;
selection->selections = newSel;
selection->mode = newMode;
}
else
{
CDKfreeChtypes (newList);
freeChecked (newPos);
freeChecked (newLen);
freeChecked (newSel);
freeChecked (newMode);
}
}
else
{
CDKfreeChtypes (newList);
freeChecked (newPos);
freeChecked (newLen);
freeChecked (newSel);
freeChecked (newMode);
}
}
else
{
destroyInfo (selection);
}
return status ? widestItem : 0;
}
dummyRefreshData (Selection)
dummySaveData (Selection)