Go.dev Code Highlight

Golang code highlighter

当前为 2023-12-19 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Go.dev Code Highlight
  3. // @version 1.1.2
  4. // @description Golang code highlighter
  5. // @author Anton Vi
  6. // @match https://*.go.dev/*
  7. // @resource IMPORTED_CSS https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism-tomorrow.min.css
  8. // @grant GM_getResourceText
  9. // @grant GM_addStyle
  10. // @license GNU GPLv3
  11. // @namespace GolangDevNamespace
  12. // ==/UserScript==
  13.  
  14. (function() {
  15.  
  16. 'use strict';
  17.  
  18. resetBlogCss();
  19. start();
  20.  
  21. })();
  22.  
  23. function resetBlogCss() {
  24. // reset styles for go dev blog
  25. let css = `#blog pre, #content .code, #content .playground {
  26. background: black;
  27. font-family: Consolas,Monaco,'Andale Mono','Ubuntu Mono',monospace;
  28. }`;
  29. GM_addStyle(css);
  30.  
  31. }
  32.  
  33. function start() {
  34. // create Prisma object
  35. window.Prism = window.Prism || {};
  36. // no highlight on auto
  37. Prism.manual = true;
  38.  
  39. // add highlight styles css
  40. const my_css = GM_getResourceText("IMPORTED_CSS");
  41. console.log(my_css);
  42. GM_addStyle(my_css);
  43.  
  44. // start prism
  45. initPrismCore();
  46.  
  47. // add languages syntax
  48. addClikeSyntax();
  49. addGoSyntax();
  50.  
  51.  
  52. // loop pre-s and add code child elements for highlight
  53. let pre_s = document.getElementsByTagName('pre');
  54. for (let el of pre_s) {
  55. let text = el.innerHTML;
  56.  
  57. el.innerHTML = '';
  58.  
  59. let code = document.createElement('code');
  60. code.className = 'language-go';
  61. code.innerHTML = text;
  62. el.appendChild(code);
  63. }
  64.  
  65. // start highlight syntax
  66. Prism.highlightAll();
  67.  
  68. }
  69.  
  70. function initPrismCore() {
  71. var _self = "undefined" != typeof window ? window : "undefined" != typeof WorkerGlobalScope && self instanceof WorkerGlobalScope ? self : {},
  72. Prism = function(e) {
  73. var n = /(?:^|\s)lang(?:uage)?-([\w-]+)(?=\s|$)/i,
  74. t = 0,
  75. r = {},
  76. a = {
  77. manual: e.Prism && e.Prism.manual,
  78. disableWorkerMessageHandler: e.Prism && e.Prism.disableWorkerMessageHandler,
  79. util: {
  80. encode: function e(n) { return n instanceof i ? new i(n.type, e(n.content), n.alias) : Array.isArray(n) ? n.map(e) : n.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/\u00a0/g, " ") },
  81. type: function(e) { return Object.prototype.toString.call(e).slice(8, -1) },
  82. objId: function(e) { return e.__id || Object.defineProperty(e, "__id", { value: ++t }), e.__id },
  83. clone: function e(n, t) {
  84. var r, i;
  85. switch (t = t || {}, a.util.type(n)) {
  86. case "Object":
  87. if (i = a.util.objId(n), t[i]) return t[i];
  88. for (var l in r = {}, t[i] = r, n) n.hasOwnProperty(l) && (r[l] = e(n[l], t));
  89. return r;
  90. case "Array":
  91. return i = a.util.objId(n), t[i] ? t[i] : (r = [], t[i] = r, n.forEach((function(n, a) { r[a] = e(n, t) })), r);
  92. default:
  93. return n
  94. }
  95. },
  96. getLanguage: function(e) {
  97. for (; e;) {
  98. var t = n.exec(e.className);
  99. if (t) return t[1].toLowerCase();
  100. e = e.parentElement
  101. }
  102. return "none"
  103. },
  104. setLanguage: function(e, t) { e.className = e.className.replace(RegExp(n, "gi"), ""), e.classList.add("language-" + t) },
  105. currentScript: function() {
  106. if ("undefined" == typeof document) return null;
  107. if ("currentScript" in document) return document.currentScript;
  108. try { throw new Error } catch (r) {
  109. var e = (/at [^(\r\n]*\((.*):[^:]+:[^:]+\)$/i.exec(r.stack) || [])[1];
  110. if (e) {
  111. var n = document.getElementsByTagName("script");
  112. for (var t in n)
  113. if (n[t].src == e) return n[t]
  114. }
  115. return null
  116. }
  117. },
  118. isActive: function(e, n, t) {
  119. for (var r = "no-" + n; e;) {
  120. var a = e.classList;
  121. if (a.contains(n)) return !0;
  122. if (a.contains(r)) return !1;
  123. e = e.parentElement
  124. }
  125. return !!t
  126. }
  127. },
  128. languages: {
  129. plain: r,
  130. plaintext: r,
  131. text: r,
  132. txt: r,
  133. extend: function(e, n) { var t = a.util.clone(a.languages[e]); for (var r in n) t[r] = n[r]; return t },
  134. insertBefore: function(e, n, t, r) {
  135. var i = (r = r || a.languages)[e],
  136. l = {};
  137. for (var o in i)
  138. if (i.hasOwnProperty(o)) {
  139. if (o == n)
  140. for (var s in t) t.hasOwnProperty(s) && (l[s] = t[s]);
  141. t.hasOwnProperty(o) || (l[o] = i[o])
  142. } var u = r[e];
  143. return r[e] = l, a.languages.DFS(a.languages, (function(n, t) { t === u && n != e && (this[n] = l) })), l
  144. },
  145. DFS: function e(n, t, r, i) {
  146. i = i || {};
  147. var l = a.util.objId;
  148. for (var o in n)
  149. if (n.hasOwnProperty(o)) {
  150. t.call(n, o, n[o], r || o);
  151. var s = n[o],
  152. u = a.util.type(s);
  153. "Object" !== u || i[l(s)] ? "Array" !== u || i[l(s)] || (i[l(s)] = !0, e(s, t, o, i)) : (i[l(s)] = !0, e(s, t, null, i))
  154. }
  155. }
  156. },
  157. plugins: {},
  158. highlightAll: function(e, n) { a.highlightAllUnder(document, e, n) },
  159. highlightAllUnder: function(e, n, t) {
  160. var r = { callback: t, container: e, selector: 'code[class*="language-"], [class*="language-"] code, code[class*="lang-"], [class*="lang-"] code' };
  161. a.hooks.run("before-highlightall", r), r.elements = Array.prototype.slice.apply(r.container.querySelectorAll(r.selector)), a.hooks.run("before-all-elements-highlight", r);
  162. for (var i, l = 0; i = r.elements[l++];) a.highlightElement(i, !0 === n, r.callback)
  163. },
  164. highlightElement: function(n, t, r) {
  165. var i = a.util.getLanguage(n),
  166. l = a.languages[i];
  167. a.util.setLanguage(n, i);
  168. var o = n.parentElement;
  169. o && "pre" === o.nodeName.toLowerCase() && a.util.setLanguage(o, i);
  170. var s = { element: n, language: i, grammar: l, code: n.textContent };
  171.  
  172. function u(e) { s.highlightedCode = e, a.hooks.run("before-insert", s), s.element.innerHTML = s.highlightedCode, a.hooks.run("after-highlight", s), a.hooks.run("complete", s), r && r.call(s.element) }
  173. if (a.hooks.run("before-sanity-check", s), (o = s.element.parentElement) && "pre" === o.nodeName.toLowerCase() && !o.hasAttribute("tabindex") && o.setAttribute("tabindex", "0"), !s.code) return a.hooks.run("complete", s), void(r && r.call(s.element));
  174. if (a.hooks.run("before-highlight", s), s.grammar)
  175. if (t && e.Worker) {
  176. var c = new Worker(a.filename);
  177. c.onmessage = function(e) { u(e.data) }, c.postMessage(JSON.stringify({ language: s.language, code: s.code, immediateClose: !0 }))
  178. } else u(a.highlight(s.code, s.grammar, s.language));
  179. else u(a.util.encode(s.code))
  180. },
  181. highlight: function(e, n, t) { var r = { code: e, grammar: n, language: t }; if (a.hooks.run("before-tokenize", r), !r.grammar) throw new Error('The language "' + r.language + '" has no grammar.'); return r.tokens = a.tokenize(r.code, r.grammar), a.hooks.run("after-tokenize", r), i.stringify(a.util.encode(r.tokens), r.language) },
  182. tokenize: function(e, n) {
  183. var t = n.rest;
  184. if (t) {
  185. for (var r in t) n[r] = t[r];
  186. delete n.rest
  187. }
  188. var a = new s;
  189. return u(a, a.head, e), o(e, a, n, a.head, 0),
  190. function(e) { for (var n = [], t = e.head.next; t !== e.tail;) n.push(t.value), t = t.next; return n }(a)
  191. },
  192. hooks: {
  193. all: {},
  194. add: function(e, n) {
  195. var t = a.hooks.all;
  196. t[e] = t[e] || [], t[e].push(n)
  197. },
  198. run: function(e, n) {
  199. var t = a.hooks.all[e];
  200. if (t && t.length)
  201. for (var r, i = 0; r = t[i++];) r(n)
  202. }
  203. },
  204. Token: i
  205. };
  206.  
  207. function i(e, n, t, r) { this.type = e, this.content = n, this.alias = t, this.length = 0 | (r || "").length }
  208.  
  209. function l(e, n, t, r) {
  210. e.lastIndex = n;
  211. var a = e.exec(t);
  212. if (a && r && a[1]) {
  213. var i = a[1].length;
  214. a.index += i, a[0] = a[0].slice(i)
  215. }
  216. return a
  217. }
  218.  
  219. function o(e, n, t, r, s, g) {
  220. for (var f in t)
  221. if (t.hasOwnProperty(f) && t[f]) {
  222. var h = t[f];
  223. h = Array.isArray(h) ? h : [h];
  224. for (var d = 0; d < h.length; ++d) {
  225. if (g && g.cause == f + "," + d) return;
  226. var v = h[d],
  227. p = v.inside,
  228. m = !!v.lookbehind,
  229. y = !!v.greedy,
  230. k = v.alias;
  231. if (y && !v.pattern.global) {
  232. var x = v.pattern.toString().match(/[imsuy]*$/)[0];
  233. v.pattern = RegExp(v.pattern.source, x + "g")
  234. }
  235. for (var b = v.pattern || v, w = r.next, A = s; w !== n.tail && !(g && A >= g.reach); A += w.value.length, w = w.next) {
  236. var E = w.value;
  237. if (n.length > e.length) return;
  238. if (!(E instanceof i)) {
  239. var P, L = 1;
  240. if (y) {
  241. if (!(P = l(b, A, e, m)) || P.index >= e.length) break;
  242. var S = P.index,
  243. O = P.index + P[0].length,
  244. j = A;
  245. for (j += w.value.length; S >= j;) j += (w = w.next).value.length;
  246. if (A = j -= w.value.length, w.value instanceof i) continue;
  247. for (var C = w; C !== n.tail && (j < O || "string" == typeof C.value); C = C.next) L++, j += C.value.length;
  248. L--, E = e.slice(A, j), P.index -= A
  249. } else if (!(P = l(b, 0, E, m))) continue;
  250. S = P.index;
  251. var N = P[0],
  252. _ = E.slice(0, S),
  253. M = E.slice(S + N.length),
  254. W = A + E.length;
  255. g && W > g.reach && (g.reach = W);
  256. var z = w.prev;
  257. if (_ && (z = u(n, z, _), A += _.length), c(n, z, L), w = u(n, z, new i(f, p ? a.tokenize(N, p) : N, k, N)), M && u(n, w, M), L > 1) {
  258. var I = { cause: f + "," + d, reach: W };
  259. o(e, n, t, w.prev, A, I), g && I.reach > g.reach && (g.reach = I.reach)
  260. }
  261. }
  262. }
  263. }
  264. }
  265. }
  266.  
  267. function s() {
  268. var e = { value: null, prev: null, next: null },
  269. n = { value: null, prev: e, next: null };
  270. e.next = n, this.head = e, this.tail = n, this.length = 0
  271. }
  272.  
  273. function u(e, n, t) {
  274. var r = n.next,
  275. a = { value: t, prev: n, next: r };
  276. return n.next = a, r.prev = a, e.length++, a
  277. }
  278.  
  279. function c(e, n, t) {
  280. for (var r = n.next, a = 0; a < t && r !== e.tail; a++) r = r.next;
  281. n.next = r, r.prev = n, e.length -= a
  282. }
  283. if (e.Prism = a, i.stringify = function e(n, t) {
  284. if ("string" == typeof n) return n;
  285. if (Array.isArray(n)) { var r = ""; return n.forEach((function(n) { r += e(n, t) })), r }
  286. var i = { type: n.type, content: e(n.content, t), tag: "span", classes: ["token", n.type], attributes: {}, language: t },
  287. l = n.alias;
  288. l && (Array.isArray(l) ? Array.prototype.push.apply(i.classes, l) : i.classes.push(l)), a.hooks.run("wrap", i);
  289. var o = "";
  290. for (var s in i.attributes) o += " " + s + '="' + (i.attributes[s] || "").replace(/"/g, "&quot;") + '"';
  291. return "<" + i.tag + ' class="' + i.classes.join(" ") + '"' + o + ">" + i.content + "</" + i.tag + ">"
  292. }, !e.document) return e.addEventListener ? (a.disableWorkerMessageHandler || e.addEventListener("message", (function(n) {
  293. var t = JSON.parse(n.data),
  294. r = t.language,
  295. i = t.code,
  296. l = t.immediateClose;
  297. e.postMessage(a.highlight(i, a.languages[r], r)), l && e.close()
  298. }), !1), a) : a;
  299. var g = a.util.currentScript();
  300.  
  301. function f() { a.manual || a.highlightAll() }
  302. if (g && (a.filename = g.src, g.hasAttribute("data-manual") && (a.manual = !0)), !a.manual) { var h = document.readyState; "loading" === h || "interactive" === h && g && g.defer ? document.addEventListener("DOMContentLoaded", f) : window.requestAnimationFrame ? window.requestAnimationFrame(f) : window.setTimeout(f, 16) }
  303. return a
  304. }(_self);
  305. "undefined" != typeof module && module.exports && (module.exports = Prism), "undefined" != typeof global && (global.Prism = Prism);
  306. }
  307.  
  308. function addClikeSyntax() {
  309. Prism.languages.clike = {
  310. comment: [{ pattern: /(^|[^\\])\/\*[\s\S]*?(?:\*\/|$)/, lookbehind: !0, greedy: !0 },
  311. { pattern: /(^|[^\\:])\/\/.*/, lookbehind: !0, greedy: !0 }
  312. ],
  313. string: { pattern: /(["'])(?:\\(?:\r\n|[\s\S])|(?!\1)[^\\\r\n])*\1/, greedy: !0 },
  314. "class-name": { pattern: /(\b(?:class|extends|implements|instanceof|interface|new|trait)\s+|\bcatch\s+\()[\w.\\]+/i, lookbehind: !0, inside: { punctuation: /[.\\]/ } },
  315. keyword: /\b(?:break|catch|continue|do|else|finally|for|function|if|in|instanceof|new|null|return|throw|try|while)\b/,
  316. boolean: /\b(?:false|true)\b/,
  317. function: /\b\w+(?=\()/,
  318. number: /\b0x[\da-f]+\b|(?:\b\d+(?:\.\d*)?|\B\.\d+)(?:e[+-]?\d+)?/i,
  319. operator: /[<>]=?|[!=]=?=?|--?|\+\+?|&&?|\|\|?|[?*/~^%]/,
  320. punctuation: /[{}[\];(),.:]/
  321. };
  322. }
  323.  
  324. function addGoSyntax() {
  325. Prism.languages.go = Prism.languages
  326. .extend("clike", {
  327. string: { pattern: /(^|[^\\])"(?:\\.|[^"\\\r\n])*"|`[^`]*`/, lookbehind: !0, greedy: !0 },
  328. keyword: /\b(?:break|case|chan|const|continue|default|defer|else|fallthrough|for|func|go(?:to)?|if|import|interface|map|package|range|return|select|struct|switch|type|var)\b/,
  329. boolean: /\b(?:_|false|iota|nil|true)\b/,
  330. number: [/\b0(?:b[01_]+|o[0-7_]+)i?\b/i, /\b0x(?:[a-f\d_]+(?:\.[a-f\d_]*)?|\.[a-f\d_]+)(?:p[+-]?\d+(?:_\d+)*)?i?(?!\w)/i, /(?:\b\d[\d_]*(?:\.[\d_]*)?|\B\.\d[\d_]*)(?:e[+-]?[\d_]+)?i?(?!\w)/i],
  331. operator: /[*\/%^!=]=?|\+[=+]?|-[=-]?|\|[=|]?|&(?:=|&|\^=?)?|>(?:>=?|=)?|<(?:<=?|=|-)?|:=|\.\.\./,
  332. builtin: /\b(?:append|bool|byte|cap|close|complex|complex(?:64|128)|copy|delete|error|float(?:32|64)|u?int(?:8|16|32|64)?|imag|len|make|new|panic|print(?:ln)?|real|recover|rune|string|uintptr)\b/
  333. }),
  334. Prism.languages.insertBefore("go", "string", { char: { pattern: /'(?:\\.|[^'\\\r\n]){0,10}'/, greedy: !0 } }), delete Prism.languages.go["class-name"];
  335. }