mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-13 22:41:32 -05:00
123 lines
3.7 KiB
JavaScript
123 lines
3.7 KiB
JavaScript
/* jshint node:true, browser:true */
|
|
|
|
// Adds footnote syntax as per Markdown Extra:
|
|
//
|
|
// https://michelf.ca/projects/php-markdown/extra/#footnotes
|
|
//
|
|
// That's some text with a footnote.[^1]
|
|
//
|
|
// [^1]: And that's the footnote.
|
|
//
|
|
// That's the second paragraph.
|
|
//
|
|
// Also supports [^n] if you don't want to worry about preserving
|
|
// the footnote order yourself.
|
|
|
|
function replaceInlineFootnotes(text) {
|
|
// Inline footnotes e.g. "foo[^1]"
|
|
var inlineRegex = /(?!^)\[\^(\d+|n)\]/gim,
|
|
i = 0;
|
|
|
|
return text.replace(inlineRegex, function (match, n) {
|
|
// We allow both automatic and manual footnote numbering
|
|
if (n === 'n') {
|
|
n = i + 1;
|
|
}
|
|
|
|
var s = '<sup id="fnref:' + n + '">' +
|
|
'<a href="#fn:' + n + '" rel="footnote">' + n + '</a>' +
|
|
'</sup>';
|
|
i += 1;
|
|
return s;
|
|
});
|
|
}
|
|
|
|
function replaceEndFootnotes(text, converter) {
|
|
// Expanded footnotes at the end e.g. "[^1]: cool stuff"
|
|
var endRegex = /\[\^(\d+|n)\]: ([\s\S]*?)$(?! )/gim,
|
|
m = text.match(endRegex),
|
|
total = m ? m.length : 0,
|
|
i = 0;
|
|
|
|
return text.replace(endRegex, function (match, n, content) {
|
|
if (n === 'n') {
|
|
n = i + 1;
|
|
}
|
|
|
|
content = content.replace(/\n /g, '<br>');
|
|
content = converter.makeHtml(content);
|
|
content = content.replace(/<\/p>$/, '');
|
|
var s = '<li class="footnote" id="fn:' + n + '">' +
|
|
content + ' <a href="#fnref:' + n +
|
|
'" title="return to article">↩</a>' +
|
|
'</p></li>';
|
|
|
|
if (i === 0) {
|
|
s = '<div class="footnotes"><ol>' + s;
|
|
}
|
|
|
|
if (i === total - 1) {
|
|
s = s + '</ol></div>';
|
|
}
|
|
|
|
i += 1;
|
|
return s;
|
|
});
|
|
}
|
|
|
|
(function () {
|
|
var footnotes = function (converter) {
|
|
return [
|
|
{
|
|
type: 'lang',
|
|
filter: function (text) {
|
|
var preExtractions = {},
|
|
hashID = 0;
|
|
|
|
function hashId() {
|
|
return hashID += 1;
|
|
}
|
|
|
|
// Extract pre blocks
|
|
text = text.replace(/<(pre|code)>[\s\S]*?<\/(\1)>/gim, function (x) {
|
|
var hash = hashId();
|
|
preExtractions[hash] = x;
|
|
return '{gfm-js-extract-pre-' + hash + '}';
|
|
}, 'm');
|
|
|
|
text = text.replace(/```[\s\S]*?\n```/gim, function (x) {
|
|
var hash = hashId();
|
|
preExtractions[hash] = x;
|
|
return '{gfm-js-extract-pre-' + hash + '}';
|
|
}, 'm');
|
|
|
|
// Extract code blocks
|
|
text = text.replace(/`[\s\S]*?`/gim, function (x) {
|
|
var hash = hashId();
|
|
preExtractions[hash] = x;
|
|
return '{gfm-js-extract-pre-' + hash + '}';
|
|
}, 'm');
|
|
|
|
text = replaceInlineFootnotes(text);
|
|
text = replaceEndFootnotes(text, converter);
|
|
|
|
// replace extractions
|
|
text = text.replace(/\{gfm-js-extract-pre-([0-9]+)\}/gm, function (x, y) {
|
|
return preExtractions[y];
|
|
});
|
|
|
|
return text;
|
|
}
|
|
}
|
|
];
|
|
};
|
|
|
|
// Client-side export
|
|
if (typeof window !== 'undefined' && window.Showdown && window.Showdown.extensions) {
|
|
window.Showdown.extensions.footnotes = footnotes;
|
|
}
|
|
// Server-side export
|
|
if (typeof module !== 'undefined') {
|
|
module.exports = footnotes;
|
|
}
|
|
}());
|