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)