Go.dev Code Highlight

Golang code highlighter

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

  1. // ==UserScript==
  2. // @name Go.dev Code Highlight
  3. // @version 1.1.4
  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. font-size: 1rem;
  29. }`;
  30. GM_addStyle(css);
  31.  
  32. }
  33.  
  34. function start() {
  35. // create Prisma object
  36. window.Prism = window.Prism || {};
  37. // no highlight on auto
  38. Prism.manual = true;
  39.  
  40. // add highlight styles css
  41. const my_css = GM_getResourceText("IMPORTED_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. // skip pre element for ==Output== code run in documentation
  56. if (el.getElementsByClassName('Documentation-exampleOutputLabel').length > 0) {
  57. continue;
  58. }
  59.  
  60.  
  61. if (!el.getElementsByTagName('code').length > 0) {
  62. let text = el.innerHTML;
  63. el.innerHTML = '';
  64.  
  65. let code = document.createElement('code');
  66. code.innerHTML = text;
  67. code.className = 'language-go';
  68. el.appendChild(code);
  69. }
  70.  
  71. }
  72.  
  73. // start highlight syntax
  74. Prism.highlightAll();
  75.  
  76. }
  77.  
  78. function initPrismCore() {
  79. var _self = "undefined" != typeof window ? window : "undefined" != typeof WorkerGlobalScope && self instanceof WorkerGlobalScope ? self : {},
  80. Prism = function(e) {
  81. var n = /(?:^|\s)lang(?:uage)?-([\w-]+)(?=\s|$)/i,
  82. t = 0,
  83. r = {},
  84. a = {
  85. manual: e.Prism && e.Prism.manual,
  86. disableWorkerMessageHandler: e.Prism && e.Prism.disableWorkerMessageHandler,
  87. util: {
  88. 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, " ") },
  89. type: function(e) { return Object.prototype.toString.call(e).slice(8, -1) },
  90. objId: function(e) { return e.__id || Object.defineProperty(e, "__id", { value: ++t }), e.__id },
  91. clone: function e(n, t) {
  92. var r, i;
  93. switch (t = t || {}, a.util.type(n)) {
  94. case "Object":
  95. if (i = a.util.objId(n), t[i]) return t[i];
  96. for (var l in r = {}, t[i] = r, n) n.hasOwnProperty(l) && (r[l] = e(n[l], t));
  97. return r;
  98. case "Array":
  99. 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);
  100. default:
  101. return n
  102. }
  103. },
  104. getLanguage: function(e) {
  105. for (; e;) {
  106. var t = n.exec(e.className);
  107. if (t) return t[1].toLowerCase();
  108. e = e.parentElement
  109. }
  110. return "none"
  111. },
  112. setLanguage: function(e, t) { e.className = e.className.replace(RegExp(n, "gi"), ""), e.classList.add("language-" + t) },
  113. currentScript: function() {
  114. if ("undefined" == typeof document) return null;
  115. if ("currentScript" in document) return document.currentScript;
  116. try { throw new Error } catch (r) {
  117. var e = (/at [^(\r\n]*\((.*):[^:]+:[^:]+\)$/i.exec(r.stack) || [])[1];
  118. if (e) {
  119. var n = document.getElementsByTagName("script");
  120. for (var t in n)
  121. if (n[t].src == e) return n[t]
  122. }
  123. return null
  124. }
  125. },
  126. isActive: function(e, n, t) {
  127. for (var r = "no-" + n; e;) {
  128. var a = e.classList;
  129. if (a.contains(n)) return !0;
  130. if (a.contains(r)) return !1;
  131. e = e.parentElement
  132. }
  133. return !!t
  134. }
  135. },
  136. languages: {
  137. plain: r,
  138. plaintext: r,
  139. text: r,
  140. txt: r,
  141. extend: function(e, n) { var t = a.util.clone(a.languages[e]); for (var r in n) t[r] = n[r]; return t },
  142. insertBefore: function(e, n, t, r) {
  143. var i = (r = r || a.languages)[e],
  144. l = {};
  145. for (var o in i)
  146. if (i.hasOwnProperty(o)) {
  147. if (o == n)
  148. for (var s in t) t.hasOwnProperty(s) && (l[s] = t[s]);
  149. t.hasOwnProperty(o) || (l[o] = i[o])
  150. } var u = r[e];
  151. return r[e] = l, a.languages.DFS(a.languages, (function(n, t) { t === u && n != e && (this[n] = l) })), l
  152. },
  153. DFS: function e(n, t, r, i) {
  154. i = i || {};
  155. var l = a.util.objId;
  156. for (var o in n)
  157. if (n.hasOwnProperty(o)) {
  158. t.call(n, o, n[o], r || o);
  159. var s = n[o],
  160. u = a.util.type(s);
  161. "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))
  162. }
  163. }
  164. },
  165. plugins: {},
  166. highlightAll: function(e, n) { a.highlightAllUnder(document, e, n) },
  167. highlightAllUnder: function(e, n, t) {
  168. var r = { callback: t, container: e, selector: 'code[class*="language-"], [class*="language-"] code, code[class*="lang-"], [class*="lang-"] code' };
  169. 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);
  170. for (var i, l = 0; i = r.elements[l++];) a.highlightElement(i, !0 === n, r.callback)
  171. },
  172. highlightElement: function(n, t, r) {
  173. var i = a.util.getLanguage(n),
  174. l = a.languages[i];
  175. a.util.setLanguage(n, i);
  176. var o = n.parentElement;
  177. o && "pre" === o.nodeName.toLowerCase() && a.util.setLanguage(o, i);
  178. var s = { element: n, language: i, grammar: l, code: n.textContent };
  179.  
  180. 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) }
  181. 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));
  182. if (a.hooks.run("before-highlight", s), s.grammar)
  183. if (t && e.Worker) {
  184. var c = new Worker(a.filename);
  185. c.onmessage = function(e) { u(e.data) }, c.postMessage(JSON.stringify({ language: s.language, code: s.code, immediateClose: !0 }))
  186. } else u(a.highlight(s.code, s.grammar, s.language));
  187. else u(a.util.encode(s.code))
  188. },
  189. 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) },
  190. tokenize: function(e, n) {
  191. var t = n.rest;
  192. if (t) {
  193. for (var r in t) n[r] = t[r];
  194. delete n.rest
  195. }
  196. var a = new s;
  197. return u(a, a.head, e), o(e, a, n, a.head, 0),
  198. function(e) { for (var n = [], t = e.head.next; t !== e.tail;) n.push(t.value), t = t.next; return n }(a)
  199. },
  200. hooks: {
  201. all: {},
  202. add: function(e, n) {
  203. var t = a.hooks.all;
  204. t[e] = t[e] || [], t[e].push(n)
  205. },
  206. run: function(e, n) {
  207. var t = a.hooks.all[e];
  208. if (t && t.length)
  209. for (var r, i = 0; r = t[i++];) r(n)
  210. }
  211. },
  212. Token: i
  213. };
  214.  
  215. function i(e, n, t, r) { this.type = e, this.content = n, this.alias = t, this.length = 0 | (r || "").length }
  216.  
  217. function l(e, n, t, r) {
  218. e.lastIndex = n;
  219. var a = e.exec(t);
  220. if (a && r && a[1]) {
  221. var i = a[1].length;
  222. a.index += i, a[0] = a[0].slice(i)
  223. }
  224. return a
  225. }
  226.  
  227. function o(e, n, t, r, s, g) {
  228. for (var f in t)
  229. if (t.hasOwnProperty(f) && t[f]) {
  230. var h = t[f];
  231. h = Array.isArray(h) ? h : [h];
  232. for (var d = 0; d < h.length; ++d) {
  233. if (g && g.cause == f + "," + d) return;
  234. var v = h[d],
  235. p = v.inside,
  236. m = !!v.lookbehind,
  237. y = !!v.greedy,
  238. k = v.alias;
  239. if (y && !v.pattern.global) {
  240. var x = v.pattern.toString().match(/[imsuy]*$/)[0];
  241. v.pattern = RegExp(v.pattern.source, x + "g")
  242. }
  243. 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) {
  244. var E = w.value;
  245. if (n.length > e.length) return;
  246. if (!(E instanceof i)) {
  247. var P, L = 1;
  248. if (y) {
  249. if (!(P = l(b, A, e, m)) || P.index >= e.length) break;
  250. var S = P.index,
  251. O = P.index + P[0].length,
  252. j = A;
  253. for (j += w.value.length; S >= j;) j += (w = w.next).value.length;
  254. if (A = j -= w.value.length, w.value instanceof i) continue;
  255. for (var C = w; C !== n.tail && (j < O || "string" == typeof C.value); C = C.next) L++, j += C.value.length;
  256. L--, E = e.slice(A, j), P.index -= A
  257. } else if (!(P = l(b, 0, E, m))) continue;
  258. S = P.index;
  259. var N = P[0],
  260. _ = E.slice(0, S),
  261. M = E.slice(S + N.length),
  262. W = A + E.length;
  263. g && W > g.reach && (g.reach = W);
  264. var z = w.prev;
  265. 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) {
  266. var I = { cause: f + "," + d, reach: W };
  267. o(e, n, t, w.prev, A, I), g && I.reach > g.reach && (g.reach = I.reach)
  268. }
  269. }
  270. }
  271. }
  272. }
  273. }
  274.  
  275. function s() {
  276. var e = { value: null, prev: null, next: null },
  277. n = { value: null, prev: e, next: null };
  278. e.next = n, this.head = e, this.tail = n, this.length = 0
  279. }
  280.  
  281. function u(e, n, t) {
  282. var r = n.next,
  283. a = { value: t, prev: n, next: r };
  284. return n.next = a, r.prev = a, e.length++, a
  285. }
  286.  
  287. function c(e, n, t) {
  288. for (var r = n.next, a = 0; a < t && r !== e.tail; a++) r = r.next;
  289. n.next = r, r.prev = n, e.length -= a
  290. }
  291. if (e.Prism = a, i.stringify = function e(n, t) {
  292. if ("string" == typeof n) return n;
  293. if (Array.isArray(n)) { var r = ""; return n.forEach((function(n) { r += e(n, t) })), r }
  294. var i = { type: n.type, content: e(n.content, t), tag: "span", classes: ["token", n.type], attributes: {}, language: t },
  295. l = n.alias;
  296. l && (Array.isArray(l) ? Array.prototype.push.apply(i.classes, l) : i.classes.push(l)), a.hooks.run("wrap", i);
  297. var o = "";
  298. for (var s in i.attributes) o += " " + s + '="' + (i.attributes[s] || "").replace(/"/g, "&quot;") + '"';
  299. return "<" + i.tag + ' class="' + i.classes.join(" ") + '"' + o + ">" + i.content + "</" + i.tag + ">"
  300. }, !e.document) return e.addEventListener ? (a.disableWorkerMessageHandler || e.addEventListener("message", (function(n) {
  301. var t = JSON.parse(n.data),
  302. r = t.language,
  303. i = t.code,
  304. l = t.immediateClose;
  305. e.postMessage(a.highlight(i, a.languages[r], r)), l && e.close()
  306. }), !1), a) : a;
  307. var g = a.util.currentScript();
  308.  
  309. function f() { a.manual || a.highlightAll() }
  310. 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) }
  311. return a
  312. }(_self);
  313. "undefined" != typeof module && module.exports && (module.exports = Prism), "undefined" != typeof global && (global.Prism = Prism);
  314. }
  315.  
  316. function addClikeSyntax() {
  317. Prism.languages.clike = {
  318. comment: [{ pattern: /(^|[^\\])\/\*[\s\S]*?(?:\*\/|$)/, lookbehind: !0, greedy: !0 },
  319. { pattern: /(^|[^\\:])\/\/.*/, lookbehind: !0, greedy: !0 }
  320. ],
  321. string: { pattern: /(["'])(?:\\(?:\r\n|[\s\S])|(?!\1)[^\\\r\n])*\1/, greedy: !0 },
  322. "class-name": { pattern: /(\b(?:class|extends|implements|instanceof|interface|new|trait)\s+|\bcatch\s+\()[\w.\\]+/i, lookbehind: !0, inside: { punctuation: /[.\\]/ } },
  323. keyword: /\b(?:break|catch|continue|do|else|finally|for|function|if|in|instanceof|new|null|return|throw|try|while)\b/,
  324. boolean: /\b(?:false|true)\b/,
  325. function: /\b\w+(?=\()/,
  326. number: /\b0x[\da-f]+\b|(?:\b\d+(?:\.\d*)?|\B\.\d+)(?:e[+-]?\d+)?/i,
  327. operator: /[<>]=?|[!=]=?=?|--?|\+\+?|&&?|\|\|?|[?*/~^%]/,
  328. punctuation: /[{}[\];(),.:]/
  329. };
  330. }
  331.  
  332. function addGoSyntax() {
  333. Prism.languages.go = Prism.languages
  334. .extend("clike", {
  335. string: { pattern: /(^|[^\\])"(?:\\.|[^"\\\r\n])*"|`[^`]*`/, lookbehind: !0, greedy: !0 },
  336. 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/,
  337. boolean: /\b(?:_|false|iota|nil|true)\b/,
  338. 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],
  339. operator: /[*\/%^!=]=?|\+[=+]?|-[=-]?|\|[=|]?|&(?:=|&|\^=?)?|>(?:>=?|=)?|<(?:<=?|=|-)?|:=|\.\.\./,
  340. 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/
  341. }),
  342. Prism.languages.insertBefore("go", "string", { char: { pattern: /'(?:\\.|[^'\\\r\n]){0,10}'/, greedy: !0 } }), delete Prism.languages.go["class-name"];
  343. }