struct termios raw;
current->fd = STDIN_FILENO;
+ current->cols = 0;
if (!isatty(current->fd) || isUnsupportedTerm() ||
tcgetattr(current->fd, &orig_termios) == -1) {
goto fatal;
}
rawmode = 1;
-
- current->cols = 0;
return 0;
}
#endif
}
-static int countColorControlChars(const char* prompt, int plen)
+static int countColorControlChars(const char* prompt)
{
/* ANSI color control sequences have the form:
- * "\x1b" "[" [0-9;]+ "m"
+ * "\x1b" "[" [0-9;]* "m"
* We parse them with a simple state machine.
*/
enum {
search_esc,
expect_bracket,
- expect_inner,
expect_trail
} state = search_esc;
int len = 0, found = 0;
* bytes in case of the extremely unlikely scenario where
* an ANSI sequence is part of a utf8 sequence.
*/
- for (; plen ; plen--, prompt++) {
- ch = *prompt;
-
+ while ((ch = *prompt++) != 0) {
switch (state) {
case search_esc:
- len = 0;
if (ch == '\x1b') {
state = expect_bracket;
- len++;
}
break;
case expect_bracket:
if (ch == '[') {
- state = expect_inner;
- len++;
- } else {
- state = search_esc;
- }
- break;
- case expect_inner:
- if (ch >= '0' && ch <= '9') {
- len++;
state = expect_trail;
- } else {
- state = search_esc;
+ /* 3 for "\e[ ... m" */
+ len = 3;
+ break;
}
+ state = search_esc;
break;
case expect_trail:
- if (ch == 'm') {
+ if ((ch == ';') || ((ch >= '0' && ch <= '9'))) {
+ /* 0-9, or semicolon */
len++;
+ break;
+ }
+ if (ch == 'm') {
found += len;
- state = search_esc;
- } else if ((ch != ';') && ((ch < '0') || (ch > '9'))) {
- state = search_esc;
}
- /* 0-9, or semicolon */
- len++;
+ state = search_esc;
break;
}
}
return found;
}
+/**
+ * Stores the current cursor column in '*cols'.
+ * Returns 1 if OK, or 0 if failed to determine cursor pos.
+ */
+static int queryCursor(int fd, int* cols)
+{
+ /* control sequence - report cursor location */
+ fd_printf(fd, "\x1b[6n");
+
+ /* Parse the response: ESC [ rows ; cols R */
+ if (fd_read_char(fd, 100) == 0x1b &&
+ fd_read_char(fd, 100) == '[') {
+
+ int n = 0;
+ while (1) {
+ int ch = fd_read_char(fd, 100);
+ if (ch == ';') {
+ /* Ignore rows */
+ n = 0;
+ }
+ else if (ch == 'R') {
+ /* Got cols */
+ if (n != 0 && n < 1000) {
+ *cols = n;
+ }
+ break;
+ }
+ else if (ch >= 0 && ch <= '9') {
+ n = n * 10 + ch - '0';
+ }
+ else {
+ break;
+ }
+ }
+ return 1;
+ }
+
+ return 0;
+}
+
+/**
+ * Updates current->cols with the current window size (width)
+ */
static int getWindowSize(struct current *current)
{
struct winsize ws;
* and reading back the cursor position.
* Note that this is only done once per call to linenoise rather than
* every time the line is refreshed for efficiency reasons.
+ *
+ * In more detail, we:
+ * (a) request current cursor position,
+ * (b) move cursor far right,
+ * (c) request cursor position again,
+ * (d) at last move back to the old position.
+ * This gives us the width without messing with the externally
+ * visible cursor position.
*/
+
if (current->cols == 0) {
+ int here;
+
current->cols = 80;
- /* Move cursor far right and report cursor position, then back to the left */
- fd_printf(current->fd, "\x1b[999C" "\x1b[6n");
-
- /* Parse the response: ESC [ rows ; cols R */
- if (fd_read_char(current->fd, 100) == 0x1b && fd_read_char(current->fd, 100) == '[') {
- int n = 0;
- while (1) {
- int ch = fd_read_char(current->fd, 100);
- if (ch == ';') {
- /* Ignore rows */
- n = 0;
- }
- else if (ch == 'R') {
- /* Got cols */
- if (n != 0 && n < 1000) {
- current->cols = n;
- }
- break;
- }
- else if (ch >= 0 && ch <= '9') {
- n = n * 10 + ch - '0';
- }
- else {
- break;
+ /* (a) */
+ if (queryCursor (current->fd, &here)) {
+ /* (b) */
+ fd_printf(current->fd, "\x1b[999C");
+
+ /* (c). Note: If (a) succeeded, then (c) should as well.
+ * For paranoia we still check and have a fallback action
+ * for (d) in case of failure..
+ */
+ if (!queryCursor (current->fd, ¤t->cols)) {
+ /* (d') Unable to get accurate position data, reset
+ * the cursor to the far left. While this may not
+ * restore the exact original position it should not
+ * be too bad.
+ */
+ fd_printf(current->fd, "\r");
+ } else {
+ /* (d) Reset the cursor back to the original location. */
+ if (current->cols > here) {
+ fd_printf(current->fd, "\x1b[%dD", current->cols - here);
}
}
- }
+ } /* 1st query failed, doing nothing => default 80 */
}
+
return 0;
}
}
}
/* Note that control characters are already translated in AsciiChar */
+ else if (k->wVirtualKeyCode == VK_CONTROL)
+ continue;
else {
#ifdef USE_UTF8
return k->uChar.UnicodeChar;
return -1;
}
-static int countColorControlChars(char* prompt, int plen)
+static int countColorControlChars(const char* prompt)
{
/* For windows we assume that there are no embedded ansi color
* control sequences.
/* Scan the prompt for embedded ansi color control sequences and
* discount them as characters/columns.
*/
- pchars -= countColorControlChars(prompt, plen);
+ pchars -= countColorControlChars(prompt);
/* Account for a line which is too long to fit in the window.
* Note that control chars require an extra column