You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

446 lines
11 KiB

  1. /* See LICENSE file for copyright and license details. */
  2. #include <stdio.h>
  3. #include <stdlib.h>
  4. #include <string.h>
  5. #include <X11/Xlib.h>
  6. #include <X11/Xft/Xft.h>
  7. #include "drw.h"
  8. #include "util.h"
  9. #define UTF_INVALID 0xFFFD
  10. static int
  11. utf8decode(const char *s_in, long *u, int *err)
  12. {
  13. static const unsigned char lens[] = {
  14. /* 0XXXX */ 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
  15. /* 10XXX */ 0, 0, 0, 0, 0, 0, 0, 0, /* invalid */
  16. /* 110XX */ 2, 2, 2, 2,
  17. /* 1110X */ 3, 3,
  18. /* 11110 */ 4,
  19. /* 11111 */ 0, /* invalid */
  20. };
  21. static const unsigned char leading_mask[] = { 0x7F, 0x1F, 0x0F, 0x07 };
  22. static const unsigned int overlong[] = { 0x0, 0x80, 0x0800, 0x10000 };
  23. const unsigned char *s = (const unsigned char *)s_in;
  24. int len = lens[*s >> 3];
  25. *u = UTF_INVALID;
  26. *err = 1;
  27. if (len == 0)
  28. return 1;
  29. long cp = s[0] & leading_mask[len - 1];
  30. for (int i = 1; i < len; ++i) {
  31. if (s[i] == '\0' || (s[i] & 0xC0) != 0x80)
  32. return i;
  33. cp = (cp << 6) | (s[i] & 0x3F);
  34. }
  35. /* out of range, surrogate, overlong encoding */
  36. if (cp > 0x10FFFF || (cp >> 11) == 0x1B || cp < overlong[len - 1])
  37. return len;
  38. *err = 0;
  39. *u = cp;
  40. return len;
  41. }
  42. Drw *
  43. drw_create(Display *dpy, int screen, Window root, unsigned int w, unsigned int h)
  44. {
  45. Drw *drw = ecalloc(1, sizeof(Drw));
  46. drw->dpy = dpy;
  47. drw->screen = screen;
  48. drw->root = root;
  49. drw->w = w;
  50. drw->h = h;
  51. drw->drawable = XCreatePixmap(dpy, root, w, h, DefaultDepth(dpy, screen));
  52. drw->gc = XCreateGC(dpy, root, 0, NULL);
  53. XSetLineAttributes(dpy, drw->gc, 1, LineSolid, CapButt, JoinMiter);
  54. return drw;
  55. }
  56. void
  57. drw_resize(Drw *drw, unsigned int w, unsigned int h)
  58. {
  59. if (!drw)
  60. return;
  61. drw->w = w;
  62. drw->h = h;
  63. if (drw->drawable)
  64. XFreePixmap(drw->dpy, drw->drawable);
  65. drw->drawable = XCreatePixmap(drw->dpy, drw->root, w, h, DefaultDepth(drw->dpy, drw->screen));
  66. }
  67. void
  68. drw_free(Drw *drw)
  69. {
  70. XFreePixmap(drw->dpy, drw->drawable);
  71. XFreeGC(drw->dpy, drw->gc);
  72. drw_fontset_free(drw->fonts);
  73. free(drw);
  74. }
  75. /* This function is an implementation detail. Library users should use
  76. * drw_fontset_create instead.
  77. */
  78. static Fnt *
  79. xfont_create(Drw *drw, const char *fontname, FcPattern *fontpattern)
  80. {
  81. Fnt *font;
  82. XftFont *xfont = NULL;
  83. FcPattern *pattern = NULL;
  84. if (fontname) {
  85. /* Using the pattern found at font->xfont->pattern does not yield the
  86. * same substitution results as using the pattern returned by
  87. * FcNameParse; using the latter results in the desired fallback
  88. * behaviour whereas the former just results in missing-character
  89. * rectangles being drawn, at least with some fonts. */
  90. if (!(xfont = XftFontOpenName(drw->dpy, drw->screen, fontname))) {
  91. fprintf(stderr, "error, cannot load font from name: '%s'\n", fontname);
  92. return NULL;
  93. }
  94. if (!(pattern = FcNameParse((FcChar8 *) fontname))) {
  95. fprintf(stderr, "error, cannot parse font name to pattern: '%s'\n", fontname);
  96. XftFontClose(drw->dpy, xfont);
  97. return NULL;
  98. }
  99. } else if (fontpattern) {
  100. if (!(xfont = XftFontOpenPattern(drw->dpy, fontpattern))) {
  101. fprintf(stderr, "error, cannot load font from pattern.\n");
  102. return NULL;
  103. }
  104. } else {
  105. die("no font specified.");
  106. }
  107. font = ecalloc(1, sizeof(Fnt));
  108. font->xfont = xfont;
  109. font->pattern = pattern;
  110. font->h = xfont->ascent + xfont->descent;
  111. font->dpy = drw->dpy;
  112. return font;
  113. }
  114. static void
  115. xfont_free(Fnt *font)
  116. {
  117. if (!font)
  118. return;
  119. if (font->pattern)
  120. FcPatternDestroy(font->pattern);
  121. XftFontClose(font->dpy, font->xfont);
  122. free(font);
  123. }
  124. Fnt*
  125. drw_fontset_create(Drw* drw, const char *fonts[], size_t fontcount)
  126. {
  127. Fnt *cur, *ret = NULL;
  128. size_t i;
  129. if (!drw || !fonts)
  130. return NULL;
  131. for (i = 1; i <= fontcount; i++) {
  132. if ((cur = xfont_create(drw, fonts[fontcount - i], NULL))) {
  133. cur->next = ret;
  134. ret = cur;
  135. }
  136. }
  137. return (drw->fonts = ret);
  138. }
  139. void
  140. drw_fontset_free(Fnt *font)
  141. {
  142. if (font) {
  143. drw_fontset_free(font->next);
  144. xfont_free(font);
  145. }
  146. }
  147. void
  148. drw_clr_create(Drw *drw, Clr *dest, const char *clrname)
  149. {
  150. if (!drw || !dest || !clrname)
  151. return;
  152. if (!XftColorAllocName(drw->dpy, DefaultVisual(drw->dpy, drw->screen),
  153. DefaultColormap(drw->dpy, drw->screen),
  154. clrname, dest))
  155. die("error, cannot allocate color '%s'", clrname);
  156. }
  157. /* Wrapper to create color schemes. The caller has to call free(3) on the
  158. * returned color scheme when done using it. */
  159. Clr *
  160. drw_scm_create(Drw *drw, const char *clrnames[], size_t clrcount)
  161. {
  162. size_t i;
  163. Clr *ret;
  164. /* need at least two colors for a scheme */
  165. if (!drw || !clrnames || clrcount < 2 || !(ret = ecalloc(clrcount, sizeof(XftColor))))
  166. return NULL;
  167. for (i = 0; i < clrcount; i++)
  168. drw_clr_create(drw, &ret[i], clrnames[i]);
  169. return ret;
  170. }
  171. void
  172. drw_setfontset(Drw *drw, Fnt *set)
  173. {
  174. if (drw)
  175. drw->fonts = set;
  176. }
  177. void
  178. drw_setscheme(Drw *drw, Clr *scm)
  179. {
  180. if (drw)
  181. drw->scheme = scm;
  182. }
  183. void
  184. drw_rect(Drw *drw, int x, int y, unsigned int w, unsigned int h, int filled, int invert)
  185. {
  186. if (!drw || !drw->scheme)
  187. return;
  188. XSetForeground(drw->dpy, drw->gc, invert ? drw->scheme[ColBg].pixel : drw->scheme[ColFg].pixel);
  189. if (filled)
  190. XFillRectangle(drw->dpy, drw->drawable, drw->gc, x, y, w, h);
  191. else
  192. XDrawRectangle(drw->dpy, drw->drawable, drw->gc, x, y, w - 1, h - 1);
  193. }
  194. int
  195. drw_text(Drw *drw, int x, int y, unsigned int w, unsigned int h, unsigned int lpad, const char *text, int invert)
  196. {
  197. int ty, ellipsis_x = 0;
  198. unsigned int tmpw, ew, ellipsis_w = 0, ellipsis_len, hash, h0, h1;
  199. XftDraw *d = NULL;
  200. Fnt *usedfont, *curfont, *nextfont;
  201. int utf8strlen, utf8charlen, utf8err, render = x || y || w || h;
  202. long utf8codepoint = 0;
  203. const char *utf8str;
  204. FcCharSet *fccharset;
  205. FcPattern *fcpattern;
  206. FcPattern *match;
  207. XftResult result;
  208. int charexists = 0, overflow = 0;
  209. /* keep track of a couple codepoints for which we have no match. */
  210. static unsigned int nomatches[128], ellipsis_width, invalid_width;
  211. static const char invalid[] = "";
  212. if (!drw || (render && (!drw->scheme || !w)) || !text || !drw->fonts)
  213. return 0;
  214. if (!render) {
  215. w = invert ? invert : ~invert;
  216. } else {
  217. XSetForeground(drw->dpy, drw->gc, drw->scheme[invert ? ColFg : ColBg].pixel);
  218. XFillRectangle(drw->dpy, drw->drawable, drw->gc, x, y, w, h);
  219. d = XftDrawCreate(drw->dpy, drw->drawable,
  220. DefaultVisual(drw->dpy, drw->screen),
  221. DefaultColormap(drw->dpy, drw->screen));
  222. x += lpad;
  223. w -= lpad;
  224. }
  225. usedfont = drw->fonts;
  226. if (!ellipsis_width && render)
  227. ellipsis_width = drw_fontset_getwidth(drw, "...");
  228. if (!invalid_width && render)
  229. invalid_width = drw_fontset_getwidth(drw, invalid);
  230. while (1) {
  231. ew = ellipsis_len = utf8err = utf8charlen = utf8strlen = 0;
  232. utf8str = text;
  233. nextfont = NULL;
  234. while (*text) {
  235. utf8charlen = utf8decode(text, &utf8codepoint, &utf8err);
  236. for (curfont = drw->fonts; curfont; curfont = curfont->next) {
  237. charexists = charexists || XftCharExists(drw->dpy, curfont->xfont, utf8codepoint);
  238. if (charexists) {
  239. drw_font_getexts(curfont, text, utf8charlen, &tmpw, NULL);
  240. if (ew + ellipsis_width <= w) {
  241. /* keep track where the ellipsis still fits */
  242. ellipsis_x = x + ew;
  243. ellipsis_w = w - ew;
  244. ellipsis_len = utf8strlen;
  245. }
  246. if (ew + tmpw > w) {
  247. overflow = 1;
  248. /* called from drw_fontset_getwidth_clamp():
  249. * it wants the width AFTER the overflow
  250. */
  251. if (!render)
  252. x += tmpw;
  253. else
  254. utf8strlen = ellipsis_len;
  255. } else if (curfont == usedfont) {
  256. text += utf8charlen;
  257. utf8strlen += utf8err ? 0 : utf8charlen;
  258. ew += utf8err ? 0 : tmpw;
  259. } else {
  260. nextfont = curfont;
  261. }
  262. break;
  263. }
  264. }
  265. if (overflow || !charexists || nextfont || utf8err)
  266. break;
  267. else
  268. charexists = 0;
  269. }
  270. if (utf8strlen) {
  271. if (render) {
  272. ty = y + (h - usedfont->h) / 2 + usedfont->xfont->ascent;
  273. XftDrawStringUtf8(d, &drw->scheme[invert ? ColBg : ColFg],
  274. usedfont->xfont, x, ty, (XftChar8 *)utf8str, utf8strlen);
  275. }
  276. x += ew;
  277. w -= ew;
  278. }
  279. if (utf8err && (!render || invalid_width < w)) {
  280. if (render)
  281. drw_text(drw, x, y, w, h, 0, invalid, invert);
  282. x += invalid_width;
  283. w -= invalid_width;
  284. }
  285. if (render && overflow)
  286. drw_text(drw, ellipsis_x, y, ellipsis_w, h, 0, "...", invert);
  287. if (!*text || overflow) {
  288. break;
  289. } else if (nextfont) {
  290. charexists = 0;
  291. usedfont = nextfont;
  292. } else {
  293. /* Regardless of whether or not a fallback font is found, the
  294. * character must be drawn. */
  295. charexists = 1;
  296. hash = (unsigned int)utf8codepoint;
  297. hash = ((hash >> 16) ^ hash) * 0x21F0AAAD;
  298. hash = ((hash >> 15) ^ hash) * 0xD35A2D97;
  299. h0 = ((hash >> 15) ^ hash) % LENGTH(nomatches);
  300. h1 = (hash >> 17) % LENGTH(nomatches);
  301. /* avoid expensive XftFontMatch call when we know we won't find a match */
  302. if (nomatches[h0] == utf8codepoint || nomatches[h1] == utf8codepoint)
  303. goto no_match;
  304. fccharset = FcCharSetCreate();
  305. FcCharSetAddChar(fccharset, utf8codepoint);
  306. if (!drw->fonts->pattern) {
  307. /* Refer to the comment in xfont_create for more information. */
  308. die("the first font in the cache must be loaded from a font string.");
  309. }
  310. fcpattern = FcPatternDuplicate(drw->fonts->pattern);
  311. FcPatternAddCharSet(fcpattern, FC_CHARSET, fccharset);
  312. FcPatternAddBool(fcpattern, FC_SCALABLE, FcTrue);
  313. FcConfigSubstitute(NULL, fcpattern, FcMatchPattern);
  314. FcDefaultSubstitute(fcpattern);
  315. match = XftFontMatch(drw->dpy, drw->screen, fcpattern, &result);
  316. FcCharSetDestroy(fccharset);
  317. FcPatternDestroy(fcpattern);
  318. if (match) {
  319. usedfont = xfont_create(drw, NULL, match);
  320. if (usedfont && XftCharExists(drw->dpy, usedfont->xfont, utf8codepoint)) {
  321. for (curfont = drw->fonts; curfont->next; curfont = curfont->next)
  322. ; /* NOP */
  323. curfont->next = usedfont;
  324. } else {
  325. xfont_free(usedfont);
  326. nomatches[nomatches[h0] ? h1 : h0] = utf8codepoint;
  327. no_match:
  328. usedfont = drw->fonts;
  329. }
  330. }
  331. }
  332. }
  333. if (d)
  334. XftDrawDestroy(d);
  335. return x + (render ? w : 0);
  336. }
  337. void
  338. drw_map(Drw *drw, Window win, int x, int y, unsigned int w, unsigned int h)
  339. {
  340. if (!drw)
  341. return;
  342. XCopyArea(drw->dpy, drw->drawable, win, drw->gc, x, y, w, h, x, y);
  343. XSync(drw->dpy, False);
  344. }
  345. unsigned int
  346. drw_fontset_getwidth(Drw *drw, const char *text)
  347. {
  348. if (!drw || !drw->fonts || !text)
  349. return 0;
  350. return drw_text(drw, 0, 0, 0, 0, 0, text, 0);
  351. }
  352. unsigned int
  353. drw_fontset_getwidth_clamp(Drw *drw, const char *text, unsigned int n)
  354. {
  355. unsigned int tmp = 0;
  356. if (drw && drw->fonts && text && n)
  357. tmp = drw_text(drw, 0, 0, 0, 0, 0, text, n);
  358. return MIN(n, tmp);
  359. }
  360. void
  361. drw_font_getexts(Fnt *font, const char *text, unsigned int len, unsigned int *w, unsigned int *h)
  362. {
  363. XGlyphInfo ext;
  364. if (!font || !text)
  365. return;
  366. XftTextExtentsUtf8(font->dpy, font->xfont, (XftChar8 *)text, len, &ext);
  367. if (w)
  368. *w = ext.xOff;
  369. if (h)
  370. *h = font->h;
  371. }
  372. Cur *
  373. drw_cur_create(Drw *drw, int shape)
  374. {
  375. Cur *cur;
  376. if (!drw || !(cur = ecalloc(1, sizeof(Cur))))
  377. return NULL;
  378. cur->cursor = XCreateFontCursor(drw->dpy, shape);
  379. return cur;
  380. }
  381. void
  382. drw_cur_free(Drw *drw, Cur *cursor)
  383. {
  384. if (!cursor)
  385. return;
  386. XFreeCursor(drw->dpy, cursor->cursor);
  387. free(cursor);
  388. }