initial implementation of selection and clipboard
This commit is contained in:
		
							
								
								
									
										107
									
								
								st.c
									
									
									
									
									
								
							
							
						
						
									
										107
									
								
								st.c
									
									
									
									
									
								
							| @@ -177,6 +177,9 @@ static char* kmap(KeySym); | |||||||
| static void kpress(XEvent *); | static void kpress(XEvent *); | ||||||
| static void resize(XEvent *); | static void resize(XEvent *); | ||||||
| static void focus(XEvent *); | static void focus(XEvent *); | ||||||
|  | static void brelease(XEvent *e); | ||||||
|  | static void bpress(XEvent *e); | ||||||
|  | static void bmotion(XEvent *e); | ||||||
|  |  | ||||||
|  |  | ||||||
| static void (*handler[LASTEvent])(XEvent *) = { | static void (*handler[LASTEvent])(XEvent *) = { | ||||||
| @@ -185,6 +188,9 @@ static void (*handler[LASTEvent])(XEvent *) = { | |||||||
| 	[ConfigureNotify] = resize, | 	[ConfigureNotify] = resize, | ||||||
| 	[FocusIn] = focus, | 	[FocusIn] = focus, | ||||||
| 	[FocusOut] = focus, | 	[FocusOut] = focus, | ||||||
|  | 	[MotionNotify] = bmotion, | ||||||
|  | 	[ButtonPress] = bpress, | ||||||
|  | 	[ButtonRelease] = brelease, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| /* Globals */ | /* Globals */ | ||||||
| @@ -195,6 +201,103 @@ static CSIEscape escseq; | |||||||
| static int cmdfd; | static int cmdfd; | ||||||
| static pid_t pid; | static pid_t pid; | ||||||
|  |  | ||||||
|  | /* selection */ | ||||||
|  | // TODO: use X11 clipboard | ||||||
|  | static int selmode = 0; | ||||||
|  | static int selbx = -1, selby; | ||||||
|  | static int selex, seley; | ||||||
|  | int sb[2], se[2]; | ||||||
|  | static const char *clipboard = NULL; | ||||||
|  |  | ||||||
|  | static inline int selected(int x, int y) { | ||||||
|  | 	if ((seley==y && selby==y)) { | ||||||
|  | 		int bx = MIN(selbx, selex); | ||||||
|  | 		int ex = MAX(selbx, selex); | ||||||
|  | 		return if(x>=bx && x<=ex) | ||||||
|  | 	} | ||||||
|  | 	return (((y>sb[1] && y<se[1]) || (y==se[1] && x<=se[0])) || \ | ||||||
|  | 		(y==sb[1] && x>=sb[0] && (x<=se[0] || sb[1]!=se[1]))) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void getbuttoninfo(XEvent *e, int *b, int *x, int *y) { | ||||||
|  | 	if(b) *b = e->xbutton.state, | ||||||
|  | 		*b=*b==4096?5:*b==2048?4:*b==1024?3:*b==512?2:*b==256?1:-1; | ||||||
|  | 	*x = e->xbutton.x/xw.cw; | ||||||
|  | 	*y = e->xbutton.y/xw.ch; | ||||||
|  | 	sb[0] = selby<seley?selbx:selex; | ||||||
|  | 	sb[1] = MIN(selby, seley); | ||||||
|  | 	se[0] = selby<seley?selex:selbx; | ||||||
|  | 	se[1] = MAX(selby, seley); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void bpress(XEvent *e) { | ||||||
|  | 	selmode = 1; | ||||||
|  | 	selex = selbx = e->xbutton.x/xw.cw; | ||||||
|  | 	seley = selby = e->xbutton.y/xw.ch; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static char *getseltext() { | ||||||
|  | 	char *str, *ptr; | ||||||
|  | 	int ls, x, y, sz; | ||||||
|  | 	if(selbx==-1) | ||||||
|  | 		return NULL; | ||||||
|  | 	sz = ((xw.w/xw.ch) * (se[1]-sb[1]+2)); | ||||||
|  | 	ptr = str = malloc (sz); | ||||||
|  | 	for(y = 0; y < term.row; y++) { | ||||||
|  | 		for(x = 0; x < term.col; x++) { | ||||||
|  | 			if(term.line[y][x].c && (ls=selected(x, y))) { | ||||||
|  | 				*ptr = term.line[y][x].c; | ||||||
|  | 				ptr++; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		if (ls) { | ||||||
|  | 			*ptr = '\n'; | ||||||
|  | 			ptr++; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	*ptr = 0; | ||||||
|  | 	return str; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void clipboard_copy(const char *str) { | ||||||
|  | 	free((void *)clipboard); | ||||||
|  | 	clipboard = str; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void clipboard_paste() { | ||||||
|  | 	if(clipboard) | ||||||
|  | 		ttywrite(clipboard, strlen(clipboard)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // TODO: doubleclick to select word | ||||||
|  | static void brelease(XEvent *e) { | ||||||
|  | 	int b; | ||||||
|  | 	selmode = 0; | ||||||
|  | 	getbuttoninfo(e, &b, &selex, &seley); | ||||||
|  | 	if(b==4) | ||||||
|  | 		tscrollup(1); | ||||||
|  | 	else | ||||||
|  | 	if(b==5) | ||||||
|  | 		tscrolldown(1); | ||||||
|  | 	else | ||||||
|  | 	if(selbx==selex && selby==seley) { | ||||||
|  | 		selbx = -1; | ||||||
|  | 		if(b==2) | ||||||
|  | 			clipboard_paste(); | ||||||
|  | 	} else { | ||||||
|  | 		if(b==1) | ||||||
|  | 			clipboard_copy(getseltext()); | ||||||
|  | 	} | ||||||
|  | 	draw(1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static void bmotion(XEvent *e) { | ||||||
|  | 	if (selmode) { | ||||||
|  | 		getbuttoninfo(e, NULL, &selex, &seley); | ||||||
|  | 		draw(1); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| #ifdef DEBUG | #ifdef DEBUG | ||||||
| void | void | ||||||
| tdump(void) { | tdump(void) { | ||||||
| @@ -1218,6 +1321,8 @@ draw(int redraw_all) { | |||||||
| 		i = ox = 0; | 		i = ox = 0; | ||||||
| 		for(x = 0; x < term.col; x++) { | 		for(x = 0; x < term.col; x++) { | ||||||
| 			new = term.line[y][x]; | 			new = term.line[y][x]; | ||||||
|  | 			if(selbx!=-1 && new.c && selected(x, y)) | ||||||
|  | 				new.mode = ATTR_REVERSE; | ||||||
| 			if(i > 0 && (!(new.state & GLYPH_SET) || ATTRCMP(base, new) || | 			if(i > 0 && (!(new.state & GLYPH_SET) || ATTRCMP(base, new) || | ||||||
| 					i >= DRAW_BUF_SIZ)) { | 					i >= DRAW_BUF_SIZ)) { | ||||||
| 				xdraws(buf, base, ox, y, i); | 				xdraws(buf, base, ox, y, i); | ||||||
| @@ -1335,7 +1440,7 @@ run(void) { | |||||||
| 	XEvent ev; | 	XEvent ev; | ||||||
| 	fd_set rfd; | 	fd_set rfd; | ||||||
| 	int xfd = XConnectionNumber(xw.dis); | 	int xfd = XConnectionNumber(xw.dis); | ||||||
| 	long mask = ExposureMask | KeyPressMask | StructureNotifyMask | FocusChangeMask; | 	long mask = ExposureMask | KeyPressMask | StructureNotifyMask | FocusChangeMask | PointerMotionMask | ButtonPressMask | ButtonReleaseMask; | ||||||
|  |  | ||||||
| 	XSelectInput(xw.dis, xw.win, mask); | 	XSelectInput(xw.dis, xw.win, mask); | ||||||
| 	XResizeWindow(xw.dis, xw.win, xw.w, xw.h); /* XXX: fix resize bug in wmii (?) */ | 	XResizeWindow(xw.dis, xw.win, xw.w, xw.h); /* XXX: fix resize bug in wmii (?) */ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 pancake
					pancake