enhanced snarkdown for anchors
ci/woodpecker/push/woodpecker Pipeline was successful
Podrobnosti
ci/woodpecker/push/woodpecker Pipeline was successful
Podrobnosti
This commit is contained in:
rodič
061d612034
revize
29218c6bdf
|
@ -0,0 +1,109 @@
|
|||
const TAGS = {
|
||||
'': ['<em>', '</em>'],
|
||||
_: ['<strong>', '</strong>'],
|
||||
'*': ['<strong>', '</strong>'],
|
||||
'~': ['<s>', '</s>'],
|
||||
'\n': ['<br />'],
|
||||
' ': ['<br />'],
|
||||
'-': ['<hr />']
|
||||
};
|
||||
|
||||
/** Outdent a string based on the first indented line's leading whitespace
|
||||
* @private
|
||||
*/
|
||||
function outdent(str) {
|
||||
return str.replace(RegExp('^' + (str.match(/^(\t| )+/) || '')[0], 'gm'), '');
|
||||
}
|
||||
|
||||
/** Encode special attribute characters to HTML entities in a String.
|
||||
* @private
|
||||
*/
|
||||
function encodeAttr(str) {
|
||||
return (str + '').replace(/"/g, '"').replace(/</g, '<').replace(/>/g, '>');
|
||||
}
|
||||
|
||||
/** Parse Markdown into an HTML String. */
|
||||
export default function parse(md, prevLinks) {
|
||||
let tokenizer = /((?:^|\n+)(?:\n---+|\* \*(?: \*)+)\n)|(?:^``` *(\w*)\n([\s\S]*?)\n```$)|((?:(?:^|\n+)(?:\t| {2,}).+)+\n*)|((?:(?:^|\n)([>*+-]|\d+\.)\s+.*)+)|(?:!\[([^\]]*?)\]\(([^)]+?)\))|(\[)|(\](?:\(([^)]+?)\))?)|(?:(?:^|\n+)([^\s].*)\n(-{3,}|={3,})(?:\n+|$))|(?:(?:^|\n+)(#{1,6})\s*(.+)(?:\n+|$))|(?:`([^`].*?)`)|( \n\n*|\n{2,}|__|\*\*|[_*]|~~)/gm,
|
||||
context = [],
|
||||
out = '',
|
||||
links = prevLinks || {},
|
||||
last = 0,
|
||||
chunk, prev, token, inner, t;
|
||||
|
||||
function tag(token) {
|
||||
let desc = TAGS[token[1] || ''];
|
||||
let end = context[context.length - 1] == token;
|
||||
if (!desc) return token;
|
||||
if (!desc[1]) return desc[0];
|
||||
if (end) context.pop();
|
||||
else context.push(token);
|
||||
return desc[end | 0];
|
||||
}
|
||||
|
||||
function flush() {
|
||||
let str = '';
|
||||
while (context.length) str += tag(context[context.length - 1]);
|
||||
return str;
|
||||
}
|
||||
|
||||
md = md.replace(/^\[(.+?)\]:\s*(.+)$/gm, (s, name, url) => {
|
||||
links[name.toLowerCase()] = url;
|
||||
return '';
|
||||
}).replace(/^\n+|\n+$/g, '');
|
||||
|
||||
while ((token = tokenizer.exec(md))) {
|
||||
prev = md.substring(last, token.index);
|
||||
last = tokenizer.lastIndex;
|
||||
chunk = token[0];
|
||||
if (prev.match(/[^\\](\\\\)*\\$/)) {
|
||||
// escaped
|
||||
}
|
||||
// Code/Indent blocks:
|
||||
else if (t = (token[3] || token[4])) {
|
||||
chunk = '<pre class="code ' + (token[4] ? 'poetry' : token[2].toLowerCase()) + '"><code' + (token[2] ? ` class="language-${token[2].toLowerCase()}"` : '') + '>' + outdent(encodeAttr(t).replace(/^\n+|\n+$/g, '')) + '</code></pre>';
|
||||
}
|
||||
// > Quotes, -* lists:
|
||||
else if (t = token[6]) {
|
||||
if (t.match(/\./)) {
|
||||
token[5] = token[5].replace(/^\d+/gm, '');
|
||||
}
|
||||
inner = parse(outdent(token[5].replace(/^\s*[>*+.-]/gm, '')));
|
||||
if (t == '>') t = 'blockquote';
|
||||
else {
|
||||
t = t.match(/\./) ? 'ol' : 'ul';
|
||||
inner = inner.replace(/^(.*)(\n|$)/gm, '<li>$1</li>');
|
||||
}
|
||||
chunk = '<' + t + '>' + inner + '</' + t + '>';
|
||||
}
|
||||
// Images:
|
||||
else if (token[8]) {
|
||||
chunk = `<img src="${encodeAttr(token[8])}" alt="${encodeAttr(token[7])}">`;
|
||||
}
|
||||
// Links:
|
||||
else if (token[10]) {
|
||||
out = out.replace('<a>', `<a href="${encodeAttr(token[11] || links[prev.toLowerCase()])}">`);
|
||||
chunk = flush() + '</a>';
|
||||
}
|
||||
else if (token[9]) {
|
||||
chunk = '<a>';
|
||||
}
|
||||
// Headings:
|
||||
else if (token[12] || token[14]) {
|
||||
t = 'h' + (token[14] ? token[14].length : (token[13] > '=' ? 1 : 2));
|
||||
chunk = '<' + t + ' id="' + parse(token[12] || token[15]).replaceAll(" ", "-").toLowerCase() + '">' + parse(token[12] || token[15], links) + '</' + t + '>';
|
||||
}
|
||||
// `code`:
|
||||
else if (token[16]) {
|
||||
chunk = '<code>' + encodeAttr(token[16]) + '</code>';
|
||||
}
|
||||
// Inline formatting: *em*, **strong** & friends
|
||||
else if (token[17] || token[1]) {
|
||||
chunk = tag(token[17] || '--');
|
||||
}
|
||||
out += prev;
|
||||
out += chunk;
|
||||
}
|
||||
|
||||
return (out + md.substring(last) + flush()).replace(/^\n+|\n+$/g, '');
|
||||
}
|
|
@ -1,10 +1,12 @@
|
|||
/** @type {import('./$types').PageLoad} */
|
||||
|
||||
import snarkdown from 'snarkdown';
|
||||
//import snarkdown from 'snarkdown';
|
||||
import parse from "$lib/vsnarkdown"
|
||||
import { base } from '$app/paths';
|
||||
|
||||
export async function load({ fetch }) {
|
||||
const response = await fetch(`${base}/hacker-manual.md`).then((r) => r.text());
|
||||
const compiledResponse = await snarkdown(response);
|
||||
|
||||
const compiledResponse = await parse(response);
|
||||
return { content: compiledResponse };
|
||||
}
|
||||
|
|
|
@ -1,12 +1,13 @@
|
|||
<script>
|
||||
import { base } from '$app/paths';
|
||||
import { onMount } from 'svelte';
|
||||
import snarkdown from 'snarkdown';
|
||||
//import snarkdown from 'snarkdown';
|
||||
import parse from '$lib/vsnarkdown';
|
||||
export let data;
|
||||
onMount(async () => {
|
||||
const response = await fetch(`${base}/hacker-manual.md`);
|
||||
const result = await response.text();
|
||||
data.content = await snarkdown(result);
|
||||
data.content = await parse(result);
|
||||
//console.log(data.contributors);
|
||||
});
|
||||
</script>
|
||||
|
@ -42,7 +43,7 @@
|
|||
Useful information before you arrive, like venue locations, accomodation and public
|
||||
transport.
|
||||
</p>
|
||||
<a class="mt-3 text-gray-500 inline-flex items-center" href="#before"
|
||||
<a class="mt-3 text-gray-500 inline-flex items-center" href="#before-the-hackathon"
|
||||
>Learn more<svg
|
||||
fill="none"
|
||||
stroke="currentColor"
|
||||
|
@ -126,7 +127,9 @@
|
|||
<article
|
||||
class="lg:flex-grow md:w-1/2 lg:pl-32 md:pl-16 lg:pr-32 md:pr-16 flex flex-col md:items-start md:text-left items-center text-center"
|
||||
>
|
||||
<div class="[&_h1]:header [&_h2]:head2 [&_li]:list [&_h3]:head3 [&_h4]:head4 [&_img[src*='#left']]:float-left [&_img[src*='#left']]:m-5">
|
||||
<div
|
||||
class="[&_h1]:header [&_h2]:head2 [&_li]:list [&_h3]:head3 [&_h4]:head4 [&_img[src*='#left']]:float-left [&_img[src*='#left']]:mt-5 [&_img[src*='#left']]:mr-5"
|
||||
>
|
||||
{@html data.content}
|
||||
</div>
|
||||
</article>
|
||||
|
|
|
@ -8,9 +8,9 @@
|
|||
## Overview
|
||||
|
||||
### Before the Hackathon
|
||||
|
||||
This section aims to help you better prepare for the hackathon.
|
||||
|
||||
|
||||
![](/icons/map.svg#left)
|
||||
#### Travelling
|
||||
|
||||
|
|
Načítá se…
Odkázat v novém úkolu