🌈 An enterprise-class UI components based on Ant Design and Vue. 🐜
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

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, ''),
},
},
}