├── .all-contributorsrc ├── .browserslistrc ├── .eslintrc.js ├── .github └── ISSUE_TEMPLATE │ ├── bug_report.md │ ├── question.md │ └── suggestion.md ├── .gitignore ├── .travis.yml ├── LICENSE ├── README.md ├── babel.config.js ├── build ├── bin │ └── gen-cssfile.js └── docs-deploy.sh ├── components.json ├── docs ├── .vuepress │ ├── config.js │ ├── enhanceApp.js │ └── layout.vue ├── README.md └── component │ ├── README.md │ ├── basic │ ├── deepexi-card.md │ ├── list-container.md │ └── markdown-view.md │ ├── guide │ ├── guide.md │ └── introduction.md │ └── hoc │ ├── loading-button.md │ ├── loading-dialog.md │ ├── prompt-form.md │ ├── search-input.md │ └── text-tooltip.md ├── package.json ├── plop-templates ├── component │ ├── css.hbs │ ├── export.hbs │ ├── index.hbs │ ├── md.hbs │ └── prompt.js ├── utils.js └── writeJsonAndBuildEntry.js ├── plopfile.js ├── public ├── demo.html ├── favicon.ico ├── index.html └── test.html ├── src ├── components │ ├── deepexi-card │ │ ├── index.js │ │ └── index.vue │ ├── list-container │ │ ├── index.js │ │ └── index.vue │ ├── loading-button │ │ ├── index.js │ │ └── index.vue │ ├── loading-dialog │ │ ├── index.js │ │ ├── index.vue │ │ └── render.js │ ├── markdown-view │ │ ├── index.js │ │ ├── index.vue │ │ └── utils.js │ ├── prompt-form │ │ ├── drawer.vue │ │ ├── gene-wrapper.js │ │ └── index.js │ ├── search-input │ │ ├── index.js │ │ └── index.vue │ └── text-tooltip │ │ ├── index.js │ │ └── index.vue ├── index.js ├── mixins │ └── emitter.js ├── theme-chalk │ ├── gulpfile.js │ └── src │ │ ├── common │ │ ├── markdown-preview.less │ │ └── var.less │ │ ├── deepexi-card.less │ │ ├── index.less │ │ ├── list-container.less │ │ ├── loading-button.less │ │ ├── loading-dialog.less │ │ ├── markdown-view.less │ │ ├── search-input.less │ │ └── text-tooltip.less └── utils │ └── assist.js ├── vue.config.js └── yarn.lock /.all-contributorsrc: -------------------------------------------------------------------------------- 1 | { 2 | "projectName": "deep-ui", 3 | "projectOwner": "cjfff", 4 | "repoType": "github", 5 | "repoHost": "https://github.com", 6 | "files": [ 7 | "README.md" 8 | ], 9 | "imageSize": 100, 10 | "commit": true, 11 | "commitConvention": "gitmoji", 12 | "contributors": [ 13 | { 14 | "login": "cjfff", 15 | "name": "cjfff", 16 | "avatar_url": "https://avatars1.githubusercontent.com/u/20502762?v=4", 17 | "profile": "http://www.ccc1996.cn", 18 | "contributions": [ 19 | "doc", 20 | "bug", 21 | "example", 22 | "ideas", 23 | "maintenance", 24 | "code" 25 | ] 26 | }, 27 | { 28 | "login": "JE-lee", 29 | "name": "lee", 30 | "avatar_url": "https://avatars0.githubusercontent.com/u/19794813?v=4", 31 | "profile": "https://github.com/JE-lee", 32 | "contributions": [ 33 | "code", 34 | "doc", 35 | "example" 36 | ] 37 | } 38 | ], 39 | "contributorsPerLine": 7 40 | } 41 | -------------------------------------------------------------------------------- /.browserslistrc: -------------------------------------------------------------------------------- 1 | > 1% 2 | last 2 versions 3 | -------------------------------------------------------------------------------- /.eslintrc.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | root: true, 3 | env: { 4 | node: true 5 | }, 6 | extends: ["plugin:vue/essential", "eslint:recommended", "@vue/prettier"], 7 | parserOptions: { 8 | parser: "babel-eslint" 9 | }, 10 | rules: { 11 | "no-console": process.env.NODE_ENV === "production" ? "error" : "off", 12 | "no-debugger": process.env.NODE_ENV === "production" ? "error" : "off" 13 | } 14 | }; 15 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/bug_report.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: 'Bug Report' 3 | about: '创建错误报告以帮助我们改进。' 4 | title: '[bug]' 5 | labels: 'bug' 6 | assignees: '' 7 | 8 | --- 9 | 10 | 13 | 14 | ## 问题描述 15 | 16 | 17 | ## 最小可复现仓库 18 | > 请创建最小可复现代码,并上传到你的 GitHub 仓库 19 | 20 | 21 | ## 预期的行为和实际行为 22 | 23 | 24 | ## 复现步骤,具体代码 25 | 26 | 27 | 28 | 29 | 30 | 31 | ## 相关环境信息 32 | - **操作系统**: 33 | - **Node 版本**: 34 | - **deep-ui 版本**: 35 | - **Vue 版本**: 36 | - **引用方式**: CDN / NPM ? -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/question.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Question 3 | about: 问我们一个关于这个项目的问题 4 | title: '[question]' 5 | labels: 'question' 6 | assignees: '' 7 | 8 | --- 9 | 10 | 11 | ## 你想知道什么? 12 | 13 | 14 | 你好,我想知道这个项目是否可以... 15 | 16 | ## 描述你考虑过的替代方案 17 | 18 | 19 | 我想... 20 | 21 | ## 其它 22 | 23 | 无 -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/suggestion.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Suggestion 3 | about: 为这个项目提出一个建议 4 | title: '[suggest]' 5 | labels: 'help wanted' 6 | assignees: '' 7 | 8 | --- 9 | 10 | ## 你建议我们做什么? 11 | 12 | 13 | 现在... 14 | 15 | 如果你能... 16 | 17 | 它将使... 18 | 19 | ## 其它 20 | 21 | 无 -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules 3 | /dist 4 | 5 | # local env files 6 | .env.local 7 | .env.*.local 8 | 9 | # Log files 10 | npm-debug.log* 11 | yarn-debug.log* 12 | yarn-error.log* 13 | 14 | # Editor directories and files 15 | .idea 16 | .vscode 17 | *.suo 18 | *.ntvs* 19 | *.njsproj 20 | *.sln 21 | *.sw? 22 | 23 | lib/* -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | language: node_js 2 | node_js: 3 | - lts/* 4 | install: 5 | - yarn install # npm ci 6 | script: npm run docs:deploy -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | The MIT License (MIT) 2 | 3 | Copyright (c) 2018-present, Vivint, Inc. 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in 13 | all copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 21 | THE SOFTWARE. -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # deep-ui 2 | 3 | [![Build Status](https://travis-ci.com/cjfff/deep-ui.svg?branch=master)](https://travis-ci.com/cjfff/deep-ui) 4 | [![NPM License](https://badgen.net/npm/license/@femessage/el-data-table)](https://github.com/cjfff/deep-ui/raw/master/LICENSE) 5 | [![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg)](https://github.com/cjfff/deep-ui/pulls) 6 | 7 | 8 | ## 介绍 9 | 10 | 每家公司/每个团队,在持续协作维护项目的过程中,总会产出一些较为通用性的组件,或者 UI 会设计一些样式组件。 11 | 12 | 但是如果放在项目里的话,大家其实都不会去翻文件看,那这时候有个文档记录一下这些通用组件的话,会让大家协作的意愿更强烈。:fire: 13 | 14 | 更重要的是,可以提高代码的复用性, 打包编译发布到 `npm` 上,告别多项目的复制粘贴。 :beers: 15 | 16 | 使用及其简单的方式 (vue-cli3 以及 vuepress) 维护 团队/个人的 组件库。:tada::tada::tada: 17 | 18 | ## Features 19 | 20 | - [x] 代码打包 📦 npm lib 即可 21 | - [x] 代码即文档,代码里面加注释即可解决文档问题 22 | - [x] `yarn new` 交互式创建新组件文件,不用再繁琐的写 `template` 23 | - [x] 按需加载 css [栗子仓库](https://github.com/cjfff/deep-ui-demo) 24 | - [ ] 添加 eslint 25 | - [ ] standard-version 26 | - [ ] 自动生成 release logs 27 | 28 | ## Project setup 29 | 30 | 由于 `vuepress` 依赖的是 `core-js` 是 2.x 版本,而 `vue-cli3` 中依赖的是 3.x 版本. 31 | 32 | 所以 `vuepress` 需要进行全局安装。 33 | 34 | ```js 35 | 36 | npm i -g vuepress core-js@2.6.11 37 | 38 | ``` 39 | 40 | ``` 41 | git clone 42 | 43 | yarn install 44 | 45 | yarn docs:dev // 启动开发模式 46 | 47 | yarn new // 创建新组件 48 | ``` 49 | 50 | ## 发布 51 | 52 | 现在还没有自动化... 53 | 54 | ```shell 55 | yarn pub 56 | 57 | # 更新 package.json 然后 58 | 59 | npm publish 60 | "pub": "yarn build:lib && yarn build:theme", 61 | ``` 62 | 63 | ### 参考链接 64 | - https://github.com/vue-styleguidist/vuepress-plugin-live 65 | - https://github.com/BuptStEve/markdown-it-vuese 66 | - https://github.com/shuidi-fed/vuese 67 | - https://markdown-it.github.io/markdown-it/ 68 | - https://github.com/koca/vue-prism-editor 69 | - https://juejin.im/post/5dd234635188254a1f44646a#heading-27 70 | - https://segmentfault.com/a/1190000015261753 71 | 72 | 73 | 74 | ## Contributors ✨ 75 | 76 | Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)): 77 | 78 | 79 | 80 | 81 | 82 | 83 | 84 | 85 | 86 |

