lee-ui(vue版本)

开始动手写vue ui库

创建vue项目

可以利用现成的vue-cli脚手架创建项目结构

vue init webpack XXX
cd XXX
yarn  // or npm install
npm run dev

到此为止,一个普通的vue项目就创建成功,接下来需要改变一下现有的目录结构。

更改目录结构

.
...
|-- examples  //原src目录改成examples用作示例展示
|-- packages //新增 packages用于组件编写目录
...
.

这样改完之后项目是直接跑不起来的,需要改下webpack的配置,打开./build/webpack.base.conf.js,将里面所有有关src的位置改成examples。大致有下面几个位置需要修改:

1、
const createLintingRule = () => ({
  test: /\.(js|vue)$/,
  loader: 'eslint-loader',
  enforce: 'pre',
  include: [resolve('examples'), resolve('test')],  //这里的src改成examples
  options: {
    formatter: require('eslint-friendly-formatter'),
    emitWarning: !config.dev.showEslintErrorsInOverlay
  }
})
2、
entry: {
  app: './examples/main.js'  // 入口文件改成examples目录
},
3、
resolve: {
  extensions: ['.js', '.vue', '.json'],
  alias: {
    'vue$': 'vue/dist/vue.esm.js',
    '@': resolve('examples'),  //将@解释成examples
  }
}
4、
{
  test: /\.js$/,
  loader: 'babel-loader',
  include: [resolve('examples'), resolve('test')]
}

最后为了最后 npm run build能正常编译packages,我们需要为babel-loader增加一个规则

{
   test: /\.js$/,
   loader: 'babel-loader',
   include: [resolve('examples'), resolve('test'), resolve('packages')]
}

这时候,再运行 npm run build就能正常打包了,但是有个问题,打包出来的文件直接运行会报404错误,这是因为打包的根目录设置有问题,打开./config/index.js,找到build下面的assetsPublicPath: '/',改成如下即可

assetsPublicPath: './'

编写文档

一个好的ui库,说明文档自然是不可或缺的,然而markdown是最合适不过的,然而我们现在是一个vue项目,怎样再vue项目里面用markdown来编写文档呢?答案是:vue-markdown-loader打开如上链接,按照里面的说明文档配置webpack

npm i vue-markdown-loader -D

安装完成之后修改webpack.base.config.js文件,在rules下面添加一条

{
  test: /\.md$/,
  loader: 'vue-markdown-loader'
}

接下来创建文件./examples/docs/home.md

## 这是首页
::: demo
``` html
<div class="input">
  <input type="text" placeholder="请输入用户名"/>
</div>

修改router/index.js,将路由中的vue文件路径改成markdown的路径

import Vue from 'vue'
import Router from 'vue-router'

Vue.use(Router)

export default new Router({
  routes: [
    {
      path: '/',
      name: 'home',
      component: r => require.ensure([], () => r(require('../docs/home.md')))
    }
  ]
})

提示: 如果当你运行项目的时候遇到如下的报错,你需要重新运行一下 yarn 或者npm install

'webpack-dev-server' 不是内部或外部命令,也不是可运行的程序
或批处理文件。

这时候再次运行项目,应该可以看到刚才markdown中的内容!但是我们的说明文档可不止止是文字,链接,图片这么简单,我们还需要案例展示和代码展示,接下来,我们需要配置vue-mark-down的option选项

const MarkdownItContainer = require('markdown-it-container')
const striptags = require('./strip-tags')

const vueMarkdown = {
  preprocess: (MarkdownIt, source) => {
    MarkdownIt.renderer.rules.table_open = function () {
      return '<table class="table">'
    }
    MarkdownIt.renderer.rules.fence = utils.wrapCustomClass(MarkdownIt.renderer.rules.fence)
    return source
  },
  use: [
    [MarkdownItContainer, 'demo', {
      // 用于校验包含demo的代码块
      validate: params => params.trim().match(/^demo\s*(.*)$/),
      render: function(tokens, idx) {

        var m = tokens[idx].info.trim().match(/^demo\s*(.*)$/);

        if (tokens[idx].nesting === 1) {
          var desc = tokens[idx + 2].content;
          // 编译成html
          const html = utils.convertHtml(striptags(tokens[idx + 1].content, 'script'))
          // 移除描述,防止被添加到代码块
          tokens[idx + 2].children = [];

          return `<demo-block>
                        <div slot="desc">${html}</div>
                        <div slot="highlight">`;
        }
        return '</div></demo-block>\n';
      }
    }]
  ]
}

将上面的这个变量放在./build/webpack.base.config.js中并修改如下

{
  test: /\.md$/,
  loader: 'vue-markdown-loader',
  options: vueMarkdown
}

在./build/utils.js中添加如下

/**
 * 增加 hljs 的 classname
 */
exports.wrapCustomClass = function (render) {
  return function (...args) {
    return render(...args)
      .replace('<code class="', '<code class="hljs ')
      .replace('<code>', '<code class="hljs">')
  }
}

/**
 * Format HTML string
 */
exports.convertHtml = function (str) {
  return str.replace(/(&#x)(\w{4});/gi, $0 => String.fromCharCode(parseInt(encodeURIComponent($0).replace(/(%26%23x)(\w{4})(%3B)/g, '$2'), 16)))
}

新增./build/strip-tags.js

/**
 * 转换成DOM字符串
 */
const cheerio = require('cheerio')

module.exports = (str, tags) => {
  const $ = cheerio.load(str, { decodeEntities: false })

  if (!tags || tags.length === 0) {
    return str
  }

  tags = !Array.isArray(tags) ? [tags] : tags
  let len = tags.length

  while (len--) {
    $(tags[len]).remove()
  }

  return $.html()
}

上面都准备好了之后,我们需要安装相应的依赖

npm i markdown-it-container --save-dev

到此之后我们便可以看到markdown文件中既可以展示文字,代码,还可以展示可交互的输入框等等。但是这时候,页面中的代码并没有显示代码高亮,其实我们的页面元素已经有了代码高亮,只是没有引用css文件,只需要将 node_modules/highlight.js/styles/color-brewer.css引入到项目中即可,接下来就是全心全意的写插件了。

未完待续!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容