/* $Id: alphalist_ex.c,v 1.30 2016/12/04 15:38:55 tom Exp $ */ #include <cdk_test.h> #ifdef HAVE_XCURSES char *XCursesProgramName = "alphalist_ex"; #endif /* * This program demonstrates the Cdk alphalist widget. * * Options (in addition to normal CLI parameters): * -c create the data after the widget */ static CDKSCREEN *cdkscreen = 0; static char **myUserList = 0; static int userSize; typedef struct { int deleted; /* index in current list which is deleted */ int original; /* index in myUserList[] of deleted item */ int position; /* position before delete */ int topline; /* top-line before delete */ } UNDO; static UNDO *myUndoList; static int undoSize; /* * This reads the passwd file and retrieves user information. */ static int getUserList (char ***list) { #if defined (HAVE_PWD_H) struct passwd *ent; #endif int x = 0; unsigned used = 0; #if defined (HAVE_PWD_H) while ((ent = getpwent ()) != 0) { used = CDKallocStrings (list, ent->pw_name, (unsigned)x++, used); } endpwent (); #endif return x; } #define CB_PARAMS EObjectType cdktype GCC_UNUSED, void* object GCC_UNUSED, void* clientdata GCC_UNUSED, chtype key GCC_UNUSED static void fill_undo (CDKALPHALIST *widget, int deleted, char *data) { int top = getCDKScrollCurrentTop (widget->scrollField); int item = getCDKAlphalistCurrentItem (widget); int n; myUndoList[undoSize].deleted = deleted; myUndoList[undoSize].topline = top; myUndoList[undoSize].original = -1; myUndoList[undoSize].position = item; for (n = 0; n < userSize; ++n) { if (!strcmp (myUserList[n], data)) { myUndoList[undoSize].original = n; break; } } ++undoSize; } static int do_delete (CB_PARAMS) { CDKALPHALIST *widget = (CDKALPHALIST *)clientdata; int size; char **list = getCDKAlphalistContents (widget, &size); int result = FALSE; if (size) { int save = getCDKScrollCurrentTop (widget->scrollField); int first = getCDKAlphalistCurrentItem (widget); int n; fill_undo (widget, first, list[first]); for (n = first; n < size; ++n) list[n] = list[n + 1]; setCDKAlphalistContents (widget, (CDK_CSTRING *)list, size - 1); setCDKScrollCurrentTop (widget->scrollField, save); setCDKAlphalistCurrentItem (widget, first); drawCDKAlphalist (widget, BorderOf (widget)); result = TRUE; } return result; } static int do_delete1 (CB_PARAMS) { CDKALPHALIST *widget = (CDKALPHALIST *)clientdata; int size; char **list = getCDKAlphalistContents (widget, &size); int result = FALSE; if (size) { int save = getCDKScrollCurrentTop (widget->scrollField); int first = getCDKAlphalistCurrentItem (widget); if (first-- > 0) { int n; fill_undo (widget, first, list[first]); for (n = first; n < size; ++n) list[n] = list[n + 1]; setCDKAlphalistContents (widget, (CDK_CSTRING *)list, size - 1); setCDKScrollCurrentTop (widget->scrollField, save); setCDKAlphalistCurrentItem (widget, first); drawCDKAlphalist (widget, BorderOf (widget)); result = TRUE; } } return result; } static int do_help (CB_PARAMS) { static const char *message[] = { "Alpha List tests:", "", "F1 = help (this message)", "F2 = delete current item", "F3 = delete previous item", "F4 = reload all items", "F5 = undo deletion", 0 }; popupLabel (cdkscreen, (CDK_CSTRING2)message, (int)CDKcountStrings ((CDK_CSTRING2)message)); return TRUE; } static int do_reload (CB_PARAMS) { int result = FALSE; if (userSize) { CDKALPHALIST *widget = (CDKALPHALIST *)clientdata; setCDKAlphalistContents (widget, (CDK_CSTRING *)myUserList, userSize); setCDKAlphalistCurrentItem (widget, 0); drawCDKAlphalist (widget, BorderOf (widget)); result = TRUE; } return result; } static int do_undo (CB_PARAMS) { int result = FALSE; if (undoSize > 0) { CDKALPHALIST *widget = (CDKALPHALIST *)clientdata; int size; int n; char **oldlist = getCDKAlphalistContents (widget, &size); char **newlist = (char **)malloc ((size_t) (++size + 1) * sizeof (char *)); --undoSize; newlist[size] = 0; for (n = size - 1; n > myUndoList[undoSize].deleted; --n) { newlist[n] = copyChar (oldlist[n - 1]); } newlist[n--] = copyChar (myUserList[myUndoList[undoSize].original]); while (n >= 0) { newlist[n] = copyChar (oldlist[n]); --n; } setCDKAlphalistContents (widget, (CDK_CSTRING *)newlist, size); setCDKScrollCurrentTop (widget->scrollField, myUndoList[undoSize].topline); setCDKAlphalistCurrentItem (widget, myUndoList[undoSize].position); drawCDKAlphalist (widget, BorderOf (widget)); free (newlist); result = TRUE; } return result; } int main (int argc, char **argv) { /* *INDENT-EQLS* */ CDKALPHALIST *alphaList = 0; const char *title = "<C></B/24>Alpha List\n<C>Title"; const char *label = "</B>Account: "; char *word = 0; char **userList = 0; const char *mesg[5]; char temp[256]; CDK_PARAMS params; CDKparseParams (argc, argv, ¶ms, "c" CDK_CLI_PARAMS); /* Get the user list. */ userSize = getUserList (&userList); if (userSize <= 0) { fprintf (stderr, "Cannot get user list\n"); ExitProgram (EXIT_FAILURE); } myUserList = copyCharList ((const char **)userList); myUndoList = (UNDO *) malloc ((size_t) userSize * sizeof (UNDO)); undoSize = 0; cdkscreen = initCDKScreen (NULL); /* Start color. */ initCDKColor (); /* Create the alpha list widget. */ alphaList = newCDKAlphalist (cdkscreen, CDKparamValue (¶ms, 'X', CENTER), CDKparamValue (¶ms, 'Y', CENTER), CDKparamValue (¶ms, 'H', 0), CDKparamValue (¶ms, 'W', 0), title, label, (CDKparamNumber (¶ms, 'c') ? 0 : (CDK_CSTRING *)userList), (CDKparamNumber (¶ms, 'c') ? 0 : userSize), '_', A_REVERSE, CDKparamValue (¶ms, 'N', TRUE), CDKparamValue (¶ms, 'S', FALSE)); if (alphaList == 0) { destroyCDKScreen (cdkscreen); endCDK (); fprintf (stderr, "Cannot create widget\n"); ExitProgram (EXIT_FAILURE); } bindCDKObject (vALPHALIST, alphaList, '?', do_help, NULL); bindCDKObject (vALPHALIST, alphaList, KEY_F1, do_help, NULL); bindCDKObject (vALPHALIST, alphaList, KEY_F2, do_delete, alphaList); bindCDKObject (vALPHALIST, alphaList, KEY_F3, do_delete1, alphaList); bindCDKObject (vALPHALIST, alphaList, KEY_F4, do_reload, alphaList); bindCDKObject (vALPHALIST, alphaList, KEY_F5, do_undo, alphaList); if (CDKparamNumber (¶ms, 'c')) { setCDKAlphalistContents (alphaList, (CDK_CSTRING *)userList, userSize); } /* Let them play with the alpha list. */ word = activateCDKAlphalist (alphaList, 0); /* Determine what the user did. */ if (alphaList->exitType == vESCAPE_HIT) { mesg[0] = "<C>You hit escape. No word was selected."; mesg[1] = ""; mesg[2] = "<C>Press any key to continue."; popupLabel (cdkscreen, (CDK_CSTRING2)mesg, 3); } else if (alphaList->exitType == vNORMAL) { mesg[0] = "<C>You selected the following"; sprintf (temp, "<C>(%.*s)", (int)(sizeof (temp) - 10), word); mesg[1] = temp; mesg[2] = ""; mesg[3] = "<C>Press any key to continue."; popupLabel (cdkscreen, (CDK_CSTRING2)mesg, 4); } freeCharList (myUserList, (unsigned)userSize); free (myUserList); destroyCDKAlphalist (alphaList); destroyCDKScreen (cdkscreen); endCDK (); ExitProgram (EXIT_SUCCESS); }