vi: discover window size even on serial consoles. optional
function old new delta edit_file 671 761 +90 wh_helper - 57 +57 query_screen_dimensions 54 63 +9 ar_main 533 542 +9 refresh 767 773 +6 vi_main 242 243 +1 text_yank 56 54 -2 get_terminal_width_height 180 135 -45 ------------------------------------------------------------------------------ (add/remove: 1/0 grow/shrink: 5/2 up/down: 172/-47) Total: 125 bytes Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
parent
def4783a8a
commit
c175c46647
@ -168,6 +168,18 @@ config FEATURE_VI_WIN_RESIZE
|
|||||||
help
|
help
|
||||||
Make busybox vi behave nicely with terminals that get resized.
|
Make busybox vi behave nicely with terminals that get resized.
|
||||||
|
|
||||||
|
config FEATURE_VI_ASK_TERMINAL
|
||||||
|
bool "Use 'tell me cursor position' ESC sequence to measure window"
|
||||||
|
default n
|
||||||
|
depends on VI
|
||||||
|
help
|
||||||
|
If terminal size can't be retrieved and $LINES/$COLUMNS are not set,
|
||||||
|
this option makes vi perform a last-ditch effort to find it:
|
||||||
|
vi positions cursor to 999,999 and asks terminal to report real
|
||||||
|
cursor position using "ESC [ 6 n" escape sequence, then reads stdin.
|
||||||
|
|
||||||
|
This is not clean but helps a lot on serial lines and such.
|
||||||
|
|
||||||
config FEATURE_VI_OPTIMIZE_CURSOR
|
config FEATURE_VI_OPTIMIZE_CURSOR
|
||||||
bool "Optimize cursor movement"
|
bool "Optimize cursor movement"
|
||||||
default y
|
default y
|
||||||
|
21
editors/vi.c
21
editors/vi.c
@ -138,6 +138,9 @@ struct globals {
|
|||||||
int save_argc; // how many file names on cmd line
|
int save_argc; // how many file names on cmd line
|
||||||
int cmdcnt; // repetition count
|
int cmdcnt; // repetition count
|
||||||
unsigned rows, columns; // the terminal screen is this size
|
unsigned rows, columns; // the terminal screen is this size
|
||||||
|
#if ENABLE_FEATURE_VI_ASK_TERMINAL
|
||||||
|
int get_rowcol_error;
|
||||||
|
#endif
|
||||||
int crow, ccol; // cursor is on Crow x Ccol
|
int crow, ccol; // cursor is on Crow x Ccol
|
||||||
int offset; // chars scrolled off the screen to the left
|
int offset; // chars scrolled off the screen to the left
|
||||||
int have_status_msg; // is default edit status needed?
|
int have_status_msg; // is default edit status needed?
|
||||||
@ -503,6 +506,10 @@ static int init_text_buffer(char *fn)
|
|||||||
#if ENABLE_FEATURE_VI_WIN_RESIZE
|
#if ENABLE_FEATURE_VI_WIN_RESIZE
|
||||||
static void query_screen_dimensions(void)
|
static void query_screen_dimensions(void)
|
||||||
{
|
{
|
||||||
|
# if ENABLE_FEATURE_VI_ASK_TERMINAL
|
||||||
|
if (!G.get_rowcol_error)
|
||||||
|
G.get_rowcol_error =
|
||||||
|
# endif
|
||||||
get_terminal_width_height(STDIN_FILENO, &columns, &rows);
|
get_terminal_width_height(STDIN_FILENO, &columns, &rows);
|
||||||
if (rows > MAX_SCR_ROWS)
|
if (rows > MAX_SCR_ROWS)
|
||||||
rows = MAX_SCR_ROWS;
|
rows = MAX_SCR_ROWS;
|
||||||
@ -530,6 +537,20 @@ static void edit_file(char *fn)
|
|||||||
columns = 80;
|
columns = 80;
|
||||||
size = 0;
|
size = 0;
|
||||||
query_screen_dimensions();
|
query_screen_dimensions();
|
||||||
|
#if ENABLE_FEATURE_VI_ASK_TERMINAL
|
||||||
|
if (G.get_rowcol_error /* TODO? && no input on stdin */) {
|
||||||
|
uint64_t k;
|
||||||
|
write1("\033[999;999H" "\033[6n");
|
||||||
|
fflush_all();
|
||||||
|
k = read_key(STDIN_FILENO, readbuffer, /*timeout_ms:*/ 100);
|
||||||
|
if ((int32_t)k == KEYCODE_CURSOR_POS) {
|
||||||
|
uint32_t rc = (k >> 32);
|
||||||
|
columns = (rc & 0x7fff);
|
||||||
|
rows = ((rc >> 16) & 0x7fff);
|
||||||
|
}
|
||||||
|
query_screen_dimensions();
|
||||||
|
}
|
||||||
|
#endif
|
||||||
new_screen(rows, columns); // get memory for virtual screen
|
new_screen(rows, columns); // get memory for virtual screen
|
||||||
init_text_buffer(fn);
|
init_text_buffer(fn);
|
||||||
|
|
||||||
|
@ -214,7 +214,7 @@ int64_t FAST_FUNC read_key(int fd, char *buffer, int timeout)
|
|||||||
}
|
}
|
||||||
n++;
|
n++;
|
||||||
/* Try to decipher "ESC [ NNN ; NNN R" sequence */
|
/* Try to decipher "ESC [ NNN ; NNN R" sequence */
|
||||||
if (ENABLE_FEATURE_EDITING_ASK_TERMINAL
|
if ((ENABLE_FEATURE_EDITING_ASK_TERMINAL || ENABLE_FEATURE_VI_ASK_TERMINAL)
|
||||||
&& n >= 5
|
&& n >= 5
|
||||||
&& buffer[0] == '['
|
&& buffer[0] == '['
|
||||||
&& buffer[n-1] == 'R'
|
&& buffer[n-1] == 'R'
|
||||||
|
@ -210,34 +210,40 @@ char* FAST_FUNC xmalloc_ttyname(int fd)
|
|||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int wh_helper(int value, int def_val, const char *env_name, int *err)
|
||||||
|
{
|
||||||
|
if (value == 0) {
|
||||||
|
char *s = getenv(env_name);
|
||||||
|
if (s) {
|
||||||
|
value = atoi(s);
|
||||||
|
/* If LINES/COLUMNS are set, pretent that there is
|
||||||
|
* no error getting w/h, this prevents some ugly
|
||||||
|
* cursor tricks by our callers */
|
||||||
|
*err = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (value <= 1 || value >= 30000)
|
||||||
|
value = def_val;
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
/* It is perfectly ok to pass in a NULL for either width or for
|
/* It is perfectly ok to pass in a NULL for either width or for
|
||||||
* height, in which case that value will not be set. */
|
* height, in which case that value will not be set. */
|
||||||
int FAST_FUNC get_terminal_width_height(int fd, unsigned *width, unsigned *height)
|
int FAST_FUNC get_terminal_width_height(int fd, unsigned *width, unsigned *height)
|
||||||
{
|
{
|
||||||
struct winsize win = { 0, 0, 0, 0 };
|
struct winsize win;
|
||||||
int ret = ioctl(fd, TIOCGWINSZ, &win);
|
int err;
|
||||||
|
|
||||||
if (height) {
|
win.ws_row = 0;
|
||||||
if (!win.ws_row) {
|
win.ws_col = 0;
|
||||||
char *s = getenv("LINES");
|
/* I've seen ioctl returning 0, but row/col is (still?) 0.
|
||||||
if (s) win.ws_row = atoi(s);
|
* We treat that as an error too. */
|
||||||
}
|
err = ioctl(fd, TIOCGWINSZ, &win) != 0 || win.ws_row == 0;
|
||||||
if (win.ws_row <= 1 || win.ws_row >= 30000)
|
if (height)
|
||||||
win.ws_row = 24;
|
*height = wh_helper(win.ws_row, 24, "LINES", &err);
|
||||||
*height = (int) win.ws_row;
|
if (width)
|
||||||
}
|
*width = wh_helper(win.ws_col, 80, "COLUMNS", &err);
|
||||||
|
return err;
|
||||||
if (width) {
|
|
||||||
if (!win.ws_col) {
|
|
||||||
char *s = getenv("COLUMNS");
|
|
||||||
if (s) win.ws_col = atoi(s);
|
|
||||||
}
|
|
||||||
if (win.ws_col <= 1 || win.ws_col >= 30000)
|
|
||||||
win.ws_col = 80;
|
|
||||||
*width = (int) win.ws_col;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int FAST_FUNC tcsetattr_stdin_TCSANOW(const struct termios *tp)
|
int FAST_FUNC tcsetattr_stdin_TCSANOW(const struct termios *tp)
|
||||||
|
Loading…
Reference in New Issue
Block a user