/* $Id: cdkmatrix.c,v 1.10 2005/03/08 19:52:01 tom Exp $ */
#include <cdk.h>
#ifdef XCURSES
char *XCursesProgramName = "cdkmatrix";
#endif
#define MY_INFO(x,y) info[(x + 1) * cols + (y + 1)]
/*
* Declare file local prototypes.
*/
static int widgetCB (EObjectType cdktype, void *object, void *clientData, chtype key);
/*
* Define file local variables.
*/
static char *FPUsage = "-r Row Titles -c Column Titles -v Visible Rows -w Column Widths [-t Column Types] [-d Default Values] [-F Field Character] [-T Title] [-B Buttons] [-O Output File] [-X X Position] [-Y Y Position] [-N] [-S]";
/*
*
*/
int main (int argc, char **argv)
{
/* Declare variables. */
CDKSCREEN *cdkScreen = 0;
CDKMATRIX *widget = 0;
CDKBUTTONBOX *buttonWidget = 0;
WINDOW *cursesWindow = 0;
chtype *holder = 0;
char *buttons = 0;
char *CDK_WIDGET_COLOR = 0;
char *temp = 0;
chtype filler = A_NORMAL | '.';
int rows = -1;
int cols = -1;
int buttonCount = 0;
int selection = 0;
int shadowHeight = 0;
FILE *fp = stderr;
char **rowTitles;
char **colTitles;
char **rowTemp = 0;
char **colTemp = 0;
char **buttonList = 0;
int *colWidths;
int *colTypes;
int count, infoLines, x, y, j1, j2;
CDK_PARAMS params;
boolean boxWidget;
boolean shadowWidget;
char *defaultValue;
char *myColTitles;
char *myColTypes;
char *myColWidths;
char *myFiller;
char *myRowTitles;
char *outputFile;
char *title;
int vrows;
int xpos;
int ypos;
CDKparseParams(argc, argv, ¶ms, "c:d:r:t:w:v:B:F:O:T:" "X:Y:NS");
xpos = CDKparamValue(¶ms, 'X', CENTER);
ypos = CDKparamValue(¶ms, 'Y', CENTER);
boxWidget = CDKparamValue(¶ms, 'N', TRUE);
shadowWidget = CDKparamValue(¶ms, 'S', FALSE);
vrows = CDKparamValue(¶ms, 'v', -1);
myColTitles = CDKparamString(¶ms, 'c');
defaultValue = CDKparamString(¶ms, 'd');
myRowTitles = CDKparamString(¶ms, 'r');
myColTypes = CDKparamString(¶ms, 't');
myColWidths = CDKparamString(¶ms, 'w');
buttons = CDKparamString(¶ms, 'B');
myFiller = CDKparamString(¶ms, 'F');
outputFile = CDKparamString(¶ms, 'O');
title = CDKparamString(¶ms, 'T');
/* If the user asked for an output file, try to open it. */
if (outputFile != 0)
{
if ((fp = fopen (outputFile, "w")) == 0)
{
fprintf (stderr, "%s: Can not open output file %s\n", argv[0], outputFile);
exit (-1);
}
}
/* Make sure all the needed command line parameters were provided. */
if ((myRowTitles == 0) ||
(myColTitles == 0) ||
(myColWidths == 0) ||
(vrows == -1))
{
fprintf (stderr, "Usage: %s %s\n", argv[0], FPUsage);
exit (-1);
}
/* Convert the char * titles to a char ** */
rowTemp = CDKsplitString (myRowTitles, '\n');
rows = CDKcountStrings (rowTemp);
rowTitles = (char **) calloc (rows + 1, sizeof(char *));
for (x=0; x < rows; x++)
{
rowTitles[x+1] = rowTemp[x];
}
colTemp = CDKsplitString (myColTitles, '\n');
cols = CDKcountStrings (colTemp);
colTitles = (char **) calloc (cols + 1, sizeof(char *));
for (x=0; x < cols; x++)
{
colTitles[x+1] = colTemp[x];
}
free (colTemp);
/* Convert the column widths. */
colTemp = CDKsplitString (myColWidths, '\n');
count = CDKcountStrings (colTemp);
colWidths = (int *) calloc (count + 1, sizeof(int));
for (x=0; x < count; x++)
{
colWidths[x+1] = atoi (colTemp[x]);
}
CDKfreeStrings(colTemp);
/* If they passed in the column types, convert them. */
if (myColTypes != 0)
{
colTemp = CDKsplitString (myColTypes, '\n');
count = CDKcountStrings (colTemp);
colTypes = (int *) calloc (MAXIMUM(cols, count) + 1, sizeof(int));
for (x=0; x < count; x++)
{
colTypes[x+1] = char2DisplayType (colTemp[x]);
}
CDKfreeStrings(colTemp);
}
else
{
/* If they didn't set default values. */
colTypes = (int *) calloc (cols + 1, sizeof(int));
for (x=0; x < cols; x++)
{
colTypes[x+1] = vMIXED;
}
}
/* Start curses. */
cursesWindow = initscr();
/* Create the CDK screen. */
cdkScreen = initCDKScreen (cursesWindow);
/* Start color. */
initCDKColor();
/* Check if the user wants to set the background of the main screen. */
if ((temp = getenv ("CDK_SCREEN_COLOR")) != 0)
{
holder = char2Chtype (temp, &j1, &j2);
wbkgd (cdkScreen->window, holder[0]);
wrefresh (cdkScreen->window);
freeChtype (holder);
}
/* Get the widget color background color. */
if ((CDK_WIDGET_COLOR = getenv ("CDK_WIDGET_COLOR")) == 0)
{
CDK_WIDGET_COLOR = 0;
}
/* If the set the filler character, set it now. */
if (myFiller != 0)
{
holder = char2Chtype (myFiller, &j1, &j2);
filler = holder[0];
freeChtype (holder);
}
/* Create the matrix widget. */
widget = newCDKMatrix (cdkScreen, xpos, ypos,
rows, cols, vrows, cols,
title, rowTitles, colTitles,
colWidths, colTypes, 1, 1,
filler, COL,
boxWidget, TRUE, shadowWidget);
/* Make sure we could create the widget. */
if (widget == 0)
{
/* Shut down curses and CDK. */
destroyCDKScreen (cdkScreen);
endCDK();
/* Spit out the message. */
fprintf (stderr, "Error: Could not create the matrix. Is the window too small?\n");
/* Exit with an error. */
exit (-1);
}
/*
* If the user sent in a file of default values, read it and
* stick the values read in from the file into the matrix.
*/
if (defaultValue != 0)
{
int limit = (rows + 1) * (cols + 1);
char **info = (char **) calloc (limit, sizeof(char *));
/* Read the file. */
infoLines = CDKreadFile (defaultValue, &rowTemp);
if (infoLines > 0)
{
int *subSize = (int *) calloc(infoLines + 1, sizeof(int *));
/* For each line, split on a CTRL-V. */
for (x=0; x < infoLines; x++)
{
colTemp = CDKsplitString (rowTemp[x], CTRL('V'));
subSize[x+1] = CDKcountStrings (colTemp);
for (y=0; y < subSize[x+1]; y++)
{
MY_INFO(x, y) = colTemp[y];
}
free (colTemp);
}
CDKfreeStrings (rowTemp);
setCDKMatrixCells (widget, info, rows, cols, subSize);
/* Clean up. */
for (x=0; x < infoLines; x++)
{
for (y=0; y < subSize[x+1]; y++)
{
freeChar (MY_INFO(x, y));
}
}
free (info);
free (subSize);
}
}
/* Split the buttons if they supplied some. */
if (buttons != 0)
{
/* Split the button list up. */
buttonList = CDKsplitString (buttons, '\n');
buttonCount = CDKcountStrings (buttonList);
/* We need to create a buttonbox widget. */
buttonWidget = newCDKButtonbox (cdkScreen,
getbegx (widget->win),
getbegy (widget->win) + widget->boxHeight - 1,
1, widget->boxWidth - 1,
NULL, 1, buttonCount,
buttonList, buttonCount,
A_REVERSE, boxWidget, FALSE);
setCDKButtonboxULChar (buttonWidget, ACS_LTEE);
setCDKButtonboxURChar (buttonWidget, ACS_RTEE);
/*
* We need to set the lower left and right
* characters of the widget.
*/
setCDKMatrixLLChar (widget, ACS_LTEE);
setCDKMatrixLRChar (widget, ACS_RTEE);
/*
* Bind the Tab key in the widget to send a
* Tab key to the button box widget.
*/
bindCDKObject (vMATRIX, widget, KEY_TAB, widgetCB, buttonWidget);
bindCDKObject (vMATRIX, widget, CDK_NEXT, widgetCB, buttonWidget);
bindCDKObject (vMATRIX, widget, CDK_PREV, widgetCB, buttonWidget);
/* Check if the user wants to set the background of the widget. */
setCDKButtonboxBackgroundColor (buttonWidget, CDK_WIDGET_COLOR);
/* Draw the button widget. */
drawCDKButtonbox (buttonWidget, boxWidget);
}
/*
* If the user asked for a shadow, we need to create one.
* I do this instead of using the shadow parameter because
* the button widget isn't part of the main widget and if
* the user asks for both buttons and a shadow, we need to
* create a shadow big enough for both widgets. We'll create
* the shadow window using the widgets shadowWin element, so
* screen refreshes will draw them as well.
*/
if (shadowWidget == TRUE)
{
/* Determine the height of the shadow window. */
shadowHeight = (buttonWidget == (CDKBUTTONBOX *)NULL ?
widget->boxHeight :
widget->boxHeight + buttonWidget->boxHeight - 1);
/* Create the shadow window. */
widget->shadowWin = newwin (shadowHeight,
widget->boxWidth,
getbegy (widget->win) + 1,
getbegx (widget->win) + 1);
/* Make sure we could have created the shadow window. */
if (widget->shadowWin != 0)
{
widget->shadow = TRUE;
/*
* We force the widget and buttonWidget to be drawn so the
* buttonbox widget will be drawn when the widget is activated.
* Otherwise the shadow window will draw over the button widget.
*/
drawCDKMatrix (widget, ObjOf(widget)->box);
eraseCDKButtonbox (buttonWidget);
drawCDKButtonbox (buttonWidget, ObjOf(buttonWidget)->box);
}
}
/* Check if the user wants to set the background of the widget. */
setCDKMatrixBackgroundColor (widget, CDK_WIDGET_COLOR);
/* Let them play. */
activateCDKMatrix (widget, 0);
/* Print out the matrix cells. */
if (widget->exitType == vNORMAL)
{
for (x=0; x < widget->rows; x++)
{
for (y=0; y < widget->cols; y++)
{
char *data = getCDKMatrixCell(widget, x, y);
if (data != 0)
{
fprintf (fp, "%s%c", data, CTRL('V'));
}
else
{
fprintf (fp, "%c", CTRL('V'));
}
}
fprintf (fp, "\n");
}
}
/* If there were buttons, get the button selected. */
if (buttonWidget != 0)
{
selection = buttonWidget->currentButton;
destroyCDKButtonbox (buttonWidget);
}
/* cleanup (not really needed) */
CDKfreeStrings (buttonList);
free (colTypes);
free (colWidths);
for (x=0; x < rows; x++)
{
free (rowTitles[x+1]);
}
free (rowTitles);
for (x=0; x < cols; x++)
{
free (colTitles[x+1]);
}
free (colTitles);
destroyCDKMatrix (widget);
destroyCDKScreen (cdkScreen);
/* this is needed */
endCDK();
exit (selection);
}
static int widgetCB (EObjectType cdktype GCC_UNUSED, void *object GCC_UNUSED, void *clientData, chtype key)
{
CDKBUTTONBOX *buttonbox = (CDKBUTTONBOX *)clientData;
injectCDKButtonbox (buttonbox, key);
return (TRUE);
}
syntax highlighted by Code2HTML, v. 0.9.1