Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >在不同 webpack 版本的 Vue 项目中配置 Storybook

在不同 webpack 版本的 Vue 项目中配置 Storybook

作者头像
江米小枣
发布于 2021-04-29 09:07:45
发布于 2021-04-29 09:07:45
1.1K00
代码可运行
举报
文章被收录于专栏:云前端云前端
运行总次数:0
代码可运行

在之前的一篇文章中,介绍过组件化搭建工具 storybook 在 vue 项目中的安装和配置。相比于其成文的时间,vue 项目依赖的工具多有发展;并且在实际应用中,多种历史版本的项目并存的状况比比皆是,用官方提供的 npx sb init 往往会出现配置失败的情况,而较新或过旧的资料都在网上难觅 -- 所以在此特别补充一篇,记录 新、旧 两种典型配置下,storybook 可用的手动配置方法:

1. babel7 + webpack5

1.1 安装过程

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
diff --git a/.babelrc b/.babelrc
index e1f4817..71b5fe8 100644
--- a/.babelrc
+++ b/.babelrc
@@ -6,7 +6,8 @@
         "modules": false,
         "targets": "last 2 years, ie > 8"
       }
-    ]
+    ],
+    "@vue/babel-preset-jsx"
   ],
   "plugins": [
     "@babel/plugin-transform-runtime",
diff --git a/.eslintignore b/.eslintignore
index e2192c5..0858135 100644
--- a/.eslintignore
+++ b/.eslintignore
@@ -2,4 +2,5 @@
+/test/*
+*.spec.js
diff --git a/.storybook/demo.css b/.storybook/demo.css
new file mode 100644
index 0000000..820f051
--- /dev/null
+++ b/.storybook/demo.css
@@ -0,0 +1,12 @@
+.demo {
+    padding: 50px;
+    display: inline-block;
+}
+.demo .block {
+    margin-bottom: 30px;
+}
diff --git a/.storybook/main.js b/.storybook/main.js
new file mode 100644
index 0000000..ceca416
--- /dev/null
+++ b/.storybook/main.js
@@ -0,0 +1,22 @@
+const path = require("path");
+const pathResolve = p => path.resolve(__dirname, "../", p);
+
+module.exports = {
+  stories: ["../src/**/*.stories.mdx", "../src/**/*.stories.@(js|jsx|ts|tsx)"],
+  addons: ["@storybook/addon-links", "@storybook/addon-essentials"],
+  core: {
+    builder: "webpack5"
+  },
+  webpackFinal: config => {
+    config.resolve.alias = {
+      ...config.resolve.alias,
+      "@": pathResolve("src"),
+      "~": pathResolve("node_modules")
+    };
+    config.module.rules.push({
+      test: /\.scss$/,
+      use: ["vue-style-loader", "css-loader", "sass-loader"]
+    });
+    return config;
+  }
+};
diff --git a/.storybook/manager-head.html b/.storybook/manager-head.html
new file mode 100644
index 0000000..c5c2955
--- /dev/null
+++ b/.storybook/manager-head.html
@@ -0,0 +1,14 @@
+<style>
+  .sidebar-header {
+    text-align: center;
+  }
+  .sidebar-header > div > a {
+    font-size: 28px;
+  }
+</style>
diff --git a/.storybook/manager.js b/.storybook/manager.js
new file mode 100644
index 0000000..c2321a4
--- /dev/null
+++ b/.storybook/manager.js
@@ -0,0 +1,10 @@
+import { addons } from "@storybook/addons";
+import { create } from "@storybook/theming";
+
+addons.setConfig({
+  theme: create({
+    base: "light",
+    brandImage: null,
+    brandTitle: "前端组件"
+  })
+});
diff --git a/.storybook/preview.js b/.storybook/preview.js
new file mode 100644
index 0000000..55d75d7
--- /dev/null
+++ b/.storybook/preview.js
@@ -0,0 +1,23 @@
+import "./demo.css";
+
+export const parameters = {
+    actions: { argTypesRegex: "^on[A-Z].*" },
+    controls: {
+        matchers: {
+            color: /(background|color)$/i,
+            date: /Date$/
+        }
+    },
+    // https://github.com/storybookjs/storybook/pull/9090/commits/2db3faa727fed585fb5d9e6db23cc3835fc88829
+    viewMode: "docs"
+};
diff --git a/package.json b/package.json
index 673f3a9..c0b7b27 100644
--- a/package.json
+++ b/package.json
@@ -1,19 +1,23 @@
+    "storybook": "start-storybook -p 6006 --no-manager-cache",
+    "build-storybook": "build-storybook"
   },
   "dependencies": {
+    "@babel/plugin-syntax-jsx": "^7.12.13",
+    "@storybook/addon-actions": "^6.2.7",
+    "@storybook/addon-essentials": "^6.2.7",
+    "@storybook/addon-links": "^6.2.7",
+    "@storybook/builder-webpack5": "^6.2.7",
+    "@storybook/theming": "^6.2.8",
+    "@storybook/vue": "^6.2.7",
+    "@vue/babel-helper-vue-jsx-merge-props": "^1.2.1",
+    "@vue/babel-preset-jsx": "^1.2.4",
     "babel-plugin-transform-vue-jsx": "~3.7.0",
diff --git a/src/utils/utils.js b/src/utils/utils.js
new file mode 100644
index 0000000..3e8413d
--- /dev/null
+++ b/src/utils/utils.js
@@ -0,0 +1,103 @@
+export const formatCode = jsCode =>
+    JSON.stringify(
+        jsCode,
+        // eslint-disable-next-line
+    (key, value) => {
+            if (typeof value === 'function') {
+                return value.toString();
+            } else {
+                return value;
+            }
+        },
+        2
+    ).replace(/\\n/g, '\n');
diff --git a/src/utils/storyUtils.js b/src/utils/storyUtils.js
new file mode 100644
index 0000000..4ad5149
--- /dev/null
+++ b/src/utils/storyUtils.js
@@ -0,0 +1,49 @@
+import {formatCode} from './utils';
+
+/**
+ * 生成 storybook 用例辅助函数
+ * @param {String} path - 组件的层级式路径,由 `/` 分割
+ * @param {Object} storyComponent - 用于展示的用例
+ * @param {String} markdown - 文档
+ * @param {Object} [originComponent] - 可选,如果需要从原始组件内部的 jsdoc 自动生成 API,则传入原始组件
+ * @return {Object} story - {metadata, named}
+ * @see https://github.com/storybookjs/storybook/issues/8527
+ */
+export const storyOf = (path, storyComponent, markdown, originComponent) => {
+    const pathArr = path.split('/').map(p => {
+    // https://storybook.js.org/docs/vue/writing-stories/args#setting-args-through-the-url
+        if (!/^[0-9a-zA-Z]/.test(p)) {
+            return '#' + p;
+        }
+        return p;
+    });
+    const storyName = pathArr[pathArr.length - 1];
+
+    const metadata = {
+        component: originComponent || void 0,
+        title: pathArr.join('/'),
+        decorators: [
+            () => ({template: '<div class="demo"><story></story></div>'})
+        ]
+    };
+
+    const named = () => storyComponent;
+    named.storyName = storyName;
+    named.parameters = {
+        docs: {
+            description: {
+                component: markdown || '&nbsp;'
+            },
+            source: {
+                // type: 'dynamic'
+                type: 'code',
+                code: formatCode(storyComponent)
+            }
+        }
+    };
+
+    return {
+        metadata,
+        named
+    };
+};

1.2 组件和用例

用例1:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import { Meta, Preview, Description } from '@storybook/addon-docs/blocks'
import readme from './README.md'
import pathPrefix from './pathPrefix.js'

<Meta title={pathPrefix + "/API"} />

<Description markdown={readme} />

组件2:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<template functional>
    <x-table
        v-bind="{ ...data.attrs, ...props }"
        v-on="listeners"
    >
       <!-- ... --> 
    </x-table>
</template>

<script>
export default {
    props: {
    /**
     * 配置中增加了 __自定义 component__ 的能力,避免了在 template 中再分别写 slot;
     * 自定义组件对象会默认接收 `row` 和 `column-config` 两个属性
     */
        columns: {
            type: Array,
            default: () => []
        }
    },
    emits: [
    /**
     * 自定义组件中如果发出同名事件,会被 table 的容器监听到;
     * 参数为 `key, ...values`
     */
        'column-action'
    ]
};
</script>

用例2:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import {storyOf} from '@/utils/storyUtils';
import Comp from '@/components/SimpleTable';

const title = 'Example/SimpleTable|全配置化表格';

const markdown = `
  > 基于 \`x-table\` 的扩展,可以接收其原有属性和事件
`;

const comp = {
    template: `
        <x-simple-table
            :width="600"
            :columns="columns"
            :data="list"
            @column-action="onColAction"
        />
    `,
    data() {
        return {
            //...
        };
    },
    methods: {
        onColAction(act, value) {
            alert([act, value]);
        }
    }
};

const {metadata, named} = storyOf(title, comp, markdown, Comp);
export default metadata;
export const story = named;

2. babel6 + webpack4

2.1 安装过程

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
diff --git a/.storybook/addons.js b/.storybook/addons.js
new file mode 100644
index 0000000..7106272
--- /dev/null
+++ b/.storybook/addons.js
@@ -0,0 +1,4 @@
+import 'storybook-addon-vue-info/lib/register';
+import '@storybook/addon-backgrounds/register';
+import '@storybook/addon-viewport/register';
+import '@storybook/addon-storysource/register';
diff --git a/.storybook/config.js b/.storybook/config.js
new file mode 100644
index 0000000..f84ef78
--- /dev/null
+++ b/.storybook/config.js
@@ -0,0 +1,34 @@
+import {configure, addDecorator, addParameters} from '@storybook/vue';
+import {create} from '@storybook/theming';
+import {withInfo, setDefaults} from 'storybook-addon-vue-info';
+import './demo.css';
+
+setDefaults({
+    header: false,
+    useDocgen: false
+});
+addDecorator(withInfo);
+addParameters({
+    options: {
+        theme: create({
+            base: 'light',
+            brandImage: null,
+            brandTitle: '前端组件'
+        })
+    }
+});
+const req = require.context('../src', true, /\.stories\.js$/);
+function loadStories() {
+    req.keys().forEach(filename => {
+        try {
+            req(filename);
+        } catch (ex) {
+            console.log('storybook-req', ex);
+        }
+    });
+}
+configure(loadStories, module);
diff --git a/.storybook/manager-head.html b/.storybook/manager-head.html
new file mode 100644
index 0000000..c5c2955
--- /dev/null
+++ b/.storybook/manager-head.html
@@ -0,0 +1,14 @@
+<style>
+  .sidebar-header {
+    text-align: center;
+  }
+  .sidebar-header > div > a {
+    font-size: 28px;
+  }
+</style>
diff --git a/.storybook/demo.css b/.storybook/demo.css
new file mode 100644
index 0000000..820f051
--- /dev/null
+++ b/.storybook/demo.css
@@ -0,0 +1,12 @@
+.demo {
+    padding: 50px;
+    display: inline-block;
+}
+
+.demo .block {
+    margin-bottom: 30px;
+}
diff --git a/.storybook/preview-head.html b/.storybook/preview-head.html
new file mode 100644
index 0000000..e69de29
diff --git a/.storybook/webpack.config.js b/.storybook/webpack.config.js
new file mode 100644
index 0000000..3186d15
--- /dev/null
+++ b/.storybook/webpack.config.js
@@ -0,0 +1,32 @@
+const path = require('path');
+const pathResolve = p => path.resolve(__dirname, '../', p);
+module.exports = ({config, mode}) => {
+    config.resolve.alias = {
+        ...config.resolve.alias,
+        '@': pathResolve('src'),
+        '~': pathResolve('node_modules')
+    };
+    config.module.rules.push({
+        test: /\.scss$/,
+        use: ['style-loader', 'css-loader', 'sass-loader']
+    });
+    config.module.rules.push({
+        test: /\.vue$/,
+        loader: 'storybook-addon-vue-info/loader',
+        enforce: 'post'
+    });
+    config.module.rules.push({
+        test: /\.stories\.js$/,
+        loaders: [require.resolve('@storybook/addon-storysource/loader')],
+        enforce: 'pre'
+    });
+    if (process.env.NODE_ENV === 'production') {
+        config.output.filename = 'bundle.[name].js';
+        config.optimization.splitChunks.automaticNameDelimiter = '.';
+        config.optimization.runtimeChunk = {
+            name: entrypoint => `runtime.${entrypoint.name}`
+        };
+    }
+    // console.log(config);
+    return config;
+};
diff --git a/package.json b/package.json
index d83bbba..6027c58 100644
--- a/package.json
+++ b/package.json
@@ -21,7 +21,9 @@
+        "build-storybook": "build-storybook",
+        "storybook": "start-storybook -p 6006"
     "devDependencies": {
+        "@storybook/addon-backgrounds": "^5.3.18",
+        "@storybook/addon-info": "^5.3.18",
+        "@storybook/addon-storysource": "^5.3.18",
+        "@storybook/addon-viewport": "^5.3.18",
+        "@storybook/addons": "^5.3.18",
+        "@storybook/theming": "^5.3.18",
+        "@storybook/vue": "^5.3.18",
+        "storybook-addon-vue-info": "^1.4.2",

2.2 用例

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import {storiesOf} from '@storybook/vue';
import CustomCols from './index';

const totalColumns = [
    // ...
];

const description = {
    CustomCols: {
        props: {
            choose: '用于 v-model 的值',
            totalColumns: '可选择的列',
            storageName: '本地存储的key',
        },
        events: {
            'on-change': '选择的列改变时触发'
        }
    }
};

const info = {
    useDocgen: false,
    summary: '基于 xxx'
};

storiesOf('增强的自定义表格列', module)
    .add(
        '本机存储上次选择结果',
        () => ({
            components: {
                CustomCols
            },
            template: `
            <div class="demo">
                    <div id="demo-result"></div>
                    <custom-cols
                        :storage-name="selColsStoName"
                        :total-columns="totalColumns"
                        v-model="selectedCols"
                        @on-change="onChange"
                    />
            </div>
            `,
            data() {
                return {
                    totalColumns,
                    selColsStoName: `${window.location.hostname}_overview_custom_cols`,
                    selectedCols: ['impression', 'click']
                };
            },
            methods: {
                onChange(v) {
                    //...
                }
            },
            description
        }),
        {
            info
        }
    );

--End--

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-04-20,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 云前端 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
59.Vue 使用webpack构建vue项目
在前面的篇章中,已经说明了 webpack4 的基本使用,那么本章节开始在 webpack4 中构建vue项目,同时对比看看这种构建方式,与在网页script导入 vue.js 中的区别。
Devops海洋的渔夫
2020/08/20
2.7K0
59.Vue 使用webpack构建vue项目
从零开始,手摸手搭建前端组件库
https://majunchang.github.io/mi.vant/#/quickStart
念念不忘
2019/11/11
2.9K1
​(非软文)Webpack从入门到实战搭建Vue脚手架(一万字总结)
安装依赖包 (-D 将依赖记录成开发依赖, 只是开发中需要用的依赖, 实际上线不需要的)
Vam的金豆之路
2021/12/01
1.3K0
webpack 基础知识整理
webpack是一个 模块打包工具,支持所有的打包语法,比如 ES Module、CommonJS、CMD、AMD。初期的webpack是用来模块打包js的,发展到现在,已经可以打包很多种文件类型,比如 css、img 。
神葳
2021/01/22
1.4K0
手摸手教你用 Storybook 改善组件库的开发
在上一篇文章 《手摸手教你封装跨项目复用的 Vue 组件》 中,介绍了一例用 rollup.js 封装 Vue.js 组件库的实践;限于篇幅和复杂度,其中组件的即时调试预览部分,也同样采用了 rollup 一并配置出来,虽然完全够用,但运行起来稍嫌麻烦,bigger 上感觉也差强人意。
江米小枣
2020/06/15
2K0
webpack 入门教程
本质上,webpack 是一个现代 JavaScript 应用程序的静态模块打包器(module bundler)。当 webpack 处理应用程序时,它会递归地构建一个依赖关系图(dependency graph),其中包含应用程序需要的每个模块,然后将所有这些模块打包成一个或多个 bundle。
老马
2018/08/20
4.2K0
webpack 入门教程
从零搭建基于react与ts的组件库(一)项目搭建与封装antd组件
为什么会有这样一篇文章?因为网上的教程/示例只说了怎么做,没有系统详细的介绍引入这些依赖、为什么要这样配置,甚至有些文章还是错的!迫于技术洁癖,我希望更多的开发小伙伴能够真正的理解一个项目搭建各个方面的细节,做到面对对于工程出现的错误能够做到有把握。
w4ngzhen
2023/10/17
1.3K0
从零搭建基于react与ts的组件库(一)项目搭建与封装antd组件
从0开始Vue.js 和 Webpack 4 [2]
原因:当配置index.html时,指定了javascript的具体路径,为了使热模块重新加载工作,需要允许注入此路径,这样就可以实时更新html了。
bit20041436
2018/09/05
1.2K0
Vue学习-Webpack
从本质上来讲,webpack是一个现代的JavaScript应用的静态模块打包工具。
花猪
2022/02/17
1.3K0
Vue学习-Webpack
前端基建处理之组件库优化方案
前端基建开发一直被认为是前端开发中的 “高阶技能”。而内部组件库的开发则算是基建中比较“容易”入手的一个方向。所以咱们今天就利用这篇文章,来看一看 组件库开发 的优化方案。
winty
2024/01/03
5010
前端基建处理之组件库优化方案
使用webpack打包vue工程
记得去年十月份的时候,自己在研究webpack,当时只是知道大致的用法,写了一个简单的demo,现在,经过了7个月对公司产品架构的使用,以及对vue-cli的使用,在了解了实际应用中各种需求之后,我自
windseek
2018/06/15
1K0
【个人笔记】2023年搭建基于webpack5与typescript的react项目
由于我在另外的一些文章所讨论或分析的内容可能基于一个已经初始化好的项目,为了避免每一个文章都重复的描述如何搭建项目,我在本文会统一记录下来,今后相关的文章直接引用文本,方便读者阅读。此文主要为个人笔记,不会有太多的关于思路的描述;另外,本文仅仅描述如何搭建基础react项目,不涉及图片等资源的加载,关于图片等资源的处理,我会单独编写一期。
w4ngzhen
2023/10/17
4460
【个人笔记】2023年搭建基于webpack5与typescript的react项目
webpack3.x文件配置
webpack3.x 主配置webpack.config.js文件,每个模块的详细信息都有一定的注释,如果有不对的地方,还请指教!
White feathe
2021/12/08
9770
webpack高级配置_2023-03-01
我主要是想说摇树失败的原因(tree shaking 失败的原因),先讲下摇树本身效果
gogo2027
2023/03/01
9430
Angular 工具篇之Storybook
Storybook 是一个 UI 组件的开发环境。它允许你能够浏览一个组件库,查看每个组件的不同状态,以及支持交互式的方式开发和测试组件。
阿宝哥
2019/11/05
2.1K0
从0到1 Webpack搭建Vue3开发、生产环境
起步 创建项目目录 mkdir webpack-vue3-demo 初始化 package.json npm init -y 参考文档 安装 webpack webpack-cli webpack-dev-server webpack-merge npm install webpack webpack-cli webpack-dev-server webpack-merge --save-dev 创建配置文件 mkdir build cd build touch webpack.b
peng_tianyu
2022/12/15
1.2K0
从0到1 Webpack搭建Vue3开发、生产环境
Webpack5构造React多页面应用
来源 | https://github.com/zhedh/react-multi-page-app/
winty
2021/01/07
3.8K0
webpack+vue项目实战(一,搭建运行环境和相关配置)
现在正在开发一个公司的后台管理项目,项目是一个单页面应用。功能上就是管理销售订单的各个环节,包括物流管理,回款管理,订单管理等等的功能。这些就不多说了。项目是,基于webpack3,vue2.2.6,element2.2.9。js语法是使用es6,还有使用到的一些资源比如,vue-router,vue-resource,webpack-dev-server等。运行环境是node6.10.0,npm3.10.10,其它版本的小小伙伴要注意版本兼容的问题喔!
守候i
2018/08/22
1.2K0
webpack+vue项目实战(一,搭建运行环境和相关配置)
storybook的介绍和使用 比较火的响应式UI开发及测试环境
storybook是一套最近比较火的响应式UI 开发及测试环境。 可以可视化开发调试react,vue组件 官网:https://github.com/storybooks/storybook 至于为什么叫storybook,应该是敏捷开发中的user story有关,找了篇 文章,不了解的同学可以看下
mafeifan
2018/09/10
3.2K0
storybook的介绍和使用 比较火的响应式UI开发及测试环境
Vue 3 和 Webpack 5 来了,手动搭建的知识该更新了
前言 想必绝大多数用 Vue 开发过项目的同学,或多或少会有以下两种情况: 用 Vue CLI 工具去搭建一个项目。 在领导或同事搭建好的项目基础上做业务。 长此以往,会导致你对整个项目的把控度越来
Nealyang
2021/02/01
2K0
Vue 3 和 Webpack 5 来了,手动搭建的知识该更新了
相关推荐
59.Vue 使用webpack构建vue项目
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档