aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFederico Igne <git@federicoigne.com>2020-05-23 12:01:50 +0100
committerFederico Igne <git@federicoigne.com>2020-05-23 12:01:50 +0100
commit30faefe674c5048b7ffd31c98b1a2193c43c9fe2 (patch)
tree64aad31e5682624aef1c7538b2a9c63079fd5a3a
parent81df9b428bd0330af98db9c69ffe69b0441e00fd (diff)
downloadst-30faefe674c5048b7ffd31c98b1a2193c43c9fe2.tar.gz
st-30faefe674c5048b7ffd31c98b1a2193c43c9fe2.zip
Apply Ligatures patch
Homepage link: https://st.suckless.org/patches/ligatures/ Patch links: https://st.suckless.org/patches/ligatures/0.8.3/st-ligatures-boxdraw-20200430-0.8.3.diff https://st.suckless.org/patches/ligatures/0.8.3/st-ligatures-scrollback-20200430-0.8.3.diff Additional info: the patch was applied by mixing the two patches linked above in order to provide compatibility with both BoxDraw and Scrollback patches.
-rw-r--r--Makefile5
-rw-r--r--README.md2
-rw-r--r--config.mk6
-rw-r--r--hb.c140
-rw-r--r--hb.h6
-rw-r--r--st.c3
-rw-r--r--st.h14
-rw-r--r--win.h2
-rw-r--r--x.c16
9 files changed, 179 insertions, 15 deletions
diff --git a/Makefile b/Makefile
index 8a2e1f9..8fe1520 100644
--- a/Makefile
+++ b/Makefile
@@ -4,7 +4,7 @@
4 4
5include config.mk 5include config.mk
6 6
7SRC = st.c x.c boxdraw.c 7SRC = st.c x.c boxdraw.c hb.c
8OBJ = $(SRC:.c=.o) 8OBJ = $(SRC:.c=.o)
9 9
10all: options st 10all: options st
@@ -22,7 +22,8 @@ config.h:
22 $(CC) $(STCFLAGS) -c $< 22 $(CC) $(STCFLAGS) -c $<
23 23
24st.o: config.h st.h win.h 24st.o: config.h st.h win.h
25x.o: arg.h config.h st.h win.h 25x.o: arg.h config.h st.h win.h hb.h
26hb.o: st.h
26boxdraw.o: config.h st.h boxdraw_data.h 27boxdraw.o: config.h st.h boxdraw_data.h
27 28
28$(OBJ): config.h config.mk 29$(OBJ): config.h config.mk
diff --git a/README.md b/README.md
index 7c658ea..7796f3a 100644
--- a/README.md
+++ b/README.md
@@ -13,6 +13,8 @@ were applied.
13 buffer scrollback. 13 buffer scrollback.
14+ [boxdraw](https://st.suckless.org/patches/boxdraw/) - handles 14+ [boxdraw](https://st.suckless.org/patches/boxdraw/) - handles
15 line/box/braille drawings 15 line/box/braille drawings
16+ [ligatures](https://st.suckless.org/patches/ligatures/) - add ligature
17 support
16 18
17Requirements 19Requirements
18------------ 20------------
diff --git a/config.mk b/config.mk
index 5d202e5..d058753 100644
--- a/config.mk
+++ b/config.mk
@@ -15,10 +15,12 @@ PKG_CONFIG = pkg-config
15# includes and libs 15# includes and libs
16INCS = -I$(X11INC) \ 16INCS = -I$(X11INC) \
17 `$(PKG_CONFIG) --cflags fontconfig` \ 17 `$(PKG_CONFIG) --cflags fontconfig` \
18 `$(PKG_CONFIG) --cflags freetype2` 18 `$(PKG_CONFIG) --cflags freetype2` \
19 `$(PKG_CONFIG) --cflags harfbuzz`
19LIBS = -L$(X11LIB) -lm -lrt -lX11 -lutil -lXft \ 20LIBS = -L$(X11LIB) -lm -lrt -lX11 -lutil -lXft \
20 `$(PKG_CONFIG) --libs fontconfig` \ 21 `$(PKG_CONFIG) --libs fontconfig` \
21 `$(PKG_CONFIG) --libs freetype2` 22 `$(PKG_CONFIG) --libs freetype2` \
23 `$(PKG_CONFIG) --libs harfbuzz`
22 24
23# flags 25# flags
24STCPPFLAGS = -DVERSION=\"$(VERSION)\" -D_XOPEN_SOURCE=600 26STCPPFLAGS = -DVERSION=\"$(VERSION)\" -D_XOPEN_SOURCE=600
diff --git a/hb.c b/hb.c
new file mode 100644
index 0000000..467bcac
--- /dev/null
+++ b/hb.c
@@ -0,0 +1,140 @@
1#include <stdlib.h>
2#include <stdio.h>
3#include <math.h>
4#include <X11/Xft/Xft.h>
5#include <hb.h>
6#include <hb-ft.h>
7
8#include "st.h"
9
10void hbtransformsegment(XftFont *xfont, const Glyph *string, hb_codepoint_t *codepoints, int start, int length);
11hb_font_t *hbfindfont(XftFont *match);
12
13typedef struct {
14 XftFont *match;
15 hb_font_t *font;
16} HbFontMatch;
17
18static int hbfontslen = 0;
19static HbFontMatch *hbfontcache = NULL;
20
21void
22hbunloadfonts()
23{
24 for (int i = 0; i < hbfontslen; i++) {
25 hb_font_destroy(hbfontcache[i].font);
26 XftUnlockFace(hbfontcache[i].match);
27 }
28
29 if (hbfontcache != NULL) {
30 free(hbfontcache);
31 hbfontcache = NULL;
32 }
33 hbfontslen = 0;
34}
35
36hb_font_t *
37hbfindfont(XftFont *match)
38{
39 for (int i = 0; i < hbfontslen; i++) {
40 if (hbfontcache[i].match == match)
41 return hbfontcache[i].font;
42 }
43
44 /* Font not found in cache, caching it now. */
45 hbfontcache = realloc(hbfontcache, sizeof(HbFontMatch) * (hbfontslen + 1));
46 FT_Face face = XftLockFace(match);
47 hb_font_t *font = hb_ft_font_create(face, NULL);
48 if (font == NULL)
49 die("Failed to load Harfbuzz font.");
50
51 hbfontcache[hbfontslen].match = match;
52 hbfontcache[hbfontslen].font = font;
53 hbfontslen += 1;
54
55 return font;
56}
57
58void
59hbtransform(XftGlyphFontSpec *specs, const Glyph *glyphs, size_t len, int x, int y)
60{
61 int start = 0, length = 1, gstart = 0;
62 hb_codepoint_t *codepoints = calloc(len, sizeof(hb_codepoint_t));
63
64 for (int idx = 1, specidx = 1; idx < len; idx++) {
65 if (glyphs[idx].mode & ATTR_WDUMMY) {
66 length += 1;
67 continue;
68 }
69
70 if (specs[specidx].font != specs[start].font || ATTRCMP(glyphs[gstart], glyphs[idx]) || selected(x + idx, y) != selected(x + gstart, y)) {
71 hbtransformsegment(specs[start].font, glyphs, codepoints, gstart, length);
72
73 /* Reset the sequence. */
74 length = 1;
75 start = specidx;
76 gstart = idx;
77 } else {
78 length += 1;
79 }
80
81 specidx++;
82 }
83
84 /* EOL. */
85 hbtransformsegment(specs[start].font, glyphs, codepoints, gstart, length);
86
87 /* Apply the transformation to glyph specs. */
88 for (int i = 0, specidx = 0; i < len; i++) {
89 if (glyphs[i].mode & ATTR_WDUMMY)
90 continue;
91 if (glyphs[i].mode & ATTR_BOXDRAW) {
92 specidx++;
93 continue;
94 }
95
96 if (codepoints[i] != specs[specidx].glyph)
97 ((Glyph *)glyphs)[i].mode |= ATTR_LIGA;
98
99 specs[specidx++].glyph = codepoints[i];
100 }
101
102 free(codepoints);
103}
104
105void
106hbtransformsegment(XftFont *xfont, const Glyph *string, hb_codepoint_t *codepoints, int start, int length)
107{
108 hb_font_t *font = hbfindfont(xfont);
109 if (font == NULL)
110 return;
111
112 Rune rune;
113 ushort mode = USHRT_MAX;
114 hb_buffer_t *buffer = hb_buffer_create();
115 hb_buffer_set_direction(buffer, HB_DIRECTION_LTR);
116
117 /* Fill buffer with codepoints. */
118 for (int i = start; i < (start+length); i++) {
119 rune = string[i].u;
120 mode = string[i].mode;
121 if (mode & ATTR_WDUMMY)
122 rune = 0x0020;
123 hb_buffer_add_codepoints(buffer, &rune, 1, 0, 1);
124 }
125
126 /* Shape the segment. */
127 hb_shape(font, buffer, NULL, 0);
128
129 /* Get new glyph info. */
130 hb_glyph_info_t *info = hb_buffer_get_glyph_infos(buffer, NULL);
131
132 /* Write new codepoints. */
133 for (int i = 0; i < length; i++) {
134 hb_codepoint_t gid = info[i].codepoint;
135 codepoints[start+i] = gid;
136 }
137
138 /* Cleanup. */
139 hb_buffer_destroy(buffer);
140}
diff --git a/hb.h b/hb.h
new file mode 100644
index 0000000..07888df
--- /dev/null
+++ b/hb.h
@@ -0,0 +1,6 @@
1#include <X11/Xft/Xft.h>
2#include <hb.h>
3#include <hb-ft.h>
4
5void hbunloadfonts();
6void hbtransform(XftGlyphFontSpec *, const Glyph *, size_t, int, int);
diff --git a/st.c b/st.c
index 1a3f49c..4a6c2c9 100644
--- a/st.c
+++ b/st.c
@@ -2665,7 +2665,8 @@ draw(void)
2665 drawregion(0, 0, term.col, term.row); 2665 drawregion(0, 0, term.col, term.row);
2666 if (term.scr == 0) 2666 if (term.scr == 0)
2667 xdrawcursor(cx, term.c.y, term.line[term.c.y][cx], 2667 xdrawcursor(cx, term.c.y, term.line[term.c.y][cx],
2668 term.ocx, term.ocy, term.line[term.ocy][term.ocx]); 2668 term.ocx, term.ocy, term.line[term.ocy][term.ocx],
2669 term.line[term.ocy], term.col);
2669 term.ocx = cx, term.ocy = term.c.y; 2670 term.ocx = cx, term.ocy = term.c.y;
2670 xfinishdraw(); 2671 xfinishdraw();
2671 xximspot(term.ocx, term.ocy); 2672 xximspot(term.ocx, term.ocy);
diff --git a/st.h b/st.h
index e9b3abe..70ebea0 100644
--- a/st.h
+++ b/st.h
@@ -4,17 +4,18 @@
4#include <sys/types.h> 4#include <sys/types.h>
5 5
6/* macros */ 6/* macros */
7#define MIN(a, b) ((a) < (b) ? (a) : (b)) 7#define MIN(a, b) ((a) < (b) ? (a) : (b))
8#define MAX(a, b) ((a) < (b) ? (b) : (a)) 8#define MAX(a, b) ((a) < (b) ? (b) : (a))
9#define LEN(a) (sizeof(a) / sizeof(a)[0]) 9#define LEN(a) (sizeof(a) / sizeof(a)[0])
10#define BETWEEN(x, a, b) ((a) <= (x) && (x) <= (b)) 10#define BETWEEN(x, a, b) ((a) <= (x) && (x) <= (b))
11#define DIVCEIL(n, d) (((n) + ((d) - 1)) / (d)) 11#define DIVCEIL(n, d) (((n) + ((d) - 1)) / (d))
12#define DEFAULT(a, b) (a) = (a) ? (a) : (b) 12#define DEFAULT(a, b) (a) = (a) ? (a) : (b)
13#define LIMIT(x, a, b) (x) = (x) < (a) ? (a) : (x) > (b) ? (b) : (x) 13#define LIMIT(x, a, b) (x) = (x) < (a) ? (a) : (x) > (b) ? (b) : (x)
14#define ATTRCMP(a, b) ((a).mode != (b).mode || (a).fg != (b).fg || \ 14#define ATTRCMP(a, b) (((a).mode & (~ATTR_WRAP) & (~ATTR_LIGA)) != ((b).mode & (~ATTR_WRAP) & (~ATTR_LIGA)) || \
15 (a).bg != (b).bg) 15 (a).fg != (b).fg || \
16 (a).bg != (b).bg)
16#define TIMEDIFF(t1, t2) ((t1.tv_sec-t2.tv_sec)*1000 + \ 17#define TIMEDIFF(t1, t2) ((t1.tv_sec-t2.tv_sec)*1000 + \
17 (t1.tv_nsec-t2.tv_nsec)/1E6) 18 (t1.tv_nsec-t2.tv_nsec)/1E6)
18#define MODBIT(x, set, bit) ((set) ? ((x) |= (bit)) : ((x) &= ~(bit))) 19#define MODBIT(x, set, bit) ((set) ? ((x) |= (bit)) : ((x) &= ~(bit)))
19 20
20#define TRUECOLOR(r,g,b) (1 << 24 | (r) << 16 | (g) << 8 | (b)) 21#define TRUECOLOR(r,g,b) (1 << 24 | (r) << 16 | (g) << 8 | (b))
@@ -34,6 +35,7 @@ enum glyph_attribute {
34 ATTR_WIDE = 1 << 9, 35 ATTR_WIDE = 1 << 9,
35 ATTR_WDUMMY = 1 << 10, 36 ATTR_WDUMMY = 1 << 10,
36 ATTR_BOXDRAW = 1 << 11, 37 ATTR_BOXDRAW = 1 << 11,
38 ATTR_LIGA = 1 << 12,
37 ATTR_BOLD_FAINT = ATTR_BOLD | ATTR_FAINT, 39 ATTR_BOLD_FAINT = ATTR_BOLD | ATTR_FAINT,
38}; 40};
39 41
diff --git a/win.h b/win.h
index a6ef1b9..bc0d180 100644
--- a/win.h
+++ b/win.h
@@ -25,7 +25,7 @@ enum win_mode {
25 25
26void xbell(void); 26void xbell(void);
27void xclipcopy(void); 27void xclipcopy(void);
28void xdrawcursor(int, int, Glyph, int, int, Glyph); 28void xdrawcursor(int, int, Glyph, int, int, Glyph, Line, int);
29void xdrawline(Line, int, int, int); 29void xdrawline(Line, int, int, int);
30void xfinishdraw(void); 30void xfinishdraw(void);
31void xloadcols(void); 31void xloadcols(void);
diff --git a/x.c b/x.c
index 58c2d98..0f95d60 100644
--- a/x.c
+++ b/x.c
@@ -19,6 +19,7 @@ static char *argv0;
19#include "arg.h" 19#include "arg.h"
20#include "st.h" 20#include "st.h"
21#include "win.h" 21#include "win.h"
22#include "hb.h"
22 23
23/* types used in config.h */ 24/* types used in config.h */
24typedef struct { 25typedef struct {
@@ -1117,6 +1118,9 @@ xunloadfont(Font *f)
1117void 1118void
1118xunloadfonts(void) 1119xunloadfonts(void)
1119{ 1120{
1121 /* Clear Harfbuzz font cache. */
1122 hbunloadfonts();
1123
1120 /* Free the loaded fonts in the font cache. */ 1124 /* Free the loaded fonts in the font cache. */
1121 while (frclen > 0) 1125 while (frclen > 0)
1122 XftFontClose(xw.dpy, frc[--frclen].font); 1126 XftFontClose(xw.dpy, frc[--frclen].font);
@@ -1320,7 +1324,7 @@ xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x
1320 mode = glyphs[i].mode; 1324 mode = glyphs[i].mode;
1321 1325
1322 /* Skip dummy wide-character spacing. */ 1326 /* Skip dummy wide-character spacing. */
1323 if (mode == ATTR_WDUMMY) 1327 if (mode & ATTR_WDUMMY)
1324 continue; 1328 continue;
1325 1329
1326 /* Determine font for glyph if different from previous glyph. */ 1330 /* Determine font for glyph if different from previous glyph. */
@@ -1432,6 +1436,9 @@ xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x
1432 numspecs++; 1436 numspecs++;
1433 } 1437 }
1434 1438
1439 /* Harfbuzz transformation for ligatures. */
1440 hbtransform(specs, glyphs, len, x, y);
1441
1435 return numspecs; 1442 return numspecs;
1436} 1443}
1437 1444
@@ -1585,14 +1592,17 @@ xdrawglyph(Glyph g, int x, int y)
1585} 1592}
1586 1593
1587void 1594void
1588xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og) 1595xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og, Line line, int len)
1589{ 1596{
1590 Color drawcol; 1597 Color drawcol;
1591 1598
1592 /* remove the old cursor */ 1599 /* remove the old cursor */
1593 if (selected(ox, oy)) 1600 if (selected(ox, oy))
1594 og.mode ^= ATTR_REVERSE; 1601 og.mode ^= ATTR_REVERSE;
1595 xdrawglyph(og, ox, oy); 1602
1603 /* Redraw the line where cursor was previously.
1604 * It will restore the ligatures broken by the cursor. */
1605 xdrawline(line, 0, oy, len);
1596 1606
1597 if (IS_SET(MODE_HIDE)) 1607 if (IS_SET(MODE_HIDE))
1598 return; 1608 return;