xwords/xwords4/linux/cursesletterask.c
Eric House d293517e7c toward compiling with gcc8
My VSIZE is no longer legal, and apparently there's no workaround (no
way to safely figure the length of an array whose size is known at
compile time.) To avoid the risk of duplicating little constants, added
macros that define length in a way a new VSIZE can pick it up. Couldn't
make that work for struct field arrays, however, so there I'm using
constants.
2019-01-28 17:24:53 -08:00

218 lines
6.2 KiB
C

/* -*-mode: C; fill-column: 78; c-basic-offset: 4; compile-command: "make MEMDEBUG=TRUE"; -*- */
/*
* Copyright 2003 by Eric House (xwords@eehouse.org). All rights reserved.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#ifdef PLATFORM_NCURSES
#include "linuxutl.h"
#include "cursesletterask.h"
#include "cursesdlgutil.h"
#define MAX_TILE_BUTTON_ROWS 10
#define MAX_TILE_BUTTON_WIDTH 6
static void
sizeTextsAsButtons( XP_U16 maxLen, XP_U16 nTiles, XP_U16* textsCols,
XP_U16* textsRows, XP_U16* textsOffsets )
{
XP_U16 nCols = maxLen / MAX_TILE_BUTTON_WIDTH;
XP_U16 nRows = (nTiles + nCols - 1) / nCols;
XP_U16 i;
*textsCols = nCols;
*textsRows = nRows;
for ( i = 0; i < nRows; ++i ) {
textsOffsets[i] = i * nCols;
}
XP_DEBUGF( "broke %d letters into %d rows of %d cols",
nTiles, nRows, nCols );
} /* sizeTextsAsButtons */
XP_S16
curses_askLetter( CursesAppGlobals* globals, XP_UCHAR* query,
const XP_UCHAR** texts, XP_U16 nTiles )
{
XP_S16 result;
WINDOW* confWin;
int x, y, rows, row, nLines, i;
short newSelButton = 0;
short curSelButton = 1; /* force draw by being different */
short maxWidth;
short numCtlButtons;
VDECL( const char*, ctlButtons, 2 ) = { "Ok", "Cancel" };
XP_Bool dismissed = XP_FALSE;
FormatInfo fi;
int len;
XP_U16 textsCols, textsRows;
XP_U16 textsOffsets[MAX_TILE_BUTTON_ROWS];
XP_U16 spacePerCtlButton;
char* textPtrs[MAX_UNIQUE_TILES];
for ( i = 0; i < nTiles; ++i ) {
textPtrs[i] = (char*)&texts[i];
}
getmaxyx( globals->boardWin, y, x );
numCtlButtons = VSIZE(ctlButtons);
maxWidth = x - (PAD*2) - 2; /* 2 for two borders */
measureAskText( query, maxWidth, &fi );
sizeTextsAsButtons( x, nTiles, &textsCols, &textsRows, textsOffsets );
len = XP_MAX( fi.maxLen, textsCols * MAX_TILE_BUTTON_WIDTH );
if ( len < MIN_WIDTH ) {
len = MIN_WIDTH;
}
rows = fi.nLines + textsRows + 1;
XP_DEBUGF( "set maxWidth=%d", maxWidth );
if ( len > x-2 ) {
rows = (len / maxWidth) + 1;
len = maxWidth;
}
nLines = ASK_HEIGHT + rows - 1;
XP_DEBUGF( "newwin( %d, %d, (%d/2) - (%d/2), (%d-%d-2)/2",
nLines, len,//+(PAD*2),
y, nLines, x, len );
XP_ASSERT( y >= nLines );
confWin = newwin( nLines, len,//+(PAD*2),
(y/2) - (nLines/2), (x-len-2)/2 );
keypad( confWin, TRUE );
XP_ASSERT( !!confWin );
wclear( confWin );
box( confWin, '|', '-');
for ( row = 0; row < fi.nLines; ++row ) {
mvwaddnstr( confWin, row+1, PAD,
fi.line[row].substr, fi.line[row].len );
}
spacePerCtlButton = (len+(PAD*2)) / (numCtlButtons + 1);
result = newSelButton;
while ( !dismissed ) {
int ch;
if ( newSelButton != curSelButton ) {
XP_U16 i;
for ( i = 0; i < textsRows; ++i ) {
XP_U16 nInRow = textsCols;
if ( i + 1 == textsRows ) {
nInRow = nTiles % textsCols;
if ( nInRow == 0 ) {
nInRow = textsCols;
}
}
XP_DEBUGF( "printing %d cols, row %d, first char %s",
nInRow, i, textPtrs[textsOffsets[i]] );
drawButtons( confWin, rows + 2 - textsRows + i,
MAX_TILE_BUTTON_WIDTH-1,
nInRow,
newSelButton - textsOffsets[i],
(const char**)&textPtrs[textsOffsets[i]] );
}
drawButtons( confWin, rows+2, spacePerCtlButton,
numCtlButtons,
newSelButton - nTiles, ctlButtons );
curSelButton = newSelButton;
}
ch = wgetch( confWin );
int incr = 0;
switch ( ch ) {
case '\t':
case 'R':
case KEY_RIGHT:
case 525:
incr = 1;
break;
case 'L':
case KEY_LEFT:
case 524:
incr = -1;
break;
case KEY_DOWN:
case 526:
incr = textsCols;
break;
case KEY_UP:
case 523:
incr = -textsCols;
break;
case EOF:
case 4: /* C-d */
case 27: /* ESC */
curSelButton = 0; /* should be the cancel case */
case KEY_B2: /* "center of keypad" */
case '\r':
case '\n':
dismissed = XP_TRUE;
break;
default:
if ( ch >= 'a' && ch <= 'z' ) {
ch += 'A' - 'a';
}
for ( i = 0; i < nTiles; ++i ) {
if ( ch == texts[i][0] ) {
XP_DEBUGF( "picking %c", (char)ch);
newSelButton = i;
result = i;
break;
}
}
}
if ( incr != 0 ) {
newSelButton = curSelButton + incr;
if ( newSelButton < 0 ) {
newSelButton = 0;
} else if ( newSelButton >= numCtlButtons + nTiles ) {
newSelButton = numCtlButtons + nTiles - 1;
}
}
}
delwin( confWin );
/* this leaves a ghost line, but I can't figure out a better way. */
wtouchln( globals->boardWin, (y/2)-(nLines/2), ASK_HEIGHT + rows - 1, 1 );
wrefresh( globals->boardWin );
return result;
} /* curses_askLetter */
#endif /* PLATFORM_NCURSES */