cjfff

📖 🐛 💡 🤔 🚧 💻

lee

💻 📖 💡
87 | 88 | 89 | 90 | 91 | 92 | This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome! 93 | 94 | 95 | [![All Contributors](https://img.shields.io/badge/all_contributors-2-orange.svg?style=flat-square)](#contributors-) 96 | 97 | -------------------------------------------------------------------------------- /babel.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | presets: ["@vue/cli-plugin-babel/preset"] 3 | }; 4 | -------------------------------------------------------------------------------- /build/bin/gen-cssfile.js: -------------------------------------------------------------------------------- 1 | var fs = require("fs"); 2 | var path = require("path"); 3 | var Components = require("../../components.json"); 4 | var themes = ["theme-chalk"]; 5 | Components = Object.keys(Components); 6 | var basepath = path.resolve(__dirname, "../../src/"); 7 | 8 | function fileExists(filePath) { 9 | try { 10 | return fs.statSync(filePath).isFile(); 11 | } catch (err) { 12 | return false; 13 | } 14 | } 15 | 16 | themes.forEach(theme => { 17 | var isSCSS = theme !== "theme-default"; 18 | var indexContent = isSCSS 19 | ? '@import "./common/var.less";\n' 20 | : '@import "./base.css";\n'; 21 | Components.forEach(function(key) { 22 | if (["icon", "option", "option-group"].indexOf(key) > -1) return; 23 | var fileName = key + (isSCSS ? ".less" : ".css"); 24 | indexContent += '@import "./' + fileName + '";\n'; 25 | var filePath = path.resolve(basepath, theme, "src", fileName); 26 | if (!fileExists(filePath)) { 27 | fs.writeFileSync(filePath, "", "utf8"); 28 | console.log(theme, " 创建遗漏的 ", fileName, " 文件"); 29 | } 30 | }); 31 | fs.writeFileSync( 32 | path.resolve(basepath, theme, "src", isSCSS ? "index.less" : "index.css"), 33 | indexContent 34 | ); 35 | }); 36 | -------------------------------------------------------------------------------- /build/docs-deploy.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env sh 2 | 3 | # 确保脚本抛出遇到的错误 4 | set -e 5 | 6 | # 安装 vuepress 和 core-js 依赖 7 | npm i -g vuepress core-js@2.6.11 --registry=https://registry.npm.taobao.org 8 | 9 | # 生成静态文件 10 | npm run docs:build 11 | 12 | 13 | mkdir temp_web 14 | cd temp_web 15 | 16 | 17 | git clone --depth 1 -b gh-pages --single-branch https://${GITHUB_TOKEN}@github.com/cjfff/deep-ui.git && cd deep-ui 18 | 19 | rm -rf ./deep-ui/* 20 | 21 | cp -rf ../../dist/* . 22 | 23 | git add -A . 24 | git commit -m "deploy" 25 | git push origin gh-pages 26 | 27 | cd ../../ 28 | rm -rf temp_web 29 | 30 | -------------------------------------------------------------------------------- /components.json: -------------------------------------------------------------------------------- 1 | { 2 | "text-tooltip": "./components/text-tooltip/index.js", 3 | "search-input": "./components/search-input/index.js", 4 | "list-container": "./components/list-container/index.js", 5 | "deepexi-card": "./components/deepexi-card/index.js", 6 | "loading-dialog": "./components/loading-dialog/index.js", 7 | "markdown-view": "./components/markdown-view/index.js", 8 | "loading-button": "./components/loading-button/index.js" 9 | } -------------------------------------------------------------------------------- /docs/.vuepress/config.js: -------------------------------------------------------------------------------- 1 | const path = require('path'); 2 | const glob = require('glob'); 3 | 4 | const getPath = (path) => path.replace(/.*component\//, '') 5 | 6 | const getDocs = (dir = 'basic') => { 7 | 8 | const docs = glob 9 | .sync(path.resolve(__dirname, `../component/${dir}/*.md`)) 10 | .map(p => getPath(p)) 11 | 12 | return docs 13 | } 14 | 15 | module.exports = { 16 | title: 'DeepUi', 17 | base: `/deep-ui/`, 18 | dest: 'dist', 19 | description: 'deepexi.com 中台研发团队的组件库', 20 | port: 8083, 21 | head: [ 22 | ['link', { rel: 'stylesheet', href: 'https://cdn.jsdelivr.net/npm/vue-prism-editor@0.5.1/dist/VuePrismEditor.css' }], 23 | ['link', { rel: 'stylesheet', href: 'https://cdn.jsdelivr.net/npm/@femessage/element-ui@2.16.0/lib/theme-chalk/index.css' }], 24 | ], 25 | plugins: [ 26 | '@vuepress/last-updated', 27 | '@vuepress/back-to-top', 28 | [ 29 | 'live', 30 | { 31 | // to use a custom layout for your vue components 32 | layout: path.resolve(__dirname, './layout.vue'), 33 | editorProps: { 34 | lineNumbers: true, 35 | } 36 | } 37 | ], 38 | ], 39 | configureWebpack: { 40 | resolve: { 41 | alias: { 42 | '@component': path.resolve(__dirname, '../../src'), 43 | '@': path.resolve(__dirname, '../../src'), 44 | '~': path.resolve(__dirname, '../../src') 45 | } 46 | }, 47 | }, 48 | themeConfig: { 49 | nav: [ 50 | { text: '主页', link: '/' }, 51 | { text: '组件', link: '/component/guide/introduction' }, 52 | { text: 'GitHub', link: 'https://github.com/cjfff/deep-ui' }, 53 | ], 54 | sidebar: { 55 | '/component/': [ 56 | { 57 | title: '开发指南', 58 | collapsable: false, 59 | children: getDocs('guide') 60 | }, 61 | { 62 | title: '基础组件', 63 | collapsable: false, 64 | children: getDocs('basic') 65 | }, 66 | { 67 | title: '高阶组件', 68 | collapsable: false, 69 | children: getDocs('hoc') 70 | }, 71 | { 72 | title: '布局组件', 73 | collapsable: false, 74 | children: getDocs('layout') 75 | }, 76 | ] 77 | } 78 | }, 79 | markdown: { 80 | extendMarkdown: (md) => { 81 | md.use(require('markdown-it-vuese/src/index.js'), { 82 | root: `${process.cwd()}/src/components/`, 83 | useRender: (vueseRender) => { 84 | const renderRes = vueseRender.render() 85 | // 格式转换可以去这里查看详情 https://vuese.org/zh/markdown-render/#%E5%AE%9E%E4%BE%8B%E6%96%B9%E6%B3%95-render 86 | return Object.entries(renderRes).reduce((acc, [title, value]) => acc.concat(`## ${title}\r ${value}`) 87 | , []).join('\r') 88 | }, 89 | }) 90 | 91 | } 92 | } 93 | } -------------------------------------------------------------------------------- /docs/.vuepress/enhanceApp.js: -------------------------------------------------------------------------------- 1 | import FormRenderer from '@femessage/el-form-renderer'; 2 | import ELEMENT from '@femessage/element-ui' 3 | import DeepUi from '../../src/index.js' 4 | import '../../src/theme-chalk/src/index.less' 5 | 6 | export default ({ 7 | Vue, 8 | options, 9 | router 10 | }) => { 11 | Vue.use(ELEMENT) 12 | Vue.use(DeepUi) 13 | Vue.component(FormRenderer.name, FormRenderer) 14 | } -------------------------------------------------------------------------------- /docs/.vuepress/layout.vue: -------------------------------------------------------------------------------- 1 | 24 | 25 | 70 | 71 | 319 | -------------------------------------------------------------------------------- /docs/README.md: -------------------------------------------------------------------------------- 1 | --- 2 | home: true 3 | heroImage: 4 | actionText: 开始使用 → 5 | actionLink: /component/guide/introduction 6 | features: 7 | - title: 基于Vue2.X 8 | details: 享受 Vue + webpack 的开发体验,在 Markdown 中使用 Vue 组件,同时可以使用 Vue 来开发自定义主题。 9 | - title: 轻量级UI 10 | details: 该UI使用了非常轻量级的设计理念,视觉感官更强烈 11 | - title: 高可用性 12 | details: 良好的API接口设计,统一的使用习惯 13 | footer: MIT Licensed | Copyright © 2020-cjfff 14 | --- -------------------------------------------------------------------------------- /docs/component/README.md: -------------------------------------------------------------------------------- 1 | ## 阅读 -------------------------------------------------------------------------------- /docs/component/basic/deepexi-card.md: -------------------------------------------------------------------------------- 1 | --- 2 | title: deepexi-card 3 | --- 4 | # deepexi-card 5 | 6 | ## 基本使用 7 | 8 | ```vue live lineNumbers 9 | 17 | 18 | 47 | ``` 48 | 49 | <[vuese](deepexi-card/index.vue) 50 | -------------------------------------------------------------------------------- /docs/component/basic/list-container.md: -------------------------------------------------------------------------------- 1 | --- 2 | title: list-container 3 | --- 4 | # list-container 5 | 6 | 简单易用的形式实现 `tab` 点击切换效果,无论是配合动态组件渲染还是内容修改,都及其容易扩展! 7 | 8 | ## 基本使用 9 | 10 | ```vue live lineNumbers 11 | 22 | 23 | 51 | ``` 52 | 53 | 54 | 55 | ## slot header 56 | 57 | ```vue live 58 | 73 | 74 | 102 | ``` 103 | 104 | <[vuese](list-container/index.vue) 105 | -------------------------------------------------------------------------------- /docs/component/basic/markdown-view.md: -------------------------------------------------------------------------------- 1 | --- 2 | title: markdown-view 3 | --- 4 | # markdown-view 5 | 6 | 快速渲染 markdown 内容以及 toc 标题生成 7 | 8 | ## 基本使用 9 | 10 | ```vue live 11 | 16 | 17 | 259 | ``` 260 | 261 | 262 | <[vuese](markdown-view/index.vue) -------------------------------------------------------------------------------- /docs/component/guide/guide.md: -------------------------------------------------------------------------------- 1 | --- 2 | title: 安装 3 | --- 4 | # 开发指南 5 | 6 | 输入下列命令可以安装 deep-ui 7 | ``` 8 | npm install deep-ui --save 9 | ``` -------------------------------------------------------------------------------- /docs/component/guide/introduction.md: -------------------------------------------------------------------------------- 1 | --- 2 | title: 介绍 3 | --- 4 | # 介绍 5 | 6 | 由于 `vuepress` 依赖的是 `core-js` 是 2.x 版本,而 `vue-cli3` 中依赖的是 3.x 版本. 7 | 8 | 所以写文档的时候,需要你去全局安装一下 `vuepress` 以及 `core-js@2.6.11` 9 | 10 | ```js 11 | npm i -g vuepress core-js@2.6.11 12 | ``` -------------------------------------------------------------------------------- /docs/component/hoc/loading-button.md: -------------------------------------------------------------------------------- 1 | --- 2 | title: loading-button 3 | --- 4 | # loading-button 5 | 6 | 懒人专用,小写一个 `loading` 效果,传入 `promise` 就会在 `promise pending` 期间产生 `loading` 的效果,`resolve` 或者 `rejected` 后结束。 7 | 8 | ## 基本使用 9 | 10 | ```vue live 11 | 14 | 15 | 27 | ``` 28 | 29 | <[vuese](@/loading-button/index.vue) -------------------------------------------------------------------------------- /docs/component/hoc/loading-dialog.md: -------------------------------------------------------------------------------- 1 | --- 2 | title: loading-dialog 3 | --- 4 | # loading-dialog 5 | 6 | 该组件内置了 `el-footer`,以及发送请求自动对 `confirm` 按钮进行 `loading`, 从细节上减少您的工作量~ 7 | 8 | ```vue live lineNumbers 9 | 22 | 23 | 58 | ``` 59 | 60 | 61 | <[vuese](loading-dialog/index.vue) -------------------------------------------------------------------------------- /docs/component/hoc/prompt-form.md: -------------------------------------------------------------------------------- 1 | --- 2 | title: prompt-form 3 | --- 4 | # prompt-form 5 | 提供Vue.prototype.$promptDialog和Vue.prototype.$prompDrawer方法,用来向用户确认信息。 6 | 类似于window.prompt方法,让用户在弹窗中填写相关信息。好处是,不需要写那么多HTML模板。 7 | 8 | ## Arguments 9 | - [Array | Object] formOpt: 10 | - [el-form-renderer](https://github.com/FEMessage/el-form-renderer)组件接受的属性content, 11 | - 或者是el-form-render 可接受属性的对象。eg: { content: [], width: '170px'} 12 | - [String] title: 默认为'提示' 13 | - [Number] width: el-form-render的宽度, 默认是600 14 | 15 | ## 基本使用 16 | 17 | ```vue live 18 | 24 | 25 | 53 | ``` 54 | 55 | ## 提交信息到服务器 56 | 可以指定弹窗关闭前调用的函数,这个函数需要返回一个Promise. 只有当该Promise resolve之后,弹窗才会关闭。 57 | 你可以在这个函数提交信息到服务器。当然你也可以在关闭弹窗之后在后台提交到服务器,但是如果提交失败,用户就必须重新输入 58 | 之前的表单信息了。 59 | 60 | ```vue live 61 | 66 | 67 | 101 | ``` 102 | -------------------------------------------------------------------------------- /docs/component/hoc/search-input.md: -------------------------------------------------------------------------------- 1 | --- 2 | title: search-input 3 | --- 4 | # search-input 5 | 6 | 这是一个 UI 样式组件 7 | 8 | ```vue live lineNumbers 9 | 12 | 13 | 22 | ``` 23 | 24 | 25 | <[vuese](search-input/index.vue) -------------------------------------------------------------------------------- /docs/component/hoc/text-tooltip.md: -------------------------------------------------------------------------------- 1 | --- 2 | title: text-tooltip 3 | --- 4 | # text-tooltip 5 | 6 | 当一些内容固定宽度,然后需要超出宽度部分隐藏的时候,为了防止重复书写这种 css....特产出此组件 7 | 8 | ## 基本使用 9 | 10 | ```vue live lineNumbers 11 | 16 | ``` 17 | 18 | ## 多行截断 19 | 20 | ```vue live 21 | 26 | ``` 27 | 28 | <[vuese](text-tooltip/index.vue) 29 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "deep-ui", 3 | "version": "1.0.0", 4 | "private": false, 5 | "scripts": { 6 | "serve": "vue-cli-service serve", 7 | "build": "yarn lib", 8 | "lint": "vue-cli-service lint", 9 | "clean": "rimraf lib && rimraf src/*/lib", 10 | "build:theme": "node build/bin/gen-cssfile && gulp build --gulpfile src/theme-chalk/gulpfile.js && cp-cli src/theme-chalk/lib lib/theme-chalk", 11 | "build:lib": "vue-cli-service build --target lib --name deep-ui --dest lib src/index.js", 12 | "pub": "yarn build:lib && yarn build:theme", 13 | "docs:build": "vuepress build docs", 14 | "docs:dev": "vuepress dev docs", 15 | "docs:deploy": "bash build/docs-deploy.sh", 16 | "contributors:add": "all-contributors add", 17 | "contributors:generate": "all-contributors generate", 18 | "new": "plop" 19 | }, 20 | "description": "A Component Library for Vue.js.", 21 | "author": "cjfff ", 22 | "contributors": [ 23 | "https://github.com/cjfff" 24 | ], 25 | "repository": { 26 | "type": "git", 27 | "url": "https://github.com/cjfff/deep-ui.git" 28 | }, 29 | "homepage": "https://github.com/deep-ui/README.md", 30 | "files": [ 31 | "lib", 32 | "src" 33 | ], 34 | "keywords": [ 35 | "deep-ui", 36 | "vue", 37 | "components" 38 | ], 39 | "license": "MIT", 40 | "main": "lib/deep-ui.umd.min.js", 41 | "unpkg": "src/index.js", 42 | "style": "lib/theme-chalk/index.css", 43 | "devDependencies": { 44 | "@femessage/el-form-renderer": "^1.14.6", 45 | "@femessage/element-ui": "^2.16.0", 46 | "@vue/babel-preset-app": "^4.1.0", 47 | "@vue/cli-plugin-babel": "~4.2.0", 48 | "@vue/cli-plugin-eslint": "~4.2.0", 49 | "@vue/cli-plugin-router": "~4.2.0", 50 | "@vue/cli-service": "^4.2.3", 51 | "@vue/eslint-config-prettier": "^6.0.0", 52 | "all-contributors-cli": "^6.14.0", 53 | "babel-eslint": "^10.0.3", 54 | "cp-cli": "^2.0.0", 55 | "eslint": "^6.7.2", 56 | "eslint-plugin-prettier": "^3.1.1", 57 | "eslint-plugin-vue": "^6.1.2", 58 | "glob": "^7.1.6", 59 | "gulp": "^4.0.2", 60 | "gulp-autoprefixer": "^7.0.1", 61 | "gulp-cssmin": "^0.2.0", 62 | "gulp-less": "^4.0.1", 63 | "json-templater": "^1.2.0", 64 | "less": "^3.11.1", 65 | "less-loader": "^5.0.0", 66 | "markdown-it-vuese": "^0.4.0", 67 | "plop": "^2.6.0", 68 | "prettier": "^1.19.1", 69 | "vue": "^2.6.11", 70 | "vue-template-compiler": "^2.6.11", 71 | "vuepress-plugin-live": "^1.5.1" 72 | }, 73 | "dependencies": { 74 | "core-js": "3.x" 75 | } 76 | } -------------------------------------------------------------------------------- /plop-templates/component/css.hbs: -------------------------------------------------------------------------------- 1 | @import './common/var.less' -------------------------------------------------------------------------------- /plop-templates/component/export.hbs: -------------------------------------------------------------------------------- 1 | import {{ componentName }} from "./index.vue"; 2 | 3 | {{ componentName }}.install = function(Vue) { 4 | Vue.component({{ componentName }}.name, {{ componentName }}); 5 | }; 6 | 7 | export default {{ componentName }}; -------------------------------------------------------------------------------- /plop-templates/component/index.hbs: -------------------------------------------------------------------------------- 1 | {{#if template}} 2 | 5 | {{/if}} 6 | 7 | {{#if script}} 8 | 18 | {{/if}} 19 | -------------------------------------------------------------------------------- /plop-templates/component/md.hbs: -------------------------------------------------------------------------------- 1 | --- 2 | title: {{ name }} 3 | --- 4 | # {{ name }} 5 | 6 | ## 基本使用 7 | 8 | ```vue live 9 | 12 | ``` 13 | 14 | <[vuese]({{name}}/index.vue) -------------------------------------------------------------------------------- /plop-templates/component/prompt.js: -------------------------------------------------------------------------------- 1 | const { notEmpty, kebabCase, camelCase } = require("../utils.js"); 2 | 3 | module.exports = { 4 | description: "generate a component", 5 | prompts: [ 6 | { 7 | type: "input", 8 | name: "name", 9 | message: "请输入组件名字(请以驼峰命名)", 10 | validate: notEmpty("组件名字") 11 | }, 12 | { 13 | type: "list", 14 | name: "type", 15 | message: "请选择组件类型", 16 | choices: [ 17 | { 18 | name: "基础组件", 19 | value: "basic" 20 | }, 21 | { 22 | name: "高阶组件", 23 | value: "hoc" 24 | }, 25 | { 26 | name: "布局组件", 27 | value: "layout" 28 | } 29 | ] 30 | } 31 | ], 32 | actions: data => { 33 | const kebabName = kebabCase(data.name); 34 | const componentName = camelCase(data.name); 35 | 36 | const actions = [ 37 | { 38 | type: "add", 39 | path: `src/components/${kebabName}/index.vue`, 40 | templateFile: "plop-templates/component/index.hbs", 41 | data: { 42 | componentName, 43 | name: kebabName, 44 | template: true, 45 | script: true, 46 | style: true 47 | } 48 | }, 49 | { 50 | type: "add", 51 | path: `src/components/${kebabName}/index.js`, 52 | templateFile: "plop-templates/component/export.hbs", 53 | data: { 54 | componentName 55 | } 56 | }, 57 | { 58 | type: "add", 59 | path: `docs/component/${data.type}/${kebabName}.md`, 60 | templateFile: "plop-templates/component/md.hbs", 61 | data: { 62 | name: kebabName 63 | } 64 | }, 65 | { 66 | type: "add", 67 | path: `src/theme-chalk/src/${kebabName}.less`, 68 | templateFile: "plop-templates/component/css.hbs" 69 | }, 70 | { 71 | type: "writeJsonAndBuildEntry", 72 | data: { 73 | name: data.name 74 | } 75 | } 76 | ]; 77 | 78 | return actions; 79 | } 80 | }; 81 | -------------------------------------------------------------------------------- /plop-templates/utils.js: -------------------------------------------------------------------------------- 1 | exports.notEmpty = name => { 2 | return v => { 3 | if (!v || v.trim === "") { 4 | return `${name} 必须填写`; 5 | } 6 | return true; 7 | }; 8 | }; 9 | 10 | // 下划线转换驼峰 11 | function camelCase(name) { 12 | return name 13 | .replace(/[\_|-](\w)/g, function(all, letter) { 14 | return letter.toUpperCase(); 15 | }) 16 | .replace(/./, _ => _.toUpperCase()); 17 | } 18 | // 驼峰转换中划线 19 | function kebabCase(name) { 20 | return name.replace(/([A-Z])/g, "-$1").toLowerCase(); 21 | } 22 | 23 | exports.camelCase = camelCase; 24 | exports.kebabCase = kebabCase; 25 | -------------------------------------------------------------------------------- /plop-templates/writeJsonAndBuildEntry.js: -------------------------------------------------------------------------------- 1 | const fs = require("fs"); 2 | const path = require("path"); 3 | const render = require("json-templater/string"); 4 | const { kebabCase, camelCase } = require("./utils.js"); 5 | 6 | const componentsFileName = "components.json"; 7 | const fcomponentsFileNameFilePath = path.resolve( 8 | __dirname, 9 | `../${componentsFileName}` 10 | ); 11 | 12 | const INDEX_OUTPUT_PATH = path.resolve(__dirname, "../src/index.js"); 13 | 14 | const endOfLine = require("os").EOL; 15 | 16 | const IMPORT_TEMPLATE = `import {{name}} from "{{path}}";`; 17 | const INSTALL_COMPONENT_TEMPLATE = " {{name}}"; 18 | const MAIN_TEMPLATE = `/* Automatically generated by plop buildEntryData action */ 19 | 20 | {{componentsImport}} 21 | 22 | // 存储组件列表 23 | const components = [ 24 | {{componentNames}} 25 | ]; 26 | 27 | // 定义 install 方法,接收 Vue 作为参数。如果使用 use 注册插件,则所有的组件都将被注册 28 | const install = function(Vue) { 29 | // 判断是否安装 30 | if (install.installed) return; 31 | // 遍历注册全局组件 32 | components.forEach(component => Vue.component(component.name, component)); 33 | }; 34 | 35 | // 判断是否是直接引入文件 36 | if (typeof window !== "undefined" && window.Vue) { 37 | install(window.Vue); 38 | } 39 | 40 | export default { 41 | // 导出的对象必须具有 install,才能被 Vue.use() 方法安装 42 | install, 43 | // 以下是具体的组件列表 44 | {{componentNames}} 45 | }; 46 | `; 47 | module.exports = function({ name }) { 48 | const componentJson = require(fcomponentsFileNameFilePath); 49 | 50 | const kebName = kebabCase(name); 51 | 52 | componentJson[kebName] = `./components/${kebName}/index.js`; 53 | 54 | const componetsEntries = Object.entries(componentJson); 55 | 56 | const componentsImport = componetsEntries 57 | .map(([name, path]) => 58 | render(IMPORT_TEMPLATE, { 59 | name: camelCase(name), 60 | path 61 | }) 62 | ) 63 | .join(endOfLine); 64 | 65 | const componentNames = componetsEntries 66 | .map(([name]) => 67 | render(INSTALL_COMPONENT_TEMPLATE, { name: camelCase(name) }) 68 | ) 69 | .join(`,${endOfLine}`); 70 | 71 | fs.writeFileSync( 72 | fcomponentsFileNameFilePath, 73 | JSON.stringify(componentJson, null, " "), 74 | "utf-8" 75 | ); 76 | 77 | fs.writeFileSync( 78 | INDEX_OUTPUT_PATH, 79 | render(MAIN_TEMPLATE, { 80 | componentNames, 81 | componentsImport 82 | }), 83 | "utf-8" 84 | ); 85 | 86 | return "write component.json and index.js is ok"; 87 | }; 88 | -------------------------------------------------------------------------------- /plopfile.js: -------------------------------------------------------------------------------- 1 | const componentGenerator = require("./plop-templates/component/prompt"); 2 | const writeJsonAndBuildEntry = require("./plop-templates/writeJsonAndBuildEntry"); 3 | 4 | module.exports = plop => { 5 | // 增加一种 action 6 | plop.setActionType("writeJsonAndBuildEntry", writeJsonAndBuildEntry); 7 | 8 | plop.setGenerator("component", componentGenerator); 9 | }; 10 | -------------------------------------------------------------------------------- /public/demo.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | <%= htmlWebpackPlugin.options.title %> 12 | 13 | 14 | 17 |
18 | 19 | 20 | 21 | -------------------------------------------------------------------------------- /public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/cjfff/deep-ui/5efa3ef655d6d26bc298acc40d07da715d21c909/public/favicon.ico -------------------------------------------------------------------------------- /public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | <%= htmlWebpackPlugin.options.title %> 9 | 10 | 11 | 14 |
15 | 16 | 17 | 18 | -------------------------------------------------------------------------------- /public/test.html: -------------------------------------------------------------------------------- 1 | 2 | deepexi demo 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 |
11 | {{msg}} 12 | 13 |
14 | 15 | -------------------------------------------------------------------------------- /src/components/deepexi-card/index.js: -------------------------------------------------------------------------------- 1 | import DeepexiCard from "./index.vue"; 2 | 3 | DeepexiCard.install = function(Vue) { 4 | Vue.component(DeepexiCard.name, DeepexiCard); 5 | }; 6 | 7 | export default DeepexiCard; 8 | -------------------------------------------------------------------------------- /src/components/deepexi-card/index.vue: -------------------------------------------------------------------------------- 1 | 10 | 11 | 20 | -------------------------------------------------------------------------------- /src/components/list-container/index.js: -------------------------------------------------------------------------------- 1 | import ListContainer from "./index.vue"; 2 | 3 | ListContainer.install = function(Vue) { 4 | Vue.component(ListContainer.name, ListContainer); 5 | }; 6 | 7 | export default ListContainer; 8 | -------------------------------------------------------------------------------- /src/components/list-container/index.vue: -------------------------------------------------------------------------------- 1 | 25 | 26 | 115 | -------------------------------------------------------------------------------- /src/components/loading-button/index.js: -------------------------------------------------------------------------------- 1 | import LoadingButton from "./index.vue"; 2 | 3 | LoadingButton.install = function(Vue) { 4 | Vue.component(LoadingButton.name, LoadingButton); 5 | }; 6 | 7 | export default LoadingButton; 8 | -------------------------------------------------------------------------------- /src/components/loading-button/index.vue: -------------------------------------------------------------------------------- 1 | 12 | 13 | 39 | -------------------------------------------------------------------------------- /src/components/loading-dialog/index.js: -------------------------------------------------------------------------------- 1 | import LoadingDialog from "./index.vue"; 2 | 3 | LoadingDialog.install = function(Vue) { 4 | Vue.component(LoadingDialog.name, LoadingDialog); 5 | }; 6 | 7 | export default LoadingDialog; 8 | -------------------------------------------------------------------------------- /src/components/loading-dialog/index.vue: -------------------------------------------------------------------------------- 1 | 33 | 34 | 136 | -------------------------------------------------------------------------------- /src/components/loading-dialog/render.js: -------------------------------------------------------------------------------- 1 | export default { 2 | name: "render", 3 | functional: true, 4 | props: { 5 | render: Function, 6 | data: Object, 7 | node: Array 8 | }, 9 | render: (h, ctx) => { 10 | const params = { 11 | data: ctx.props.data 12 | }; 13 | const { render } = ctx.props; 14 | 15 | return typeof render === "object" ? render : render(h, params); 16 | } 17 | }; 18 | -------------------------------------------------------------------------------- /src/components/markdown-view/index.js: -------------------------------------------------------------------------------- 1 | export { default } from "./index.vue"; 2 | -------------------------------------------------------------------------------- /src/components/markdown-view/index.vue: -------------------------------------------------------------------------------- 1 | 37 | 38 | 326 | -------------------------------------------------------------------------------- /src/components/markdown-view/utils.js: -------------------------------------------------------------------------------- 1 | /* eslint-disable */ 2 | /** 3 | * 生成带有 id 的标题 DOM 4 | * @param {*} param0 5 | */ 6 | export const renderTitle = ({level, id, content}) => `${content}`; 7 | 8 | /** 9 | * 生成带有 height & width 的 img DOM 10 | * @param {*} param0 11 | */ 12 | export const renderImage = ({src, height, width}) => 13 | ``; 14 | 15 | /** 16 | * 生成随机值的 id 17 | * @param {*} name 18 | */ 19 | export const ganerateId = name => 20 | name + (Math.random().toString(16) + Date.now()).toString(16).slice(2); 21 | 22 | // 进行 23 | export const handleMenuReplace = (html, menuList) => 24 | menuList.reduce((content, {origin, htmlContent}) => content.replace(origin, htmlContent), html); 25 | 26 | // 批量对图片节点进行替换 27 | export const handleImageList = async (content, maxImageWidth) => { 28 | let imageList = []; 29 | 30 | try { 31 | imageList = await handleContentImage(content, maxImageWidth); // 批量获取图片宽高 32 | } catch (error) { 33 | console.error(error); 34 | return imageList; 35 | } 36 | 37 | return imageList.reduce( 38 | (html, {origin, src, height, width}) => html.replace(origin, renderImage({src, height, width})), 39 | content, 40 | ); 41 | }; 42 | 43 | /** 44 | * 获取图片高度宽度 45 | * @param {*} param0 46 | * @param {*} maxImageWidth 47 | */ 48 | export const getImageWidthAndHeight = ({src, dom}, maxImageWidth = 746) => { 49 | return new Promise(resolve => { 50 | const image = new Image(); 51 | image.referrerPolicy = 'no-referrer'; 52 | image.src = src; 53 | image.onload = () => { 54 | const {width, height} = image; 55 | const rate = width > maxImageWidth ? parseFloat((image.width / maxImageWidth).toFixed(2)) : 1; 56 | const isOrigin = rate === 1; 57 | resolve({ 58 | src, 59 | origin: dom, 60 | width: isOrigin ? width : maxImageWidth, 61 | height: isOrigin ? height : ~~(height / rate), 62 | }); 63 | }; 64 | }); 65 | }; 66 | 67 | export const handleContentImage = async (content, maxImageWidth = 746) => { 68 | const imgRegExp = //i; 69 | // 取得内容中所有的 img 标签字符 70 | 71 | const imgListReg = RegExp(imgRegExp.source, 'ig'); 72 | const imgList = content.match(imgListReg); 73 | 74 | if (!imgList) return []; 75 | 76 | // 这个正则是用来取得 src="" 里面的链接的 77 | const srcRegexp = RegExp(imgRegExp.source, 'i'); 78 | 79 | return Promise.all( 80 | imgList.map(imgItem => { 81 | const [dom, src] = srcRegexp.exec(imgItem); 82 | return getImageWidthAndHeight( 83 | { 84 | dom, 85 | src, 86 | }, 87 | maxImageWidth, 88 | ); 89 | }), 90 | ); 91 | }; 92 | 93 | export const handleMenuList = html => { 94 | const TITLE_REGEXP = /(.*)<\/h\1>/; 95 | 96 | // 匹配到所有的 title 97 | const titleListRegexep = RegExp(TITLE_REGEXP.source, 'g'); 98 | 99 | const titleList = html.match(titleListRegexep); 100 | 101 | const titleRegExp = RegExp(TITLE_REGEXP.source, 'i'); 102 | 103 | const menuList = (titleList || []).reduce((acc, item) => { 104 | const id = ganerateId('menu_'); 105 | 106 | const [str, level, _content] = titleRegExp.exec(item); 107 | 108 | // 只提取标签内的内容 109 | const content = _content.replace(/<[^>]+>/gi, '').replace(/ /g, ' '); 110 | 111 | if (level && content) { 112 | const htmlContent = renderTitle({level, content, id}); 113 | acc.push({ 114 | origin: str, 115 | htmlContent, 116 | level, 117 | content, 118 | id, 119 | }); 120 | } 121 | return acc; 122 | }, []); 123 | return menuList; 124 | }; 125 | 126 | /** 127 | * 节流,一段时间只运行一次 128 | * @param {*} fn 回调函数 129 | * @param {*} wait 130 | */ 131 | export const throttle = (fn, wait) => { 132 | let inThrottle, lastFn, lastTime; 133 | return function() { 134 | const context = this, 135 | args = arguments; 136 | if (!inThrottle) { 137 | fn.apply(context, args); 138 | lastTime = Date.now(); 139 | inThrottle = true; 140 | } else { 141 | clearTimeout(lastFn); 142 | lastFn = setTimeout(function() { 143 | if (Date.now() - lastTime >= wait) { 144 | fn.apply(context, args); 145 | lastTime = Date.now(); 146 | } 147 | }, Math.max(wait - (Date.now() - lastTime), 0)); 148 | } 149 | }; 150 | }; 151 | 152 | /** 153 | * 获取element元素对于target元素的相对高度 154 | * @param {*} element 155 | */ 156 | export function getElementViewTop(element, isRelative = false) { 157 | let actualTop = element.offsetTop; 158 | let current = element.offsetParent; 159 | if (!isRelative) { 160 | while (current !== null && current !== parent) { 161 | actualTop += current.offsetTop; 162 | current = current.offsetParent; 163 | } 164 | } 165 | let elementScrollTop = 0; 166 | if (document.compatMode == 'BackCompat') { 167 | elementScrollTop = document.body.scrollTop; 168 | } else { 169 | elementScrollTop = document.documentElement.scrollTop; 170 | } 171 | return actualTop - elementScrollTop; 172 | } 173 | -------------------------------------------------------------------------------- /src/components/prompt-form/drawer.vue: -------------------------------------------------------------------------------- 1 | 22 | 35 | 57 | -------------------------------------------------------------------------------- /src/components/prompt-form/gene-wrapper.js: -------------------------------------------------------------------------------- 1 | import Drawer from "./drawer"; // hack: 因为 el-drawer对超过屏幕的内容不会应用滚动条 2 | 3 | export function geneDialogWrapper(width = 600) { 4 | return { 5 | props: { 6 | visible: Boolean, 7 | title: String, 8 | sureLoading: Boolean 9 | }, 10 | methods: { 11 | proxyEvent(eventType, ...rest) { 12 | return () => this.$emit(eventType, rest); 13 | } 14 | }, 15 | render() { 16 | const footerStyle = { 17 | textAlign: "right" 18 | }; 19 | const handleUpdateVisible = val => { 20 | this.proxyEvent("update:visible", val)(); 21 | }; 22 | const visiListener = { 23 | on: { 24 | "update:visible": handleUpdateVisible 25 | } 26 | }; 27 | return ( 28 | 35 |
36 | {this.$slots.default} 37 |
38 |
39 | 取 消 40 | 45 | 确 定 46 | 47 |
48 |
49 | ); 50 | } 51 | }; 52 | } 53 | 54 | export function geneDrawerWrapper(width = 600) { 55 | return { 56 | props: { 57 | visible: Boolean, 58 | title: String 59 | }, 60 | components: { 61 | Drawer 62 | }, 63 | methods: { 64 | proxyEvent(eventType, ...rest) { 65 | return () => this.$emit(eventType, rest); 66 | }, 67 | sureHandler() { 68 | this.proxyEvent("sure")(); 69 | return Promise.resolve(); 70 | } 71 | }, 72 | render() { 73 | const _this = this; 74 | const spread = { 75 | on: { 76 | "update:visible"(val) { 77 | _this.proxyEvent("update:visible", val)(); 78 | } 79 | } 80 | }; 81 | return ( 82 | 90 |
91 | {this.$slots.default} 92 |
93 |
94 | ); 95 | } 96 | }; 97 | } 98 | -------------------------------------------------------------------------------- /src/components/prompt-form/index.js: -------------------------------------------------------------------------------- 1 | import Vue from "vue"; 2 | import { geneDialogWrapper, geneDrawerWrapper } from "./gene-wrapper"; 3 | let boxId = 0; 4 | 5 | // eslint-disable-next-line no-unused-vars 6 | function createPrompt(PromptWrapper, title, formOpt) { 7 | // formOpt的处理 8 | const formOpts = []; 9 | if (Array.isArray(formOpt.content)) { 10 | formOpts.push(formOpt); 11 | } 12 | const geneRef = i => `form${i}`; 13 | const geneVueOpt = (resolve, reject) => { 14 | return { 15 | data() { 16 | return { 17 | visible: false, 18 | id: `prompt-${boxId++}`, 19 | defaultLoading: false, 20 | sureLoading: false 21 | }; 22 | }, 23 | destroyed() { 24 | const child = document.getElementById(this.id); 25 | child && document.body.removeChild(child); 26 | }, 27 | methods: { 28 | prompt() { 29 | this.visible = true; 30 | }, 31 | shutdown() { 32 | this.visible = false; 33 | }, 34 | handleClosed() { 35 | this.$destroy(); 36 | // dialog 或者 drawer关闭的时候一律reject 37 | // 根据Promise的状态不可逆,这不会影响到已经resolve状态的Promise 38 | reject(new Error("cancel")); 39 | }, 40 | async handleSure() { 41 | let forms = []; 42 | 43 | try { 44 | if (formOpts.length <= 1) { 45 | const ref = this.$refs[geneRef(0)]; 46 | forms = ref.getFormValue(); 47 | await ref.validate(); 48 | } 49 | } catch { 50 | return; 51 | } 52 | 53 | if (typeof formOpt.beforeClose === "function") { 54 | this.sureLoading = true; 55 | await formOpt.beforeClose(forms).finally(() => { 56 | this.sureLoading = false; 57 | }); 58 | } 59 | resolve(forms); 60 | this.shutdown(); 61 | } 62 | }, 63 | mounted() { 64 | // TODO: 将rule validator函数绑定this到当前form 65 | /* formOpts.forEach((opt) => { 66 | (opt.content || []).forEach(item => { 67 | (item.rules || []).forEach(rule => { 68 | if (typeof rule.validator === 'function') { 69 | const validator = rule.validator 70 | rule.validator = (...rest) => { 71 | validator.call({}, ...rest); 72 | }; 73 | } 74 | }); 75 | }); 76 | }); */ 77 | }, 78 | components: { PromptWrapper }, 79 | render() { 80 | const _this = this; 81 | const visiListener = { 82 | on: { 83 | "update:visible"() { 84 | _this.shutdown(); 85 | } 86 | } 87 | }; 88 | const renderForm = (opt, ref) => { 89 | const { labelWidth = "120px", content, ...rest } = opt; 90 | return ( 91 | 99 | ); 100 | }; 101 | 102 | let body = null; 103 | // 单个表单 104 | if (formOpts.length <= 1) { 105 | body = renderForm(formOpts[0], geneRef(0)); 106 | } 107 | return ( 108 | 118 | {body} 119 | 120 | ); 121 | } 122 | }; 123 | }; 124 | return new Promise((resolve, reject) => { 125 | const box = document.createElement("div"); 126 | document.body.appendChild(box); 127 | // 实例化并且挂载 128 | const comp = (window.comp = new Vue(geneVueOpt(resolve, reject))); 129 | comp.$mount(box); 130 | 131 | comp.prompt(); 132 | // nextTick 确保el-form-render已经渲染 133 | Vue.nextTick(() => { 134 | const initDefaultValue = (opt, index) => { 135 | const ref = comp.$refs[geneRef(index)]; 136 | const defaultValue = opt.defaultValue; 137 | if (typeof defaultValue === "function") { 138 | comp.defaultLoading = true; 139 | defaultValue() 140 | .then(res => ref.updateForm(res)) 141 | .finally(() => (comp.defaultLoading = false)); 142 | } else { 143 | ref.updateForm(defaultValue || {}); 144 | } 145 | }; 146 | // 设置每个表单的初始值 147 | formOpts.forEach((form, index) => initDefaultValue(form, index)); 148 | }); 149 | }); 150 | } 151 | 152 | /** 153 | * 154 | * @param {*} formOpt | el-form-render 配置对象 或者是 155 | * { 156 | * beforeClose: fn, //点击确定之后调用的函数,必须得返回一个promise, promise resolve 后将会关闭对话框 157 | * content: el-form-render 配置对象, 158 | * defaultValue: 表单的初始值 159 | * } 160 | * @param {*} title 161 | */ 162 | export function promptDialog(formOpt, title, width) { 163 | const opt = Array.isArray(formOpt) ? { content: formOpt } : formOpt; 164 | return createPrompt(geneDialogWrapper(width), formOpt.title || title, opt); 165 | } 166 | 167 | /** 168 | * @param {*} formOpt | el-form-render 配置对象 或者是 169 | * { 170 | * beforeClose: fn, //点击确定之后调用的函数,必须得返回一个promise, promise resolve 后将会关闭对话框 171 | * content: el-form-render 配置对象, 172 | * defaultValue: 表单的初始值 173 | * } 174 | * @param {*} title 175 | */ 176 | export function promptDrawer(formOpt, title, width) { 177 | const opt = Array.isArray(formOpt) ? { content: formOpt } : formOpt; 178 | return createPrompt(geneDrawerWrapper(width), formOpt.title || title, opt); 179 | } 180 | -------------------------------------------------------------------------------- /src/components/search-input/index.js: -------------------------------------------------------------------------------- 1 | import SearchInput from "./index.vue"; 2 | 3 | SearchInput.install = function(Vue) { 4 | Vue.component(SearchInput.name, SearchInput); 5 | }; 6 | 7 | export default SearchInput; 8 | -------------------------------------------------------------------------------- /src/components/search-input/index.vue: -------------------------------------------------------------------------------- 1 | 12 | 13 | 50 | -------------------------------------------------------------------------------- /src/components/text-tooltip/index.js: -------------------------------------------------------------------------------- 1 | import textTooltip from "./index.vue"; 2 | 3 | textTooltip.install = function(Vue) { 4 | Vue.component(textTooltip.name, textTooltip); 5 | }; 6 | 7 | export default textTooltip; 8 | -------------------------------------------------------------------------------- /src/components/text-tooltip/index.vue: -------------------------------------------------------------------------------- 1 | 97 | -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | /* Automatically generated by plop buildEntryData action */ 2 | 3 | import TextTooltip from "./components/text-tooltip/index.js"; 4 | import SearchInput from "./components/search-input/index.js"; 5 | import ListContainer from "./components/list-container/index.js"; 6 | import DeepexiCard from "./components/deepexi-card/index.js"; 7 | import LoadingDialog from "./components/loading-dialog/index.js"; 8 | import MarkdownView from "./components/markdown-view/index.js"; 9 | import LoadingButton from "./components/loading-button/index.js"; 10 | import { promptDialog, promptDrawer } from "./components/prompt-form"; 11 | 12 | // 存储组件列表 13 | const components = [ 14 | TextTooltip, 15 | SearchInput, 16 | ListContainer, 17 | DeepexiCard, 18 | LoadingDialog, 19 | MarkdownView, 20 | LoadingButton 21 | ]; 22 | 23 | // 定义 install 方法,接收 Vue 作为参数。如果使用 use 注册插件,则所有的组件都将被注册 24 | const install = function(Vue) { 25 | // 判断是否安装 26 | if (install.installed) return; 27 | // 遍历注册全局组件 28 | components.forEach(component => Vue.component(component.name, component)); 29 | // prompt-form 方法 30 | Vue.prototype.$promptDialog = promptDialog; 31 | Vue.prototype.$promptDrawer = promptDrawer; 32 | }; 33 | 34 | // 判断是否是直接引入文件 35 | if (typeof window !== "undefined" && window.Vue) { 36 | install(window.Vue); 37 | } 38 | 39 | export default { 40 | // 导出的对象必须具有 install,才能被 Vue.use() 方法安装 41 | install, 42 | // 以下是具体的组件列表 43 | TextTooltip, 44 | SearchInput, 45 | ListContainer, 46 | DeepexiCard, 47 | LoadingDialog, 48 | MarkdownView, 49 | LoadingButton 50 | }; 51 | -------------------------------------------------------------------------------- /src/mixins/emitter.js: -------------------------------------------------------------------------------- 1 | function broadcast(componentName, eventName, params) { 2 | this.$children.forEach(child => { 3 | const name = child.$options.name; 4 | 5 | if (name === componentName) { 6 | child.$emit.apply(child, [eventName].concat(params)); 7 | } else { 8 | broadcast.apply(child, [componentName, eventName].concat([params])); 9 | } 10 | }); 11 | } 12 | export default { 13 | methods: { 14 | dispatch(componentName, eventName, params) { 15 | let parent = this.$parent || this.$root; 16 | let name = parent.$options.name; 17 | 18 | while (parent && (!name || name !== componentName)) { 19 | parent = parent.$parent; 20 | 21 | if (parent) { 22 | name = parent.$options.name; 23 | } 24 | } 25 | if (parent) { 26 | parent.$emit.apply(parent, [eventName].concat(params)); 27 | } 28 | }, 29 | broadcast(componentName, eventName, params) { 30 | broadcast.call(this, componentName, eventName, params); 31 | } 32 | } 33 | }; 34 | -------------------------------------------------------------------------------- /src/theme-chalk/gulpfile.js: -------------------------------------------------------------------------------- 1 | "use strict"; 2 | 3 | const { series, src, dest } = require("gulp"); 4 | const less = require("gulp-less"); 5 | const autoprefixer = require("gulp-autoprefixer"); 6 | const cssmin = require("gulp-cssmin"); 7 | 8 | function compile() { 9 | return src("./src/*.less") 10 | .pipe( 11 | less({ 12 | plugins: [ 13 | autoprefixer({ 14 | browsers: ["ie > 9", "last 2 versions"], 15 | cascade: false 16 | }) 17 | ] 18 | }) 19 | ) 20 | .pipe(cssmin()) 21 | .pipe(dest("./lib")); 22 | } 23 | 24 | // function copyfont() { 25 | // return src("./src/fonts/**") 26 | // .pipe(cssmin()) 27 | // .pipe(dest("./lib/fonts")); 28 | // } 29 | 30 | exports.build = series(compile); 31 | -------------------------------------------------------------------------------- /src/theme-chalk/src/common/markdown-preview.less: -------------------------------------------------------------------------------- 1 | @margin: 8px 0; 2 | //主体颜色 3 | @primary: #292d35; 4 | @light-primary: #323741; 5 | @dark-primary: #1c1e24; 6 | // 7 | @info: #007acc; 8 | @success: #19be6b; 9 | @warning: #ff9900; 10 | @error: #ed3f14; 11 | // 12 | @title: #252525; 13 | @content: #555; 14 | @sub-color: #80848f; 15 | @disabled: #bbbec4; 16 | @border: #dddee1; 17 | @divider: #e9eaec; 18 | @background: #f7f7f7; 19 | @tip: #c1c1c1; 20 | @primary-rgba: rgba(49, 204, 102, 0.2); 21 | @mask: rgba(0, 0, 0, 0.3); 22 | @blue-rgba: rgba(0, 122, 204, 0.2); 23 | @dark-info: #0169af; 24 | @line-height: 1.8rem; 25 | 26 | 27 | pre { 28 | padding: 20px 20px 30px !important; 29 | display: block; 30 | color: #abb2bf; 31 | font-family: Menlo, Consolas, "Courier New", Courier, FreeMono, monospace; 32 | background: #292c34; 33 | border-radius: 4px; 34 | overflow-y: hidden !important; 35 | overflow-x: auto !important; 36 | margin: 20px 0 !important; 37 | 38 | * { 39 | line-height: 1.6 !important; 40 | font-size: 14px; 41 | font-family: Menlo, Consolas, "Courier New", Courier, FreeMono, 42 | monospace; 43 | } 44 | } 45 | 46 | .hljs-comment, 47 | .hljs-quote { 48 | color: #5c6370; 49 | font-style: italic; 50 | } 51 | 52 | .hljs-doctag, 53 | .hljs-formula, 54 | .hljs-keyword { 55 | color: #c678dd; 56 | } 57 | 58 | .hljs-deletion, 59 | .hljs-name, 60 | .hljs-section, 61 | .hljs-selector-tag, 62 | .hljs-subst { 63 | color: #e06c75; 64 | } 65 | 66 | .hljs-literal { 67 | color: #56b6c2; 68 | } 69 | 70 | .hljs-addition, 71 | .hljs-attribute, 72 | .hljs-meta-string, 73 | .hljs-regexp, 74 | .hljs-string { 75 | color: #98c379; 76 | } 77 | 78 | .hljs-built_in, 79 | .hljs-class .hljs-title { 80 | color: #e6c07b; 81 | } 82 | 83 | .hljs-attr, 84 | .hljs-number, 85 | .hljs-selector-attr, 86 | .hljs-selector-class, 87 | .hljs-selector-pseudo, 88 | .hljs-template-variable, 89 | .hljs-type, 90 | .hljs-variable { 91 | color: #d19a66; 92 | } 93 | 94 | .hljs-bullet, 95 | .hljs-link, 96 | .hljs-meta, 97 | .hljs-selector-id, 98 | .hljs-symbol, 99 | .hljs-title { 100 | color: #61aeee; 101 | } 102 | 103 | .hljs-emphasis { 104 | font-style: italic; 105 | } 106 | 107 | .hljs-strong { 108 | font-weight: bold; 109 | } 110 | 111 | .hljs-link { 112 | text-decoration: underline; 113 | } 114 | 115 | .markdown-preview { 116 | flex: 1; 117 | overflow: hidden; 118 | background: #fff; 119 | // padding: 20px 12px !important; 120 | font-family: "Open Sans", 121 | "Helvetica Neue", 122 | Helvetica, 123 | Arial, 124 | sans-serif; 125 | 126 | 127 | >div { 128 | padding: 10px 12px !important; 129 | background: #fff; 130 | 131 | &::-webkit-scrollbar { 132 | display: none; 133 | } 134 | } 135 | 136 | &::-webkit-scrollbar { 137 | display: none; 138 | } 139 | 140 | ul { 141 | list-style: none; 142 | padding: 0 20px; 143 | 144 | li { 145 | position: relative; 146 | 147 | &:after { 148 | display: block; 149 | content: ""; 150 | width: 8px; 151 | height: 8px; 152 | border-radius: 50%; 153 | position: absolute; 154 | z-index: 99; 155 | top: 7px; 156 | left: -20px; 157 | background: @content; 158 | } 159 | } 160 | } 161 | 162 | ol, 163 | ul { 164 | margin: 20px 0; 165 | padding: 0 40px; 166 | 167 | li { 168 | font-size: 14px !important; 169 | color: @content; 170 | margin-bottom: 10px; 171 | line-height: 24px; 172 | padding-left: 12px; 173 | 174 | input[type="checkbox"] { 175 | position: relative; 176 | cursor: pointer; 177 | overflow: visible; 178 | position: absolute; 179 | left: 0; 180 | top: 0; 181 | 182 | &:after { 183 | display: block; 184 | content: ""; 185 | width: 16px; 186 | height: 16px; 187 | position: absolute; 188 | z-index: 99999; 189 | background: #fff; 190 | top: 0; 191 | right: 0; 192 | } 193 | 194 | &:before { 195 | display: block; 196 | width: 18px; 197 | height: 18px; 198 | position: absolute; 199 | content: ""; 200 | top: 2px; 201 | left: -25px; 202 | z-index: 999999; 203 | background-position: center; 204 | background: url("") no-repeat; 205 | background-size: contain; 206 | } 207 | } 208 | 209 | input[type="checkbox"]:checked { 210 | &:before { 211 | background: url("") no-repeat; 212 | background-size: contain; 213 | } 214 | } 215 | } 216 | } 217 | 218 | ol { 219 | list-style-type: decimal; 220 | } 221 | 222 | hr { 223 | color: @border; 224 | height: 1px; 225 | border: 0; 226 | border-top: 1px solid @border; 227 | margin: 20px 0; 228 | padding: 0; 229 | } 230 | 231 | del, 232 | em, 233 | strong { 234 | display: inline-block; 235 | margin: @margin; 236 | } 237 | 238 | blockquote { 239 | position: relative; 240 | background: @background; 241 | padding: 6px 12px; 242 | border-left: 5px solid @divider; 243 | border-radius: 2px; 244 | margin: @margin; 245 | } 246 | 247 | /*基本样式*/ 248 | 249 | h1, 250 | h2, 251 | h3, 252 | h4, 253 | h5, 254 | h6 { 255 | color: @title; 256 | } 257 | 258 | h1 { 259 | font-size: 28px; 260 | border-bottom: 1px solid @border; 261 | //text-align: center; 262 | } 263 | 264 | h2 { 265 | font-size: 24px; 266 | } 267 | 268 | h3 { 269 | font-size: 18px; 270 | } 271 | 272 | h4 { 273 | font-size: 16px; 274 | } 275 | 276 | h5 { 277 | font-size: 14px; 278 | } 279 | 280 | h6 { 281 | font-size: 12px; 282 | } 283 | 284 | h1, 285 | h2, 286 | h3, 287 | h4, 288 | h5, 289 | h6 { 290 | /* border-bottom: 1px solid @border; */ 291 | padding: 20px 0; 292 | font-weight: 600; 293 | } 294 | 295 | p { 296 | font-size: 14px !important; 297 | color: @content; 298 | margin: @margin; 299 | line-height: @line-height; 300 | } 301 | 302 | img { 303 | display: block; 304 | width: 90%; 305 | cursor: pointer; 306 | } 307 | 308 | table { 309 | width: 100%; 310 | border: 1px solid @border; 311 | border-bottom: 0; 312 | background: #fff; 313 | border-spacing: 0; 314 | border-collapse: collapse; 315 | margin: 20px 0; 316 | 317 | tr { 318 | -webkit-transition: background 0.1s; 319 | transition: background 0.1s; 320 | text-align: nav; 321 | } 322 | 323 | tr td, 324 | tr th { 325 | padding: 0 8px; 326 | font-size: 14px; 327 | line-height: 39px; 328 | color: #333; 329 | border-bottom: 1px solid @border; 330 | cursor: pointer; 331 | } 332 | 333 | th { 334 | background: #f8f8f9; 335 | text-align: left; 336 | font-weight: bold; 337 | } 338 | 339 | tr:nth-of-type(even) { 340 | td { 341 | background: #f8f8f9; 342 | } 343 | } 344 | 345 | tr { 346 | &:hover { 347 | td { 348 | background: #eaf5f6; 349 | } 350 | } 351 | } 352 | 353 | td, 354 | th { 355 | border: 1px solid @border; 356 | } 357 | } 358 | 359 | input[type="checkbox"] { 360 | display: inline-block; 361 | border-radius: 0; 362 | margin-right: 8px; 363 | } 364 | 365 | a { 366 | text-decoration: none; 367 | color: @info; 368 | font-size: 14px; 369 | line-height: @line-height; 370 | } 371 | } 372 | -------------------------------------------------------------------------------- /src/theme-chalk/src/common/var.less: -------------------------------------------------------------------------------- 1 | /* 2 | 支持自动引入less变量表,使用时无需手动引入 3 | */ 4 | 5 | @primary-color: #5D81F9; 6 | // base color 7 | /// color|1|BrandColor|0 8 | @--color-primary: #5D81F9; // 9 | /// color|1|SecondaryColor|1 10 | @--color-success: #02C5E2; 11 | /// color|1|SecondaryColor|1 12 | @--color-warning: #F5A623; 13 | /// color|1|SecondaryColor|1 14 | @--color-danger: #E24156; 15 | /// color|1|SecondaryColor|1 16 | @--color-info: #6A6E7B; 17 | 18 | /// color|1|FontColor|2 19 | @--color-text-primary: #2D303B; 20 | /// color|1|FontColor|2 21 | @--color-text-regular: #2D303B; 22 | /// color|1|FontColor|2 23 | @--color-text-secondary: #93959b; 24 | /// color|1|FontColor|2 25 | @--color-text-placeholder: #9CA6C7; 26 | /// color|1|BorderColor|3 27 | @--border-color-base: #CAD1E8; 28 | /// color|1|BorderColor|3 29 | @--border-color-light: #CAD1E8; 30 | /// color|1|BorderColor|3 31 | @--border-color-lighter: #EBEEF5; 32 | /// color|1|BorderColor|3 33 | @--border-color-extra-light: #F2F6FC; 34 | 35 | @--font-color-disabled-base: #bbb; 36 | 37 | // Background 38 | /// color|1|BackgroundColor|4 39 | @--background-color-base: #f5f7fa; 40 | 41 | //sidebar 42 | @menuText: rgba(255, 255, 255, 1); 43 | @menuFontColor: rgba(255, 255, 255, 0.7); 44 | @menuActiveText: #fff; 45 | @subMenuActiveText:#f4f4f5; //https://github.com/ElemeFE/element/issues/12951 46 | 47 | @menuBg: rgba(51, 54, 67, 1); 48 | 49 | @subMenuBg:#1f2d3d; 50 | @subMenuHover:#001528; 51 | 52 | @sideBarMaxWidth: 200px; // 侧边栏的最大宽度 53 | @sideBarMinWidth: 64px; // 侧边栏的最小宽度 54 | 55 | @padding-size: 24px; 56 | @margin-size: 16px; 57 | 58 | @cyan: #02C5E2; 59 | @black: #2D303B; 60 | @white: #fff; 61 | @gray: #9CA6C7; 62 | @error: #E24156; 63 | 64 | @font-l: 16px; 65 | @font-m: 14px; 66 | @font-s: 12px; 67 | 68 | @border-color: #CAD1E8; 69 | @bg-color: #F0F2F5; 70 | @bg-code: #f6f8fa; 71 | 72 | @basic-font-family: PingFangSC-Regular; 73 | @font-weight: 400; 74 | .setFontHidden { 75 | // 字体超出显示省略号 76 | white-space: nowrap; 77 | overflow: hidden; 78 | text-overflow: ellipsis; 79 | } 80 | -------------------------------------------------------------------------------- /src/theme-chalk/src/deepexi-card.less: -------------------------------------------------------------------------------- 1 | @import './common/var'; 2 | 3 | .deepexi-card { 4 | background: #fff; 5 | border-radius: 4px; 6 | & + .deepexi-card { 7 | margin-top: @margin-size; 8 | } 9 | &-title { 10 | height: 48px; 11 | line-height: 48px; 12 | border-bottom: 1px solid #e4e8f3; 13 | padding: 0 @padding-size; 14 | font-size: 14px; 15 | font-weight: 500; 16 | color: rgba(106, 108, 115, 1); 17 | } 18 | &-content { 19 | padding: @padding-size; 20 | } 21 | } -------------------------------------------------------------------------------- /src/theme-chalk/src/index.less: -------------------------------------------------------------------------------- 1 | @import "./common/var.less"; 2 | @import "./text-tooltip.less"; 3 | @import "./search-input.less"; 4 | @import "./list-container.less"; 5 | @import "./deepexi-card.less"; 6 | @import "./loading-dialog.less"; 7 | @import "./markdown-view.less"; 8 | @import "./loading-button.less"; 9 | -------------------------------------------------------------------------------- /src/theme-chalk/src/list-container.less: -------------------------------------------------------------------------------- 1 | @import './common/var'; 2 | 3 | .list-container { 4 | @lineHeight: 60px; 5 | 6 | &-header { 7 | position: relative; 8 | display: flex; 9 | border-bottom: 1px solid #eee; 10 | background: #fff; 11 | overflow-x: auto; 12 | 13 | .tab-active-bar { 14 | position: absolute; 15 | bottom: 0; 16 | left: 0; 17 | z-index: 1; 18 | width: 80px; 19 | height: 2px; 20 | background: @primary-color; 21 | 22 | &.animation { 23 | transition: 0.3s cubic-bezier(0.645, 0.045, 0.355, 1); // element.ui tabbar 的动画 24 | } 25 | } 26 | 27 | .tab-like-box { 28 | list-style: none; 29 | padding: 0; 30 | font-weight: 400; 31 | display: inherit; 32 | width: auto; 33 | white-space: nowrap; 34 | 35 | .tab { 36 | padding: 0 16px; 37 | line-height: @lineHeight; 38 | display: inline-block; 39 | cursor: pointer; 40 | color: #000; 41 | 42 | &.is-active { 43 | color: @primary-color; 44 | } 45 | } 46 | } 47 | 48 | .right-part { 49 | line-height: @lineHeight; 50 | } 51 | } 52 | } -------------------------------------------------------------------------------- /src/theme-chalk/src/loading-button.less: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/cjfff/deep-ui/5efa3ef655d6d26bc298acc40d07da715d21c909/src/theme-chalk/src/loading-button.less -------------------------------------------------------------------------------- /src/theme-chalk/src/loading-dialog.less: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/cjfff/deep-ui/5efa3ef655d6d26bc298acc40d07da715d21c909/src/theme-chalk/src/loading-dialog.less -------------------------------------------------------------------------------- /src/theme-chalk/src/markdown-view.less: -------------------------------------------------------------------------------- 1 | @import './common/var'; 2 | @import "./common/markdown-preview.less"; 3 | 4 | .mardkdown-view { 5 | position: relative; 6 | background: #fff; 7 | display: flex; 8 | 9 | .left-content { 10 | width: 100%; 11 | min-width: 300px; 12 | margin-right: 80px; 13 | 14 | .markdown-preview { 15 | h1 { 16 | border-bottom: none; 17 | } 18 | } 19 | } 20 | 21 | .right-content { 22 | width: 230px; 23 | } 24 | 25 | .deepexi-scrollbar-wrap { 26 | position: relative; 27 | transform: translateX(-20px); 28 | 29 | &.fixed { 30 | top: 0; 31 | right: 0; 32 | position: fixed; 33 | } 34 | } 35 | 36 | .deepexi-scrollbar { 37 | overflow-y: auto; 38 | width: 200px; 39 | 40 | &::-webkit-scrollbar { 41 | width: 4px; 42 | height: 20px; 43 | } 44 | 45 | &::-webkit-scrollbar-track { 46 | box-shadow: inset 0 0 6px #e5e5e5; 47 | border-radius: 5px; 48 | } 49 | 50 | &::-webkit-scrollbar-thumb { 51 | border-radius: 5px; 52 | box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.2); 53 | } 54 | } 55 | 56 | .fe-menu { 57 | list-style: none; 58 | position: relative; 59 | padding: 0; 60 | 61 | &-item { 62 | cursor: pointer; 63 | position: relative; 64 | line-height: 2; 65 | font-size: 12px; 66 | width: 100%; 67 | overflow: hidden; /* 超出部分隐藏 */ 68 | white-space: nowrap; 69 | text-overflow: ellipsis; /* 适用IE */ 70 | padding: 0 10px; 71 | transition: color 0.5s; 72 | border-left: 2px solid #e8e8e8; 73 | 74 | &:hover { 75 | opacity: 0.8; 76 | } 77 | } 78 | } 79 | } 80 | -------------------------------------------------------------------------------- /src/theme-chalk/src/search-input.less: -------------------------------------------------------------------------------- 1 | .search-input { 2 | width : 240px; 3 | z-index: 10; 4 | 5 | .search-icon /deep/ { 6 | display : inline-block; 7 | width : 14px; 8 | margin-left: 5px; 9 | background : url("https://deepexi.oss-cn-shenzhen.aliyuncs.com/deepexi-services-dashboard/common/search.svg") center / contain no-repeat; 10 | } 11 | 12 | .el-icon-search:before { 13 | font-size: 20px; 14 | color : #eee; 15 | } 16 | 17 | .el-input__inner { 18 | padding-left: 34px; 19 | border : none; 20 | border : 1px solid #eee; 21 | 22 | &:focus { 23 | border: 1px solid #bcc3d9; 24 | } 25 | } 26 | } -------------------------------------------------------------------------------- /src/theme-chalk/src/text-tooltip.less: -------------------------------------------------------------------------------- 1 | .text-tooltip { 2 | display : inline-block; 3 | overflow: hidden; 4 | width : 100%; 5 | 6 | &.ellipsis { 7 | text-overflow: ellipsis; 8 | white-space : nowrap; 9 | } 10 | } -------------------------------------------------------------------------------- /src/utils/assist.js: -------------------------------------------------------------------------------- 1 | export function oneOf(value, validList) { 2 | for (let i = 0; i < validList.length; i++) { 3 | if (value === validList[i]) { 4 | return true; 5 | } 6 | } 7 | return false; 8 | } 9 | -------------------------------------------------------------------------------- /vue.config.js: -------------------------------------------------------------------------------- 1 | const path = require("path"); 2 | 3 | const buildMode = process.env.BUILD_MODE; 4 | 5 | console.log(buildMode); 6 | 7 | module.exports = { 8 | productionSourceMap: false, 9 | pages: { 10 | index: { 11 | entry: "src/index.js", 12 | template: "public/index.html", 13 | filename: "index.html" 14 | } 15 | }, 16 | configureWebpack: { 17 | externals: { 18 | "@femessage/element-ui": "Element", 19 | vue: { 20 | root: "Vue", 21 | commonjs: "vue", 22 | commonjs2: "vue", 23 | amd: "vue" 24 | } 25 | } 26 | }, 27 | chainWebpack: config => { 28 | if (process.env.NODE_ENV === "development") { 29 | config.devtool = "source-map"; 30 | } 31 | } 32 | }; 33 | --------------------------------------------------------------------------------