2015-04-04 22:33:30 +03:00
|
|
|
/*
|
|
|
|
* rline - a line reading library.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "kbd.h"
|
|
|
|
#include "rline.h"
|
|
|
|
|
|
|
|
void rline_redraw(rline_context_t * context) {
|
|
|
|
printf("\033[u%s\033[K", context->buffer);
|
|
|
|
for (int i = context->offset; i < context->collected; ++i) {
|
|
|
|
printf("\033[D");
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
void rline_redraw_clean(rline_context_t * context) {
|
|
|
|
printf("\033[u%s", context->buffer);
|
|
|
|
for (int i = context->offset; i < context->collected; ++i) {
|
|
|
|
printf("\033[D");
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
int rline(char * buffer, int buf_size, rline_callbacks_t * callbacks) {
|
|
|
|
/* Initialize context */
|
|
|
|
rline_context_t context = {
|
|
|
|
buffer,
|
|
|
|
callbacks,
|
|
|
|
0,
|
|
|
|
buf_size,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
};
|
|
|
|
|
|
|
|
printf("\033[s");
|
|
|
|
fflush(stdout);
|
|
|
|
|
|
|
|
key_event_state_t kbd_state = {0};
|
|
|
|
|
|
|
|
/* Read keys */
|
|
|
|
while ((context.collected < context.requested) && (!context.newline)) {
|
|
|
|
uint32_t key_sym = kbd_key(&kbd_state, fgetc(stdin));
|
|
|
|
if (key_sym == KEY_NONE) continue;
|
|
|
|
switch (key_sym) {
|
|
|
|
case KEY_CTRL_C:
|
|
|
|
printf("^C\n");
|
|
|
|
context.buffer[0] = '\0';
|
|
|
|
return 0;
|
|
|
|
case KEY_CTRL_R:
|
|
|
|
if (callbacks->rev_search) {
|
|
|
|
callbacks->rev_search(&context);
|
|
|
|
return context.collected;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case KEY_ARROW_UP:
|
2015-05-19 05:29:33 +03:00
|
|
|
case KEY_CTRL_P:
|
2015-04-04 22:33:30 +03:00
|
|
|
if (callbacks->key_up) {
|
|
|
|
callbacks->key_up(&context);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case KEY_ARROW_DOWN:
|
2015-05-19 05:29:33 +03:00
|
|
|
case KEY_CTRL_N:
|
2015-04-04 22:33:30 +03:00
|
|
|
if (callbacks->key_down) {
|
|
|
|
callbacks->key_down(&context);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case KEY_ARROW_RIGHT:
|
|
|
|
if (callbacks->key_right) {
|
|
|
|
callbacks->key_right(&context);
|
|
|
|
} else {
|
|
|
|
if (context.offset < context.collected) {
|
|
|
|
printf("\033[C");
|
|
|
|
fflush(stdout);
|
|
|
|
context.offset++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case KEY_ARROW_LEFT:
|
|
|
|
if (callbacks->key_left) {
|
|
|
|
callbacks->key_left(&context);
|
|
|
|
} else {
|
|
|
|
if (context.offset > 0) {
|
|
|
|
printf("\033[D");
|
|
|
|
fflush(stdout);
|
|
|
|
context.offset--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
2015-08-20 20:56:22 +03:00
|
|
|
case KEY_CTRL_A:
|
2015-08-20 20:34:59 +03:00
|
|
|
case KEY_HOME:
|
|
|
|
while (context.offset > 0) {
|
|
|
|
printf("\033[D");
|
|
|
|
context.offset--;
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
|
|
|
continue;
|
2015-08-20 20:56:22 +03:00
|
|
|
case KEY_CTRL_E:
|
2015-08-20 20:34:59 +03:00
|
|
|
case KEY_END:
|
|
|
|
while (context.offset < context.collected) {
|
|
|
|
printf("\033[C");
|
|
|
|
context.offset++;
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
|
|
|
continue;
|
2015-08-20 20:56:22 +03:00
|
|
|
case KEY_CTRL_D:
|
|
|
|
if (context.collected == 0) {
|
|
|
|
printf("exit\n");
|
|
|
|
sprintf(context.buffer, "exit\n");
|
|
|
|
return strlen(context.buffer);
|
|
|
|
}
|
|
|
|
/* Intentional fallthrough */
|
2015-08-20 20:34:59 +03:00
|
|
|
case KEY_DEL:
|
|
|
|
if (context.collected) {
|
|
|
|
if (context.offset == context.collected) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
int remaining = context.collected - context.offset;
|
|
|
|
for (int i = 1; i < remaining; ++i) {
|
|
|
|
printf("%c", context.buffer[context.offset + i]);
|
|
|
|
context.buffer[context.offset + i - 1] = context.buffer[context.offset + i];
|
|
|
|
}
|
|
|
|
printf(" ");
|
|
|
|
for (int i = 0; i < remaining; ++i) {
|
|
|
|
printf("\033[D");
|
|
|
|
}
|
|
|
|
context.collected--;
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
continue;
|
2015-04-04 22:33:30 +03:00
|
|
|
case KEY_BACKSPACE:
|
|
|
|
if (context.collected) {
|
|
|
|
if (!context.offset) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
printf("\010 \010");
|
|
|
|
if (context.offset != context.collected) {
|
|
|
|
int remaining = context.collected - context.offset;
|
|
|
|
for (int i = 0; i < remaining; ++i) {
|
|
|
|
printf("%c", context.buffer[context.offset + i]);
|
|
|
|
context.buffer[context.offset + i - 1] = context.buffer[context.offset + i];
|
|
|
|
}
|
|
|
|
printf(" ");
|
|
|
|
for (int i = 0; i < remaining + 1; ++i) {
|
|
|
|
printf("\033[D");
|
|
|
|
}
|
|
|
|
context.offset--;
|
|
|
|
context.collected--;
|
|
|
|
} else {
|
|
|
|
context.buffer[--context.collected] = '\0';
|
|
|
|
context.offset--;
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case KEY_CTRL_L: /* ^L: Clear Screen, redraw prompt and buffer */
|
|
|
|
printf("\033[H\033[2J");
|
|
|
|
if (callbacks->redraw_prompt) {
|
|
|
|
callbacks->redraw_prompt(&context);
|
|
|
|
}
|
|
|
|
printf("\033[s");
|
|
|
|
rline_redraw_clean(&context);
|
|
|
|
continue;
|
|
|
|
case KEY_CTRL_W:
|
|
|
|
/*
|
|
|
|
* Erase word before cursor.
|
|
|
|
* If the character before the cursor is a space, delete it.
|
|
|
|
* Continue deleting until the previous character is a space.
|
|
|
|
*/
|
|
|
|
if (context.collected) {
|
|
|
|
if (!context.offset) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
do {
|
|
|
|
printf("\010 \010");
|
|
|
|
if (context.offset != context.collected) {
|
|
|
|
int remaining = context.collected - context.offset;
|
|
|
|
for (int i = 0; i < remaining; ++i) {
|
|
|
|
printf("%c", context.buffer[context.offset + i]);
|
|
|
|
context.buffer[context.offset + i - 1] = context.buffer[context.offset + i];
|
|
|
|
}
|
|
|
|
printf(" ");
|
|
|
|
for (int i = 0; i < remaining + 1; ++i) {
|
|
|
|
printf("\033[D");
|
|
|
|
}
|
|
|
|
context.offset--;
|
|
|
|
context.collected--;
|
|
|
|
} else {
|
|
|
|
context.buffer[--context.collected] = '\0';
|
|
|
|
context.offset--;
|
|
|
|
}
|
|
|
|
} while ((context.offset) && (context.buffer[context.offset-1] != ' '));
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case '\t':
|
|
|
|
if (callbacks->tab_complete) {
|
|
|
|
callbacks->tab_complete(&context);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case '\n':
|
|
|
|
while (context.offset < context.collected) {
|
|
|
|
printf("\033[C");
|
|
|
|
context.offset++;
|
|
|
|
}
|
|
|
|
if (context.collected < context.requested) {
|
|
|
|
context.buffer[context.collected] = '\n';
|
|
|
|
context.buffer[++context.collected] = '\0';
|
|
|
|
context.offset++;
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
fflush(stdout);
|
|
|
|
context.newline = 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (context.offset != context.collected) {
|
|
|
|
for (int i = context.collected; i > context.offset; --i) {
|
|
|
|
context.buffer[i] = context.buffer[i-1];
|
|
|
|
}
|
|
|
|
if (context.collected < context.requested) {
|
|
|
|
context.buffer[context.offset] = (char)key_sym;
|
|
|
|
context.buffer[++context.collected] = '\0';
|
|
|
|
context.offset++;
|
|
|
|
}
|
|
|
|
for (int i = context.offset - 1; i < context.collected; ++i) {
|
|
|
|
printf("%c", context.buffer[i]);
|
|
|
|
}
|
|
|
|
for (int i = context.offset; i < context.collected; ++i) {
|
|
|
|
printf("\033[D");
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
|
|
|
} else {
|
|
|
|
printf("%c", (char)key_sym);
|
|
|
|
if (context.collected < context.requested) {
|
|
|
|
context.buffer[context.collected] = (char)key_sym;
|
|
|
|
context.buffer[++context.collected] = '\0';
|
|
|
|
context.offset++;
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Cap that with a null */
|
|
|
|
context.buffer[context.collected] = '\0';
|
|
|
|
return context.collected;
|
|
|
|
}
|