docusaurus/lib/core/Remarkable.js
2017-10-24 10:49:13 -07:00

93 lines
No EOL
2.3 KiB
JavaScript

'use strict';
const React = require('react');
const hljs = require('highlight.js')
const Markdown = require('remarkable');
const toSlug = require("./toSlug.js");
const CWD = process.cwd();
function anchors(md) {
md.renderer.rules.heading_open = function(tokens, idx /*, options, env */) {
return '<h' + tokens[idx].hLevel + '>' + '<a class="anchor" name="' + toSlug(tokens[idx+1].content) + '"></a>';
};
md.renderer.rules.heading_close = function(tokens, idx /*, options, env */) {
return ' <a class="hash-link" href="#' + toSlug(tokens[idx-1].content) + '">#</a>' + '</h' + tokens[idx].hLevel + '>\n';
};
}
class Remarkable extends React.Component {
render() {
var Container = this.props.container;
return (
<Container>
{this.content()}
</Container>
);
}
componentWillUpdate(nextProps, nextState) {
if (nextProps.options !== this.props.options) {
this.md = new Markdown(nextProps.options);
}
}
content() {
if (this.props.source) {
return <span dangerouslySetInnerHTML={{ __html: this.renderMarkdown(this.props.source) }} />;
}
else {
return React.Children.map(this.props.children, child => {
if (typeof child === 'string') {
return <span dangerouslySetInnerHTML={{ __html: this.renderMarkdown(child) }} />;
}
else {
return child;
}
});
}
}
renderMarkdown(source) {
if (!this.md) {
this.md = new Markdown({
highlight: function (str, lang) {
if (lang && hljs.getLanguage(lang)) {
try {
return hljs.highlight(lang, str).value;
} catch (err) {}
}
try {
return hljs.highlightAuto(str).value;
} catch (err) {}
return ''; // use external default escaping
}
});
// Register anchors plugin
this.md.use(anchors);
// Allow client sites to register their own plugins
const siteConfig = require(CWD + "/siteConfig.js");
if (siteConfig.markdownPlugins) {
siteConfig.markdownPlugins.forEach(function(plugin) {
this.md.use(plugin);
}, this);
}
}
return this.md.render(source);
}
}
Remarkable.defaultProps = {
container: 'div',
options: {},
};
module.exports = Remarkable;