summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/proto.h1
-rw-r--r--include/visio.h11
-rw-r--r--mbbsd/cache.c32
-rw-r--r--mbbsd/chat.c7
-rw-r--r--mbbsd/name.c30
-rw-r--r--mbbsd/visio.c128
6 files changed, 108 insertions, 101 deletions
diff --git a/include/proto.h b/include/proto.h
index 9500b9a5..35e2fb4e 100644
--- a/include/proto.h
+++ b/include/proto.h
@@ -156,7 +156,6 @@ void resolve_boards(void);
void resolve_fcache(void);
void sem_init(int semkey,int *semid);
void sem_lock(int op,int semid);
-char *u_namearray(char buf[][IDLEN + 1], int *pnum, char *tag);
char *getuserid(int num);
int searchnewuser(int mode);
int count_logins(int uid, int show);
diff --git a/include/visio.h b/include/visio.h
index f64a1e3d..626f12e5 100644
--- a/include/visio.h
+++ b/include/visio.h
@@ -83,7 +83,14 @@ typedef struct VCOL {
#define VGETCB_END (2) // finish input
#define VGETCB_ABORT (3) // clear buffer and finish
-typedef int (*VGET_FCALLBACK)(int key, char *buf, int *picurr, int *piend, int len, void *param);
+typedef struct {
+ char *buf;
+ const int len; // callbacks should not change this.
+ int icurr; // cursor position
+ int iend; // buffer tail (= strlen(buf))
+} VGET_RUNTIME;
+
+typedef int (*VGET_FCALLBACK)(int key, VGET_RUNTIME *prt, void *instance);
typedef struct {
VGET_FCALLBACK peek; // called immediately after key hit
VGET_FCALLBACK data; // called before inserting character data
@@ -122,7 +129,7 @@ void vshowmsg(const char *msg); /// draw standard pause/message
int vgets (char *buf, int len, int flags); /// input with edit box control
int vgetstr (char *buf, int len, int flags, const char *str);/// input with default value
int vget (int y, int x, const char *prompt, char *buf, int len, int mode);
-int vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBACKS *pcb, void *cbparam);
+int vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBACKS *pcbs, void *instance);
// vs_*: formatted and themed virtual screen layout
// you cannot use ANSI escapes in these APIs.
diff --git a/mbbsd/cache.c b/mbbsd/cache.c
index 90339547..d6fc105d 100644
--- a/mbbsd/cache.c
+++ b/mbbsd/cache.c
@@ -252,38 +252,6 @@ setuserid(int num, const char *userid)
}
}
-#ifndef _BBS_UTIL_C_
-char *
-u_namearray(char buf[][IDLEN + 1], int *pnum, char *tag)
-{
- register char *ptr, tmp;
- register int n, total;
- char tagbuf[STRLEN];
- int ch, ch2, num;
-
- if (*tag == '\0') {
- *pnum = SHM->number;
- return SHM->userid[0];
- }
- for (n = 0; tag[n]; n++)
- tagbuf[n] = chartoupper(tag[n]);
- tagbuf[n] = '\0';
- ch = tagbuf[0];
- ch2 = ch - 'A' + 'a';
- total = SHM->number;
- for (n = num = 0; n < total; n++) {
- ptr = SHM->userid[n];
- tmp = *ptr;
- if (tmp == ch || tmp == ch2) {
- if (chkstr(tag, tagbuf, ptr))
- strcpy(buf[num++], ptr);
- }
- }
- *pnum = num;
- return buf[0];
-}
-#endif
-
void
getnewutmpent(const userinfo_t * up)
{
diff --git a/mbbsd/chat.c b/mbbsd/chat.c
index eda0983e..4ecfde88 100644
--- a/mbbsd/chat.c
+++ b/mbbsd/chat.c
@@ -268,11 +268,10 @@ typedef struct {
int *chatting;
} ChatCbParam;
-// static
-int
-_vgetcb_peek(int key, char *buf, int *picurr, int *piend, int len, void *ptr)
+static int
+_vgetcb_peek(int key, VGET_RUNTIME *prt, void *instance)
{
- ChatCbParam *p = (ChatCbParam*) ptr;
+ ChatCbParam *p = (ChatCbParam*) instance;
assert(p);
switch (key) {
diff --git a/mbbsd/name.c b/mbbsd/name.c
index 47416136..ffb4fc4f 100644
--- a/mbbsd/name.c
+++ b/mbbsd/name.c
@@ -159,6 +159,36 @@ int NameList_search(const struct NameList *self, const char *name)
//-----------------------------------------------------------------------
+char *
+u_namearray(char buf[][IDLEN + 1], int *pnum, char *tag)
+{
+ register char *ptr, tmp;
+ register int n, total;
+ char tagbuf[STRLEN];
+ int ch, ch2, num;
+
+ if (*tag == '\0') {
+ *pnum = SHM->number;
+ return SHM->userid[0];
+ }
+ for (n = 0; tag[n]; n++)
+ tagbuf[n] = chartoupper(tag[n]);
+ tagbuf[n] = '\0';
+ ch = tagbuf[0];
+ ch2 = ch - 'A' + 'a';
+ total = SHM->number;
+ for (n = num = 0; n < total; n++) {
+ ptr = SHM->userid[n];
+ tmp = *ptr;
+ if (tmp == ch || tmp == ch2) {
+ if (chkstr(tag, tagbuf, ptr))
+ strcpy(buf[num++], ptr);
+ }
+ }
+ *pnum = num;
+ return buf[0];
+}
+
static int
UserMaxLen(char cwlist[][IDLEN + 1], int cwnum, int morenum,
int count)
diff --git a/mbbsd/visio.c b/mbbsd/visio.c
index 89a7b71b..20f711a6 100644
--- a/mbbsd/visio.c
+++ b/mbbsd/visio.c
@@ -827,28 +827,32 @@ InputHistoryNext(char *s, int sz)
////////////////////////////////////////////////////////////////////////
static inline int
-_vgetcbhandler(VGET_FCALLBACK cbptr, int *pabort,
- int c, char *buf, int *picurr, int *piend, int len, void *cbparam)
+_vgetcbhandler(VGET_FCALLBACK cbptr, int *pabort, int c, VGET_RUNTIME *prt, void *instance)
{
if (!cbptr)
return 0;
- switch(cbptr(c, buf, picurr, piend, len, cbparam))
+ switch(cbptr(c, prt, instance))
{
case VGETCB_NONE:
+ assert( prt->icurr >= 0 && prt->icurr <= prt->iend && prt->iend <= prt->len );
return 0;
case VGETCB_NEXT:
+ assert( prt->icurr >= 0 && prt->icurr <= prt->iend && prt->iend <= prt->len );
return 1;
case VGETCB_END:
+ assert( prt->icurr >= 0 && prt->icurr <= prt->iend && prt->iend <= prt->len );
*pabort = 1;
return 1;
case VGETCB_ABORT:
+ assert( prt->icurr >= 0 && prt->icurr <= prt->iend && prt->iend <= prt->len );
*pabort = 1;
- *picurr = *piend = 0;
- buf[0] = 0;
+ prt->icurr = 0;
+ prt->iend = 0;
+ prt->buf[0] = 0;
return 1;
}
assert(0); // shall never reach here
@@ -856,20 +860,23 @@ _vgetcbhandler(VGET_FCALLBACK cbptr, int *pabort,
}
int
-vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBACKS *pcb, void *cbparam)
+vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBACKS *pcbs, void *instance)
{
- // iend points to NUL address, and
- // icurr points to cursor.
+ // rt.iend points to NUL address, and
+ // rt.icurr points to cursor.
int line, col;
- int icurr = 0, iend = 0, abort = 0;
+ int abort = 0;
int c;
char ismsgline = 0;
+ // callback
+ VGET_CALLBACKS cb = {NULL};
+
// always use internal buffer to prevent temporary input issue.
char buf[STRLEN] = ""; // zero whole.
- // callback
- VGET_CALLBACKS cb = {NULL};
+ // runtime structure
+ VGET_RUNTIME rt = { buf, len > STRLEN ? STRLEN : len };
// it is wrong to design input with larger buffer
// than STRLEN. Although we support large screen,
@@ -882,16 +889,12 @@ vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBA
{
strlcpy(buf, defstr, len);
strip_ansi(buf, buf, STRIP_ALL); // safer...
- icurr = iend = strlen(buf);
+ rt.icurr = rt.iend = strlen(buf);
}
// setup callbacks
- if (pcb)
- {
- if (pcb->peek) cb.peek = pcb->peek;
- if (pcb->data) cb.data = pcb->data;
- if (pcb->post) cb.post = pcb->post;
- }
+ if (pcbs)
+ cb = *pcbs;
getyx(&line, &col); // now (line,col) is the beginning of our new fields.
@@ -902,6 +905,7 @@ vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBA
if (ismsgline)
msg_occupied ++;
+ // main loop
while (!abort)
{
if (!(flags & VGET_NOECHO))
@@ -910,14 +914,17 @@ vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBA
move(line, col);
clrtoeol();
SOLVE_ANSI_CACHE();
+
if (!(flags & VGET_TRANSPARENT))
outs(VCLR_INPUT_FIELD); // change color to prompt fields
+
vfill(len, 0, buf);
+
if (!(flags & VGET_TRANSPARENT))
outs(ANSI_RESET);
// move to cursor position
- move(line, col+icurr);
+ move(line, col+rt.icurr);
} else {
// to simulate the "clrtoeol" behavior...
// XXX make this call only once? or not?
@@ -926,8 +933,7 @@ vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBA
c = vkey();
// callback 1: peek
- if (_vgetcbhandler(cb.peek, &abort,
- c, buf, &icurr, &iend, len, cbparam))
+ if (_vgetcbhandler(cb.peek, &abort, c, &rt, instance))
continue;
// standard key bindings
@@ -953,7 +959,7 @@ vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBA
else
InputHistoryPrev(buf, len);
- icurr = iend = strlen(buf);
+ rt.icurr = rt.iend = strlen(buf);
break;
// exiting keys
@@ -962,7 +968,7 @@ vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBA
break;
case Ctrl('C'):
- icurr = iend = 0;
+ rt.icurr = rt.iend = 0;
buf[0] = 0;
buf[1] = c;
abort = 1;
@@ -970,65 +976,65 @@ vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBA
// standard navigation
case KEY_HOME: case Ctrl('A'):
- icurr = 0;
+ rt.icurr = 0;
break;
case KEY_END: case Ctrl('E'):
- icurr = iend;
+ rt.icurr = rt.iend;
break;
case KEY_LEFT: case Ctrl('B'):
- if (icurr > 0)
- icurr--;
+ if (rt.icurr > 0)
+ rt.icurr--;
else
bell();
- if (icurr > 0 && CHKDBCSTRAIL(buf, icurr))
- icurr--;
+ if (rt.icurr > 0 && CHKDBCSTRAIL(buf, rt.icurr))
+ rt.icurr--;
break;
case KEY_RIGHT: case Ctrl('F'):
- if (icurr < iend)
- icurr++;
+ if (rt.icurr < rt.iend)
+ rt.icurr++;
else
bell();
- if (icurr < iend && CHKDBCSTRAIL(buf, icurr))
- icurr++;
+ if (rt.icurr < rt.iend && CHKDBCSTRAIL(buf, rt.icurr))
+ rt.icurr++;
break;
// editing keys
case KEY_DEL: case Ctrl('D'):
- if (icurr+1 < iend && CHKDBCSTRAIL(buf, icurr+1)) {
+ if (rt.icurr+1 < rt.iend && CHKDBCSTRAIL(buf, rt.icurr+1)) {
// kill next one character.
- memmove(buf+icurr, buf+icurr+1, iend-icurr);
- iend--;
+ memmove(buf+rt.icurr, buf+rt.icurr+1, rt.iend-rt.icurr);
+ rt.iend--;
}
- if (icurr < iend) {
+ if (rt.icurr < rt.iend) {
// kill next one character.
- memmove(buf+icurr, buf+icurr+1, iend-icurr);
- iend--;
+ memmove(buf+rt.icurr, buf+rt.icurr+1, rt.iend-rt.icurr);
+ rt.iend--;
}
break;
case KEY_BS: case KEY_BS2:
- if (icurr > 0) {
+ if (rt.icurr > 0) {
// kill previous one charracter.
- memmove(buf+icurr-1, buf+icurr, iend-icurr+1);
- icurr--; iend--;
+ memmove(buf+rt.icurr-1, buf+rt.icurr, rt.iend-rt.icurr+1);
+ rt.icurr--; rt.iend--;
} else
bell();
- if (icurr > 0 && CHKDBCSTRAIL(buf, icurr)) {
+ if (rt.icurr > 0 && CHKDBCSTRAIL(buf, rt.icurr)) {
// kill previous one charracter.
- memmove(buf+icurr-1, buf+icurr, iend-icurr+1);
- icurr--; iend--;
+ memmove(buf+rt.icurr-1, buf+rt.icurr, rt.iend-rt.icurr+1);
+ rt.icurr--; rt.iend--;
}
break;
case Ctrl('Y'):
- icurr = 0;
+ rt.icurr = 0;
// reuse Ctrl-K code
case Ctrl('K'):
- iend = icurr;
- buf[iend] = 0;
+ rt.iend = rt.icurr;
+ buf[rt.iend] = 0;
break;
// defaults
@@ -1061,14 +1067,14 @@ vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBA
}
// size check
- if(iend+1 >= len)
+ if(rt.iend+1 >= len)
{
bell(); continue;
}
// prevent incomplete DBCS
if (c > 0x80 && num_in_buf() &&
- len - iend < 3) // we need 3 for DBCS+NUL.
+ len - rt.iend < 3) // we need 3 for DBCS+NUL.
{
drop_input();
bell();
@@ -1076,35 +1082,33 @@ vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBA
}
// callback 2: data
- if (_vgetcbhandler(cb.data, &abort,
- c, buf, &icurr, &iend, len, cbparam))
+ if (_vgetcbhandler(cb.data, &abort, c, &rt, instance))
continue;
// size check again, due to data callback.
- if(iend+1 >= len)
+ if(rt.iend+1 >= len)
{
bell(); continue;
}
// add one character.
- memmove(buf+icurr+1, buf+icurr, iend-icurr+1);
- buf[icurr++] = c;
- iend++;
+ memmove(buf+rt.icurr+1, buf+rt.icurr, rt.iend-rt.icurr+1);
+ buf[rt.icurr++] = c;
+ rt.iend++;
// callback 3: post
- if (_vgetcbhandler(cb.post, &abort,
- c, buf, &icurr, &iend, len, cbparam))
+ if (_vgetcbhandler(cb.post, &abort, c, &rt, instance))
continue;
break;
}
}
- assert(iend >= 0 && iend < len);
- buf[iend] = 0;
+ assert(rt.iend >= 0 && rt.iend < len);
+ buf[rt.iend] = 0;
// final filtering
- if (iend && (flags & VGET_LOWERCASE))
+ if (rt.iend && (flags & VGET_LOWERCASE))
buf[0] = tolower(buf[0]);
// save the history except password mode
@@ -1121,7 +1125,7 @@ vgetstring(char *_buf, int len, int flags, const char *defstr, const VGET_CALLBA
/* because some code then outs so change new line.*/
move(line+1, 0);
- return iend;
+ return rt.iend;
}
int