const hljs = require('highlight.js') const marked = require('marked') const fs = require('fs') const path = require('path') // const prettier = require('prettier') const escapeMap = { '&': '&', '<': '<', '>': '>', '"': '"', "'": ''' } function escapeForHTML (input) { return input.replace(/([&<>'"])/g, char => escapeMap[char]) } // Create your custom renderer. const renderer = new marked.Renderer() renderer.code = (code, language) => { // Check whether the given language is valid for highlight.js. const validLang = !!(language && hljs.getLanguage(language)) // Highlight only if the language is valid. // highlight.js escapes HTML in the code, but we need to escape by ourselves // when we don't use it. const highlighted = validLang ? hljs.highlight(language, code).value : escapeForHTML(code) // Render the highlighted code with `hljs` class. return `
${highlighted}
`
}
marked.setOptions({
renderer
})
const demoBlock = fs.readFileSync(path.resolve(__dirname, 'DemoBlock.vue')).toString()
function getPartsOfDemo (tokens) {
let template = null
let script = null
let style = null
let title = null
let content = null
for (const token of tokens) {
if (token.type === 'heading' && token.depth === 1) {
title = token.text
}
if (token.type === 'paragraph') {
content = token.text
}
if (token.type === 'code' && (token.lang === 'template' || token.lang === 'html')) {
template = token.text
}
if (token.type === 'code' && (token.lang === 'script' || token.lang === 'js')) {
script = token.text
}
if (token.type === 'code' && (token.lang === 'style' || token.lang === 'css')) {
style = token.text
}
}
return {
template: template,
script: script,
style: style,
title: title,
content: content
}
}
function mergeParts (parts) {
const mergedParts = {
...parts
}
mergedParts.title = parts.title
mergedParts.content = marked(parts.content)
mergedParts.code = ''
// console.log(parts)
if (parts.template) {
mergedParts.code += parts.template
}
if (parts.script) {
if (parts.template) mergedParts.code += '\n\n'
mergedParts.code += ``
}
if (parts.style) {
if (parts.template || parts.script) mergedParts.code += '\n\n'
mergedParts.code += ``
}
mergedParts.code = marked(`\`\`\`html
${mergedParts.code}
\`\`\``)
// console.log(mergedParts.code)
return mergedParts
}
function genVueComponent (parts) {
const titleReg = //
const contentReg = //
const codeReg = //
const scriptReg = /\/\*\*\sSCRIPT_SLOT\s\*\//
const styleReg = /\/\*\*STYLE_SLOT\*\//
const demoReg = //
let src = demoBlock
// console.log(src)
if (parts.content) {
src = src.replace(contentReg, parts.content)
}
if (parts.title) {
src = src.replace(titleReg, parts.title)
}
if (parts.code) {
src = src.replace(codeReg, parts.code)
}
if (parts.script) {
src = src.replace(scriptReg, parts.script)
}
if (parts.style) {
src = src.replace(styleReg, parts.style)
}
if (parts.template) {
src = src.replace(demoReg, parts.template)
}
return src.trim()
}
function convertMd2Demo (text) {
const tokens = marked.lexer(text)
const parts = getPartsOfDemo(tokens)
const mergedParts = mergeParts(parts)
const vueComponent = genVueComponent(mergedParts)
// console.log(vueComponent)
return vueComponent
}
module.exports = convertMd2Demo
// const startTime = new Date()
// for (let i = 0; i < 100; ++i) {
// const md = fs.readFileSync('./marked/basic.md').toString()
// convertMd2Demo(md)
// }
// const endTime = new Date()
// console.log(endTime - startTime)