vuecssuiant-designantdreactantantd-vueenterprisefrontendui-designvue-antdvue-antd-uivue3vuecomponent
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
225 lines
5.5 KiB
225 lines
5.5 KiB
const path = require('path') |
|
const hljs = require('highlight.js') |
|
const Token = require('markdown-it/lib/token') |
|
const cheerio = require('cheerio') |
|
const getBabelCommonConfig = require('./antd-tools/getBabelCommonConfig') |
|
const babelConfig = getBabelCommonConfig(false) |
|
|
|
babelConfig.plugins.push(require.resolve('babel-plugin-syntax-dynamic-import')) |
|
|
|
const fetch = (str, tag, scoped) => { |
|
const $ = cheerio.load(str, { |
|
decodeEntities: false, |
|
xmlMode: true, |
|
}) |
|
if (!tag) { return str } |
|
if (tag === 'style') { |
|
return scoped |
|
? $(`${tag}[scoped]`).html() |
|
: $(`${tag}`).not(`${tag}[scoped]`).html() |
|
} |
|
return $(tag).html() |
|
} |
|
|
|
/** |
|
* `{{ }}` => `<span>{{</span> <span>}}</span>` |
|
* @param {string} str |
|
* @return {string} |
|
*/ |
|
const replaceDelimiters = function (str) { |
|
return str.replace(/({{|}})/g, '<span>$1</span>') |
|
} |
|
|
|
/** |
|
* renderHighlight |
|
* @param {string} str |
|
* @param {string} lang |
|
*/ |
|
|
|
const renderHighlight = function (str, lang) { |
|
if (!(lang && hljs.getLanguage(lang))) { |
|
return '' |
|
} |
|
|
|
try { |
|
return replaceDelimiters(hljs.highlight(lang, str, true).value) |
|
} catch (err) {} |
|
} |
|
|
|
const md = require('markdown-it')('default', { |
|
html: true, |
|
breaks: true, |
|
highlight: renderHighlight, |
|
}).use(require('markdown-it-anchor'), { |
|
level: 2, |
|
slugify: string => string.trim().split(' ').join('-'), |
|
permalink: true, |
|
// renderPermalink: (slug, opts, state, permalink) => {}, |
|
permalinkClass: 'anchor', |
|
permalinkSymbol: '#', |
|
permalinkBefore: false, |
|
}) |
|
// md.renderer.rules.fence = wrap(md.renderer.rules.fence) |
|
const cnReg = new RegExp('<(cn)(?:[^<]|<)+</\\1>', 'g') |
|
const usReg = new RegExp('<(us)(?:[^<]|<)+</\\1>', 'g') |
|
md.core.ruler.push('update_template', function replace ({ tokens }) { |
|
let cn = '' |
|
let us = '' |
|
let template = '' |
|
let script = '' |
|
let style = '' |
|
let scopedStyle = '' |
|
let code = '' |
|
let sourceCode = '' |
|
tokens.forEach(token => { |
|
if (token.type === 'html_block') { |
|
if (token.content.match(cnReg)) { |
|
cn = fetch(token.content, 'cn') |
|
token.content = '' |
|
} |
|
if (token.content.match(usReg)) { |
|
us = fetch(token.content, 'us') |
|
token.content = '' |
|
} |
|
} |
|
if (token.type === 'fence' && token.info === 'html' && token.markup === '```') { |
|
sourceCode = token.content |
|
code = '````html\n' + token.content + '````' |
|
template = fetch(token.content, 'template') |
|
script = fetch(token.content, 'script') |
|
style = fetch(token.content, 'style') |
|
scopedStyle = fetch(token.content, 'style', true) |
|
token.content = '' |
|
token.type = 'html_block' |
|
} |
|
}) |
|
if (template) { |
|
let jsfiddle = { |
|
html: template, |
|
script, |
|
style, |
|
us, |
|
cn, |
|
sourceCode, |
|
} |
|
jsfiddle = md |
|
.utils |
|
.escapeHtml(JSON.stringify(jsfiddle)) |
|
const codeHtml = code |
|
? md.render(code) |
|
: '' |
|
const cnHtml = cn |
|
? md.render(cn) |
|
: '' |
|
let newContent = ` |
|
<template> |
|
<demo-box :jsfiddle="${jsfiddle}"> |
|
<template slot="component">${template}</template> |
|
<template slot="description">${cnHtml}</template> |
|
<template slot="us-description">${us |
|
? md.render(us) |
|
: ''}</template> |
|
<template slot="code">${codeHtml}</template> |
|
</demo-box> |
|
</template>` |
|
newContent += script |
|
? ` |
|
<script> |
|
${script || ''} |
|
</script> |
|
` |
|
: '' |
|
newContent += style |
|
? ` |
|
<style> |
|
${style || ''} |
|
</style> |
|
` |
|
: '' |
|
newContent += scopedStyle |
|
? ` |
|
<style scoped> |
|
${scopedStyle || ''} |
|
</style> |
|
` |
|
: '' |
|
const t = new Token('html_block', '', 0) |
|
t.content = newContent |
|
tokens.push(t) |
|
} |
|
}) |
|
|
|
module.exports = { |
|
entry: { |
|
index: [`./site/${process.env.ENTRY_INDEX || 'index'}.js`], |
|
}, |
|
module: { |
|
rules: [ |
|
{ |
|
test: /\.md/, |
|
use: [ |
|
{ |
|
loader: 'vue-antd-md-loader', |
|
options: Object.assign(md, { |
|
wrapper: 'div', |
|
vueLoaderOptions: { |
|
loaders: { |
|
js: [ |
|
{ |
|
loader: 'babel-loader', |
|
options: { |
|
presets: ['env'], |
|
plugins: ['transform-vue-jsx', 'transform-object-rest-spread'], |
|
}, |
|
}, |
|
], |
|
}, |
|
}, |
|
}), |
|
}, |
|
], |
|
}, { |
|
test: /\.vue$/, |
|
loader: 'vue-loader', |
|
options: { |
|
loaders: { |
|
js: [ |
|
{ |
|
loader: 'babel-loader', |
|
options: { |
|
presets: ['env'], |
|
plugins: ['transform-vue-jsx', 'transform-object-rest-spread', 'syntax-dynamic-import'], |
|
}, |
|
}, |
|
], |
|
}, |
|
}, |
|
}, { |
|
test: /\.(js|jsx)$/, |
|
loader: 'babel-loader', |
|
exclude: /node_modules/, |
|
options: babelConfig, |
|
}, { |
|
test: /\.(png|jpg|gif|svg)$/, |
|
loader: 'file-loader', |
|
options: { |
|
name: '[name].[ext]?[hash]', |
|
}, |
|
}, |
|
], |
|
}, |
|
resolve: { |
|
modules: [ |
|
'node_modules', path.join(__dirname, '../node_modules'), |
|
], |
|
extensions: [ |
|
'.js', '.jsx', '.vue', '.md', |
|
], |
|
alias: { |
|
'vue$': 'vue/dist/vue.esm.js', |
|
'antd': path.join(__dirname, 'components'), |
|
'ant-design-vue': path.join(__dirname, 'components'), |
|
'@': path.join(__dirname, ''), |
|
}, |
|
}, |
|
}
|
|
|