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:
		@@ -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
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										23
									
								
								editors/vi.c
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								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,7 +506,11 @@ 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)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	get_terminal_width_height(STDIN_FILENO, &columns, &rows);
 | 
					# if ENABLE_FEATURE_VI_ASK_TERMINAL
 | 
				
			||||||
 | 
						if (!G.get_rowcol_error)
 | 
				
			||||||
 | 
							G.get_rowcol_error =
 | 
				
			||||||
 | 
					# endif
 | 
				
			||||||
 | 
								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;
 | 
				
			||||||
	if (columns > MAX_SCR_COLS)
 | 
						if (columns > MAX_SCR_COLS)
 | 
				
			||||||
@@ -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)
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user