diff options
| author | xyz <gky44px1999@gmail.com> | 2022-08-13 17:44:05 -0700 | 
|---|---|---|
| committer | xyz <gky44px1999@gmail.com> | 2022-08-13 17:44:05 -0700 | 
| commit | be2066bc16fe63e6f60c81e9c17885949108cf45 (patch) | |
| tree | 796b8b0cabbca3b2fc6f596a9d002218de83760b | |
| parent | 571bfa337454f09fd13d9c5cd0f3ec5c85f4c41b (diff) | |
| parent | 44adafe0069e73aa03a3829d7bb39591cd8b3f1d (diff) | |
Merge branch 'master' into fly
| -rw-r--r-- | LICENSE | 1 | ||||
| -rw-r--r-- | config.mk | 1 | ||||
| -rw-r--r-- | drw.c | 88 | ||||
| -rw-r--r-- | drw.h | 1 | ||||
| -rw-r--r-- | dwm.c | 123 | ||||
| -rw-r--r-- | util.c | 23 | 
6 files changed, 131 insertions, 106 deletions
| @@ -17,6 +17,7 @@ MIT/X Consortium License  © 2015-2016 Quentin Rameau <quinq@fifth.space>  © 2015-2016 Eric Pruitt <eric.pruitt@gmail.com>  © 2016-2017 Markus Teich <markus.teich@stusta.mhn.de> +© 2020-2022 Chris Down <chris@chrisdown.name>  Permission is hereby granted, free of charge, to any person obtaining a  copy of this software and associated documentation files (the "Software"), @@ -19,6 +19,7 @@ FREETYPELIBS = -lfontconfig -lXft  FREETYPEINC = /usr/include/freetype2  # OpenBSD (uncomment)  #FREETYPEINC = ${X11INC}/freetype2 +#MANPREFIX = ${PREFIX}/man  # includes and libs  INCS = -I${X11INC} -I${FREETYPEINC} @@ -251,12 +251,10 @@ drw_rect(Drw *drw, int x, int y, unsigned int w, unsigned int h, int filled, int  int  drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lpad, const char *text, int invert)  { -	char buf[1024]; -	int ty; -	unsigned int ew; +	int i, ty, ellipsis_x = 0; +	unsigned int tmpw, ew, ellipsis_w = 0, ellipsis_len;  	XftDraw *d = NULL;  	Fnt *usedfont, *curfont, *nextfont; -	size_t i, len;  	int utf8strlen, utf8charlen, render = x || y || w || h;  	long utf8codepoint = 0;  	const char *utf8str; @@ -264,13 +262,17 @@ drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lp  	FcPattern *fcpattern;  	FcPattern *match;  	XftResult result; -	int charexists = 0; +	int charexists = 0, overflow = 0; +	/* keep track of a couple codepoints for which we have no match. */ +	enum { nomatches_len = 64 }; +	static struct { long codepoint[nomatches_len]; unsigned int idx; } nomatches; +	static unsigned int ellipsis_width = 0; -	if (!drw || (render && !drw->scheme) || !text || !drw->fonts) +	if (!drw || (render && (!drw->scheme || !w)) || !text || !drw->fonts)  		return 0;  	if (!render) { -		w = ~w; +		w = invert ? invert : ~invert;  	} else {  		XSetForeground(drw->dpy, drw->gc, drw->scheme[invert ? ColFg : ColBg].pixel);  		XFillRectangle(drw->dpy, drw->drawable, drw->gc, x, y, w, h); @@ -282,8 +284,10 @@ drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lp  	}  	usedfont = drw->fonts; +	if (!ellipsis_width && render) +		ellipsis_width = drw_fontset_getwidth(drw, "...");  	while (1) { -		utf8strlen = 0; +		ew = ellipsis_len = utf8strlen = 0;  		utf8str = text;  		nextfont = NULL;  		while (*text) { @@ -291,9 +295,27 @@ drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lp  			for (curfont = drw->fonts; curfont; curfont = curfont->next) {  				charexists = charexists || XftCharExists(drw->dpy, curfont->xfont, utf8codepoint);  				if (charexists) { -					if (curfont == usedfont) { +					drw_font_getexts(curfont, text, utf8charlen, &tmpw, NULL); +					if (ew + ellipsis_width <= w) { +						/* keep track where the ellipsis still fits */ +						ellipsis_x = x + ew; +						ellipsis_w = w - ew; +						ellipsis_len = utf8strlen; +					} + +					if (ew + tmpw > w) { +						overflow = 1; +						/* called from drw_fontset_getwidth_clamp(): +						 * it wants the width AFTER the overflow +						 */ +						if (!render) +							x += tmpw; +						else +							utf8strlen = ellipsis_len; +					} else if (curfont == usedfont) {  						utf8strlen += utf8charlen;  						text += utf8charlen; +						ew += tmpw;  					} else {  						nextfont = curfont;  					} @@ -301,36 +323,25 @@ drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lp  				}  			} -			if (!charexists || nextfont) +			if (overflow || !charexists || nextfont)  				break;  			else  				charexists = 0;  		}  		if (utf8strlen) { -			drw_font_getexts(usedfont, utf8str, utf8strlen, &ew, NULL); -			/* shorten text if necessary */ -			for (len = MIN(utf8strlen, sizeof(buf) - 1); len && ew > w; len--) -				drw_font_getexts(usedfont, utf8str, len, &ew, NULL); - -			if (len) { -				memcpy(buf, utf8str, len); -				buf[len] = '\0'; -				if (len < utf8strlen) -					for (i = len; i && i > len - 3; buf[--i] = '.') -						; /* NOP */ - -				if (render) { -					ty = y + (h - usedfont->h) / 2 + usedfont->xfont->ascent; -					XftDrawStringUtf8(d, &drw->scheme[invert ? ColBg : ColFg], -					                  usedfont->xfont, x, ty, (XftChar8 *)buf, len); -				} -				x += ew; -				w -= ew; +			if (render) { +				ty = y + (h - usedfont->h) / 2 + usedfont->xfont->ascent; +				XftDrawStringUtf8(d, &drw->scheme[invert ? ColBg : ColFg], +				                  usedfont->xfont, x, ty, (XftChar8 *)utf8str, utf8strlen);  			} +			x += ew; +			w -= ew;  		} +		if (render && overflow) +			drw_text(drw, ellipsis_x, y, ellipsis_w, h, 0, "...", invert); -		if (!*text) { +		if (!*text || overflow) {  			break;  		} else if (nextfont) {  			charexists = 0; @@ -340,6 +351,12 @@ drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lp  			 * character must be drawn. */  			charexists = 1; +			for (i = 0; i < nomatches_len; ++i) { +				/* avoid calling XftFontMatch if we know we won't find a match */ +				if (utf8codepoint == nomatches.codepoint[i]) +					goto no_match; +			} +  			fccharset = FcCharSetCreate();  			FcCharSetAddChar(fccharset, utf8codepoint); @@ -368,6 +385,8 @@ drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lp  					curfont->next = usedfont;  				} else {  					xfont_free(usedfont); +					nomatches.codepoint[++nomatches.idx % nomatches_len] = utf8codepoint; +no_match:  					usedfont = drw->fonts;  				}  			} @@ -397,6 +416,15 @@ drw_fontset_getwidth(Drw *drw, const char *text)  	return drw_text(drw, 0, 0, 0, 0, 0, text, 0);  } +unsigned int +drw_fontset_getwidth_clamp(Drw *drw, const char *text, unsigned int n) +{ +	unsigned int tmp = 0; +	if (drw && drw->fonts && text && n) +		tmp = drw_text(drw, 0, 0, 0, 0, 0, text, n); +	return MIN(n, tmp); +} +  void  drw_font_getexts(Fnt *font, const char *text, unsigned int len, unsigned int *w, unsigned int *h)  { @@ -35,6 +35,7 @@ void drw_free(Drw *drw);  Fnt *drw_fontset_create(Drw* drw, const char *fonts[], size_t fontcount);  void drw_fontset_free(Fnt* set);  unsigned int drw_fontset_getwidth(Drw *drw, const char *text); +unsigned int drw_fontset_getwidth_clamp(Drw *drw, const char *text, unsigned int n);  void drw_font_getexts(Fnt *font, const char *text, unsigned int len, unsigned int *w, unsigned int *h);  /* Colorscheme abstraction */ @@ -89,7 +89,7 @@ struct Client {  	float mina, maxa;  	int x, y, w, h;  	int oldx, oldy, oldw, oldh; -	int basew, baseh, incw, inch, maxw, maxh, minw, minh; +	int basew, baseh, incw, inch, maxw, maxh, minw, minh, hintsvalid;  	int bw, oldbw;  	unsigned int tags;  	int isfixed, isfloating, isurgent, neverfocus, oldstate, isfullscreen; @@ -185,7 +185,7 @@ static void monocle(Monitor *m);  static void motionnotify(XEvent *e);  static void movemouse(const Arg *arg);  static Client *nexttiled(Client *c); -static void pop(Client *); +static void pop(Client *c);  static void propertynotify(XEvent *e);  static void quit(const Arg *arg);  static Monitor *recttomon(int x, int y, int w, int h); @@ -209,7 +209,7 @@ static void sigchld(int unused);  static void spawn(const Arg *arg);  static void tag(const Arg *arg);  static void tagmon(const Arg *arg); -static void tile(Monitor *); +static void tile(Monitor *m);  static void togglebar(const Arg *arg);  static void togglefloating(const Arg *arg);  static void togglefullscr(const Arg *arg); @@ -346,6 +346,8 @@ applysizehints(Client *c, int *x, int *y, int *w, int *h, int interact)  	if (*w < bh)  		*w = bh;  	if (resizehints || c->isfloating || !c->mon->lt[c->mon->sellt]->arrange) { +		if (!c->hintsvalid) +			updatesizehints(c);  		/* see last two sentences in ICCCM 4.1.2.3 */  		baseismin = c->basew == c->minw && c->baseh == c->minh;  		if (!baseismin) { /* temporarily remove base dimensions */ @@ -917,13 +919,11 @@ gettextprop(Window w, Atom atom, char *text, unsigned int size)  	text[0] = '\0';  	if (!XGetTextProperty(dpy, w, &name, atom) || !name.nitems)  		return 0; -	if (name.encoding == XA_STRING) +	if (name.encoding == XA_STRING) {  		strncpy(text, (char *)name.value, size - 1); -	else { -		if (XmbTextPropertyToTextList(dpy, &name, &list, &n) >= Success && n > 0 && *list) { -			strncpy(text, *list, size - 1); -			XFreeStringList(list); -		} +	} else if (XmbTextPropertyToTextList(dpy, &name, &list, &n) >= Success && n > 0 && *list) { +		strncpy(text, *list, size - 1); +		XFreeStringList(list);  	}  	text[size - 1] = '\0';  	XFree(name.value); @@ -1045,14 +1045,12 @@ manage(Window w, XWindowAttributes *wa)  		applyrules(c);  	} -	if (c->x + WIDTH(c) > c->mon->mx + c->mon->mw) -		c->x = c->mon->mx + c->mon->mw - WIDTH(c); -	if (c->y + HEIGHT(c) > c->mon->my + c->mon->mh) -		c->y = c->mon->my + c->mon->mh - HEIGHT(c); -	c->x = MAX(c->x, c->mon->mx); -	/* only fix client y-offset, if the client center might cover the bar */ -	c->y = MAX(c->y, ((c->mon->by == c->mon->my) && (c->x + (c->w / 2) >= c->mon->wx) -		&& (c->x + (c->w / 2) < c->mon->wx + c->mon->ww)) ? bh : c->mon->my); +	if (c->x + WIDTH(c) > c->mon->wx + c->mon->ww) +		c->x = c->mon->wx + c->mon->ww - WIDTH(c); +	if (c->y + HEIGHT(c) > c->mon->wy + c->mon->wh) +		c->y = c->mon->wy + c->mon->wh - HEIGHT(c); +	c->x = MAX(c->x, c->mon->wx); +	c->y = MAX(c->y, c->mon->wy);  	c->bw = borderpx;  	wc.border_width = c->bw; @@ -1065,7 +1063,7 @@ manage(Window w, XWindowAttributes *wa)  	XSelectInput(dpy, w, EnterWindowMask|FocusChangeMask|PropertyChangeMask|StructureNotifyMask);  	grabbuttons(c, 0);  	if (!c->isfloating) -		c->isfloating = c->oldstate = t || c->isfixed; +		c->isfloating = c->oldstate = trans != None || c->isfixed;  	if (c->isfloating)  		XRaiseWindow(dpy, c->win);  	attach(c); @@ -1098,9 +1096,7 @@ maprequest(XEvent *e)  	static XWindowAttributes wa;  	XMapRequestEvent *ev = &e->xmaprequest; -	if (!XGetWindowAttributes(dpy, ev->window, &wa)) -		return; -	if (wa.override_redirect) +	if (!XGetWindowAttributes(dpy, ev->window, &wa) || wa.override_redirect)  		return;  	if (!wintoclient(ev->window))  		manage(ev->window, &wa); @@ -1234,7 +1230,7 @@ propertynotify(XEvent *e)  				arrange(c->mon);  			break;  		case XA_WM_NORMAL_HINTS: -			updatesizehints(c); +			c->hintsvalid = 0;  			break;  		case XA_WM_HINTS:  			updatewmhints(c); @@ -1602,7 +1598,6 @@ setup(void)  	focus(NULL);  } -  void  seturgent(Client *c, int urg)  { @@ -1652,9 +1647,7 @@ spawn(const Arg *arg)  			close(ConnectionNumber(dpy));  		setsid();  		execvp(((char **)arg->v)[0], (char **)arg->v); -		fprintf(stderr, "dwm: execvp %s", ((char **)arg->v)[0]); -		perror(" failed"); -		exit(EXIT_SUCCESS); +		die("dwm: execvp '%s' failed:", ((char **)arg->v)[0]);  	}  } @@ -1786,6 +1779,7 @@ unmanage(Client *c, int destroyed)  		wc.border_width = c->oldbw;  		XGrabServer(dpy); /* avoid race conditions */  		XSetErrorHandler(xerrordummy); +		XSelectInput(dpy, c->win, NoEventMask);  		XConfigureWindow(dpy, c->win, CWBorderWidth, &wc); /* restore border */  		XUngrabButton(dpy, AnyButton, AnyModifier, c->win);  		setclientstate(c, WithdrawnState); @@ -1883,42 +1877,42 @@ updategeom(void)  				memcpy(&unique[j++], &info[i], sizeof(XineramaScreenInfo));  		XFree(info);  		nn = j; -		if (n <= nn) { /* new monitors available */ -			for (i = 0; i < (nn - n); i++) { -				for (m = mons; m && m->next; m = m->next); -				if (m) -					m->next = createmon(); -				else -					mons = createmon(); + +		/* new monitors if nn > n */ +		for (i = n; i < nn; i++) { +			for (m = mons; m && m->next; m = m->next); +			if (m) +				m->next = createmon(); +			else +				mons = createmon(); +		} +		for (i = 0, m = mons; i < nn && m; m = m->next, i++) +			if (i >= n +			|| unique[i].x_org != m->mx || unique[i].y_org != m->my +			|| unique[i].width != m->mw || unique[i].height != m->mh) +			{ +				dirty = 1; +				m->num = i; +				m->mx = m->wx = unique[i].x_org; +				m->my = m->wy = unique[i].y_org; +				m->mw = m->ww = unique[i].width; +				m->mh = m->wh = unique[i].height; +				updatebarpos(m);  			} -			for (i = 0, m = mons; i < nn && m; m = m->next, i++) -				if (i >= n -				|| unique[i].x_org != m->mx || unique[i].y_org != m->my -				|| unique[i].width != m->mw || unique[i].height != m->mh) -				{ -					dirty = 1; -					m->num = i; -					m->mx = m->wx = unique[i].x_org; -					m->my = m->wy = unique[i].y_org; -					m->mw = m->ww = unique[i].width; -					m->mh = m->wh = unique[i].height; -					updatebarpos(m); -				} -		} else { /* less monitors available nn < n */ -			for (i = nn; i < n; i++) { -				for (m = mons; m && m->next; m = m->next); -				while ((c = m->clients)) { -					dirty = 1; -					m->clients = c->next; -					detachstack(c); -					c->mon = mons; -					attach(c); -					attachstack(c); -				} -				if (m == selmon) -					selmon = mons; -				cleanupmon(m); +		/* removed monitors if n > nn */ +		for (i = nn; i < n; i++) { +			for (m = mons; m && m->next; m = m->next); +			while ((c = m->clients)) { +				dirty = 1; +				m->clients = c->next; +				detachstack(c); +				c->mon = mons; +				attach(c); +				attachstack(c);  			} +			if (m == selmon) +				selmon = mons; +			cleanupmon(m);  		}  		free(unique);  	} else @@ -1997,6 +1991,7 @@ updatesizehints(Client *c)  	} else  		c->maxa = c->mina = 0.0;  	c->isfixed = (c->maxw && c->maxh && c->maxw == c->minw && c->maxh == c->minh); +	c->hintsvalid = 1;  }  void @@ -2130,12 +2125,10 @@ zoom(const Arg *arg)  {  	Client *c = selmon->sel; -	if (!selmon->lt[selmon->sellt]->arrange -	|| (selmon->sel && selmon->sel->isfloating)) +	if (!selmon->lt[selmon->sellt]->arrange || !c || c->isfloating) +		return; +	if (c == nexttiled(selmon->clients) && !(c = nexttiled(c->next)))  		return; -	if (c == nexttiled(selmon->clients)) -		if (!c || !(c = nexttiled(c->next))) -			return;  	pop(c);  } @@ -6,18 +6,9 @@  #include "util.h" -void * -ecalloc(size_t nmemb, size_t size) -{ -	void *p; - -	if (!(p = calloc(nmemb, size))) -		die("calloc:"); -	return p; -} -  void -die(const char *fmt, ...) { +die(const char *fmt, ...) +{  	va_list ap;  	va_start(ap, fmt); @@ -33,3 +24,13 @@ die(const char *fmt, ...) {  	exit(1);  } + +void * +ecalloc(size_t nmemb, size_t size) +{ +	void *p; + +	if (!(p = calloc(nmemb, size))) +		die("calloc:"); +	return p; +} | 
