st-copyurl-multiline-20220221-0.8.5.diff (4012B)
1 From 30a04d9ecb3998953bdbe42e5617d00d6002869b Mon Sep 17 00:00:00 2001 2 From: Santtu Lakkala <inz@inz.fi> 3 Date: Wed, 16 Feb 2022 20:34:20 +0200 4 Subject: [PATCH] Loop through urls on screen and copy to clipboard 5 6 Replace url detection heuristics with a DFA, enabling urls that span 7 multiple lines. Also fix the selection not to use snapping so that urls 8 are selected exactly. 9 --- 10 config.def.h | 1 + 11 st.c | 93 ++++++++++++++++++++++++++++++++++++++++++++++++++++ 12 st.h | 1 + 13 3 files changed, 95 insertions(+) 14 15 diff --git a/config.def.h b/config.def.h 16 index 91ab8ca..3f365c7 100644 17 --- a/config.def.h 18 +++ b/config.def.h 19 @@ -201,6 +201,7 @@ static Shortcut shortcuts[] = { 20 { TERMMOD, XK_Y, selpaste, {.i = 0} }, 21 { ShiftMask, XK_Insert, selpaste, {.i = 0} }, 22 { TERMMOD, XK_Num_Lock, numlock, {.i = 0} }, 23 + { MODKEY, XK_l, copyurl, {.i = 0} }, 24 }; 25 26 /* 27 diff --git a/st.c b/st.c 28 index 51049ba..5b6d919 100644 29 --- a/st.c 30 +++ b/st.c 31 @@ -152,6 +152,11 @@ typedef struct { 32 int narg; /* nb of args */ 33 } STREscape; 34 35 +typedef struct { 36 + int state; 37 + size_t length; 38 +} URLdfa; 39 + 40 static void execsh(char *, char **); 41 static void stty(char **); 42 static void sigchld(int); 43 @@ -200,6 +205,7 @@ static void tdefutf8(char); 44 static int32_t tdefcolor(const int *, int *, int); 45 static void tdeftran(char); 46 static void tstrsequence(uchar); 47 +static int daddch(URLdfa *, char); 48 49 static void drawregion(int, int, int, int); 50 51 @@ -2688,3 +2694,90 @@ redraw(void) 52 tfulldirt(); 53 draw(); 54 } 55 + 56 +int 57 +daddch(URLdfa *dfa, char c) 58 +{ 59 + /* () and [] can appear in urls, but excluding them here will reduce false 60 + * positives when figuring out where a given url ends. 61 + */ 62 + static const char URLCHARS[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ" 63 + "abcdefghijklmnopqrstuvwxyz" 64 + "0123456789-._~:/?#@!$&'*+,;=%"; 65 + static const char RPFX[] = "//:sptth"; 66 + 67 + if (!strchr(URLCHARS, c)) { 68 + dfa->length = 0; 69 + dfa->state = 0; 70 + 71 + return 0; 72 + } 73 + 74 + dfa->length++; 75 + 76 + if (dfa->state == 2 && c == '/') { 77 + dfa->state = 0; 78 + } else if (dfa->state == 3 && c == 'p') { 79 + dfa->state++; 80 + } else if (c != RPFX[dfa->state]) { 81 + dfa->state = 0; 82 + return 0; 83 + } 84 + 85 + if (dfa->state++ == 7) { 86 + dfa->state = 0; 87 + return 1; 88 + } 89 + 90 + return 0; 91 +} 92 + 93 +/* 94 +** Select and copy the previous url on screen (do nothing if there's no url). 95 +*/ 96 +void 97 +copyurl(const Arg *arg) { 98 + int row = 0, /* row of current URL */ 99 + col = 0, /* column of current URL start */ 100 + colend = 0, /* column of last occurrence */ 101 + passes = 0; /* how many rows have been scanned */ 102 + 103 + const char *c = NULL, 104 + *match = NULL; 105 + URLdfa dfa = { 0 }; 106 + 107 + row = (sel.ob.x >= 0 && sel.nb.y > 0) ? sel.nb.y : term.bot; 108 + LIMIT(row, term.top, term.bot); 109 + 110 + colend = (sel.ob.x >= 0 && sel.nb.y > 0) ? sel.nb.x : term.col; 111 + LIMIT(colend, 0, term.col); 112 + 113 + /* 114 + ** Scan from (term.row - 1,term.col - 1) to (0,0) and find 115 + ** next occurrance of a URL 116 + */ 117 + for (passes = 0; passes < term.row; passes++) { 118 + /* Read in each column of every row until 119 + ** we hit previous occurrence of URL 120 + */ 121 + for (col = colend; col--;) 122 + if (daddch(&dfa, term.line[row][col].u < 128 ? term.line[row][col].u : ' ')) 123 + break; 124 + 125 + if (col >= 0) 126 + break; 127 + 128 + if (--row < 0) 129 + row = term.row - 1; 130 + 131 + colend = term.col; 132 + } 133 + 134 + if (passes < term.row) { 135 + selstart(col, row, 0); 136 + selextend((col + dfa.length - 1) % term.col, row + (col + dfa.length - 1) / term.col, SEL_REGULAR, 0); 137 + selextend((col + dfa.length - 1) % term.col, row + (col + dfa.length - 1) / term.col, SEL_REGULAR, 1); 138 + xsetsel(getsel()); 139 + xclipcopy(); 140 + } 141 +} 142 diff --git a/st.h b/st.h 143 index 519b9bd..0458005 100644 144 --- a/st.h 145 +++ b/st.h 146 @@ -85,6 +85,7 @@ void printscreen(const Arg *); 147 void printsel(const Arg *); 148 void sendbreak(const Arg *); 149 void toggleprinter(const Arg *); 150 +void copyurl(const Arg *); 151 152 int tattrset(int); 153 void tnew(int, int); 154 -- 155 2.32.0 156