clear commit
15
.editorconfig
Normal file
@ -0,0 +1,15 @@
|
||||
# http://editorconfig.org
|
||||
|
||||
root = true
|
||||
|
||||
[*]
|
||||
charset = utf-8
|
||||
indent_style = space
|
||||
indent_size = 4
|
||||
end_of_line = lf
|
||||
insert_final_newline = true
|
||||
trim_trailing_whitespace = true
|
||||
|
||||
[*.md]
|
||||
insert_final_newline = false
|
||||
trim_trailing_whitespace = false
|
20
.eslintrc.js
Normal file
@ -0,0 +1,20 @@
|
||||
|
||||
module.exports = {
|
||||
extends: [
|
||||
'@webank/eslint-config-webank/vue',
|
||||
],
|
||||
globals: {
|
||||
// 这里填入你的项目需要的全局变量
|
||||
// 这里值为 false 表示这个全局变量不允许被重新赋值,比如:
|
||||
//
|
||||
// Vue: false
|
||||
},
|
||||
rules: {
|
||||
'no-plusplus': 'off',
|
||||
'no-bitwise': 'off',
|
||||
'vue/comment-directive': 'off',
|
||||
'no-param-reassign': 'off',
|
||||
'func-names': 'off',
|
||||
'class-methods-use-this': 'off'
|
||||
}
|
||||
};
|
10
.gitignore
vendored
Normal file
@ -0,0 +1,10 @@
|
||||
.DS_Store
|
||||
.idea
|
||||
.git
|
||||
.vscode
|
||||
.history
|
||||
.cache
|
||||
node_modules
|
||||
npm-debug.log
|
||||
/packages/fes-template/dist
|
||||
/packages/fes-doc/docs/.vuepress/dist
|
21
LICENSE
Normal file
@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2020-present harrywan
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
42
README.md
Normal file
@ -0,0 +1,42 @@
|
||||
# Fes.js
|
||||
|
||||
Fes.js 是一个管理台应用解决方案,提供初始项目、开发调试、编译打包的命令行工具,内置布局、权限、数据字典、状态管理、Api等多个模块,文件目录结构即路由,用户只需要编写页面内容。基于Vue.js,内置管理台常用能力,让用户写的更少,更简单。经过多个项目中打磨,趋于稳定。
|
||||
|
||||
## 页面举例
|
||||
|
||||
首页:
|
||||

|
||||
|
||||
表单页面:
|
||||

|
||||
|
||||
列表页面:
|
||||

|
||||
|
||||
列表页面:
|
||||

|
||||
|
||||
列表页面:
|
||||

|
||||
|
||||
## 使用
|
||||
|
||||
```bash
|
||||
# install
|
||||
npm install @webank/fes-cli -g
|
||||
|
||||
# create a project
|
||||
fes init [project]
|
||||
|
||||
cd [project] && npm i
|
||||
|
||||
# start dev
|
||||
npm run dev
|
||||
|
||||
# build
|
||||
npm run build
|
||||
```
|
||||
|
||||
## 文档
|
||||
|
||||
详细使用请查看[文档](https://webankfintech.github.io/fes.js/)
|
BIN
images/form.png
Normal file
After Width: | Height: | Size: 88 KiB |
BIN
images/home.png
Normal file
After Width: | Height: | Size: 188 KiB |
BIN
images/list.png
Normal file
After Width: | Height: | Size: 692 KiB |
BIN
images/listAndFrom.png
Normal file
After Width: | Height: | Size: 136 KiB |
BIN
images/listAndImg.png
Normal file
After Width: | Height: | Size: 324 KiB |
7
lerna.json
Normal file
@ -0,0 +1,7 @@
|
||||
{
|
||||
"packages": [
|
||||
"packages/*"
|
||||
],
|
||||
"version": "independent",
|
||||
"npmClient": "npm"
|
||||
}
|
27
package.json
Normal file
@ -0,0 +1,27 @@
|
||||
{
|
||||
"name": "fes",
|
||||
"description": "一个好用的前端管理台快速开发框架",
|
||||
"preferGlobal": true,
|
||||
"scripts": {
|
||||
"bootstrap": "lerna bootstrap"
|
||||
},
|
||||
"author": "harrywan qlin",
|
||||
"license": "MIT",
|
||||
"keywords": [
|
||||
"管理端",
|
||||
"fes",
|
||||
"fast",
|
||||
"easy",
|
||||
"strong"
|
||||
],
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/WeBankFinTech/fes.git"
|
||||
},
|
||||
"dependencies": {
|
||||
"lerna": "^3.18.4"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@webank/eslint-config-webank": "^0.1.6"
|
||||
}
|
||||
}
|
21
packages/fes-cli/LICENSE
Normal file
@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2020-present harrywan
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
21
packages/fes-cli/README.md
Normal file
@ -0,0 +1,21 @@
|
||||
# fes-cli
|
||||
`fes-cli`是命令行工具,解决创建工程、开发调试、打包发布。
|
||||
|
||||
## 安装:
|
||||
npm install -g @webank/fes-cli
|
||||
|
||||
## 使用
|
||||
|
||||
### 创建项目
|
||||
fes init [project]
|
||||
|
||||
### 开发调试
|
||||
fes dev
|
||||
|
||||
启动http服务,默认监听localhost:5000
|
||||
|
||||
### 编译打包
|
||||
fes build
|
||||
|
||||
## 文档
|
||||
详细使用请查看[文档](https://webankfintech.github.io/fes.js/)
|
70
packages/fes-cli/bin/index.js
Executable file
@ -0,0 +1,70 @@
|
||||
#!/usr/bin/env node
|
||||
|
||||
const commander = require('commander');
|
||||
|
||||
const pkg = require('../package.json');
|
||||
const generateConfig = require('../build/helpers/config');
|
||||
const log = require('../build/helpers/log');
|
||||
|
||||
commander.usage('<command> [options]')
|
||||
.version(pkg.version)
|
||||
.option('-e, --env <env>', '配置环境 local(本地) | sit(测试) | prod(生产)')
|
||||
.description(pkg.description);
|
||||
|
||||
commander.command('init [name]')
|
||||
.description('创建项目')
|
||||
.action(async (name) => {
|
||||
const projectInit = require('../build/tasks/init');
|
||||
const config = generateConfig('init');
|
||||
await projectInit(config, name);
|
||||
});
|
||||
|
||||
commander.command('update')
|
||||
.description('将 fes2 项目升级到 fes3')
|
||||
.action(() => {
|
||||
const update = require('../build/tasks/update');
|
||||
const config = generateConfig('update');
|
||||
update(config);
|
||||
});
|
||||
|
||||
commander.command('route')
|
||||
.description('根据当前项目结构自动生成 route')
|
||||
.action(() => {
|
||||
const route = require('../build/tasks/route');
|
||||
const config = generateConfig('route');
|
||||
route(config);
|
||||
});
|
||||
|
||||
commander.command('components')
|
||||
.description('预编译 src/components 里面的组件')
|
||||
.action(() => {
|
||||
const components = require('../build/tasks/components');
|
||||
const config = generateConfig('components');
|
||||
components(config);
|
||||
});
|
||||
|
||||
commander.command('dev')
|
||||
.description('开发调试, 默认 local')
|
||||
.action(() => {
|
||||
const dev = require('../build/tasks/dev');
|
||||
const config = generateConfig('dev', commander.env || 'local');
|
||||
dev(config);
|
||||
});
|
||||
|
||||
commander.command('build')
|
||||
.description('打包压缩,默认 prod')
|
||||
.action(() => {
|
||||
const build = require('../build/tasks/build');
|
||||
const config = generateConfig('build', commander.env || 'prod');
|
||||
build(config);
|
||||
});
|
||||
|
||||
commander.parse(process.argv);
|
||||
|
||||
|
||||
if (!process.argv.slice(2).length) {
|
||||
commander.outputHelp((text) => {
|
||||
log.message(text);
|
||||
return '';
|
||||
});
|
||||
}
|
8
packages/fes-cli/build/configs/postcss.config.js
Normal file
@ -0,0 +1,8 @@
|
||||
const autoprefixer = require('autoprefixer');
|
||||
const browsers = require('../helpers/browser');
|
||||
|
||||
module.exports = {
|
||||
plugins: [
|
||||
autoprefixer({ browsers })
|
||||
]
|
||||
};
|
504
packages/fes-cli/build/configs/webpack.config.js
Normal file
@ -0,0 +1,504 @@
|
||||
const path = require('path');
|
||||
const fs = require('fs');
|
||||
const merge = require('webpack-merge');
|
||||
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
|
||||
const VueLoaderPlugin = require('vue-loader/lib/plugin');
|
||||
const CaseSensitivePathsPlugin = require('case-sensitive-paths-webpack-plugin');
|
||||
const FriendlyErrorsPlugin = require('friendly-errors-webpack-plugin');
|
||||
const CopyPlugin = require('copy-webpack-plugin');
|
||||
const OptimizeCssnanoPlugin = require('@intervolga/optimize-cssnano-plugin');
|
||||
const TerserPlugin = require('terser-webpack-plugin');
|
||||
const { CleanWebpackPlugin } = require('clean-webpack-plugin');
|
||||
const HtmlPlugin = require('html-webpack-plugin');
|
||||
const browsers = require('../helpers/browser');
|
||||
|
||||
module.exports = function webpackConfig(configs, webpack, mode) {
|
||||
let template = path.resolve(
|
||||
configs.folders.PROJECT_DIR,
|
||||
'./src/index.html'
|
||||
);
|
||||
if (!fs.existsSync(template)) {
|
||||
template = path.resolve(configs.folders.FES_DIR, './src/index.html');
|
||||
}
|
||||
|
||||
const isDev = mode === 'dev';
|
||||
const isBuild = mode === 'build';
|
||||
|
||||
const projectNodeModulesDir = path.resolve(
|
||||
configs.folders.PROJECT_DIR,
|
||||
'./node_modules'
|
||||
);
|
||||
const cliNodeModulesDir = path.resolve(
|
||||
configs.folders.CLI_DIR,
|
||||
'./node_modules'
|
||||
);
|
||||
const nodeModulesDir = ['node_modules', projectNodeModulesDir, cliNodeModulesDir];
|
||||
const presets = [
|
||||
[
|
||||
path.resolve(cliNodeModulesDir, '@babel/preset-env'),
|
||||
{
|
||||
modules: false,
|
||||
useBuiltIns: 'entry',
|
||||
corejs: 3,
|
||||
targets: {
|
||||
browsers
|
||||
}
|
||||
}
|
||||
]
|
||||
];
|
||||
const plugins = [
|
||||
path.resolve(cliNodeModulesDir, '@babel/plugin-proposal-object-rest-spread'),
|
||||
path.resolve(cliNodeModulesDir, '@babel/plugin-syntax-dynamic-import')
|
||||
];
|
||||
const cssloaders = [
|
||||
isDev
|
||||
? {
|
||||
loader: 'vue-style-loader',
|
||||
options: {
|
||||
sourceMap: false,
|
||||
shadowMode: false
|
||||
}
|
||||
}
|
||||
: {
|
||||
loader: MiniCssExtractPlugin.loader,
|
||||
options: {
|
||||
publicPath: '../'
|
||||
}
|
||||
},
|
||||
{
|
||||
loader: 'css-loader',
|
||||
options: {
|
||||
sourceMap: false,
|
||||
importLoaders: 2
|
||||
}
|
||||
},
|
||||
{
|
||||
loader: 'postcss-loader',
|
||||
options: {
|
||||
config: {
|
||||
path: path.resolve(configs.folders.CLI_DIR, 'build/configs/postcss.config.js')
|
||||
},
|
||||
sourceMap: false
|
||||
}
|
||||
}
|
||||
];
|
||||
|
||||
|
||||
const baseConfig = {
|
||||
|
||||
mode: isDev ? 'development' : 'production',
|
||||
|
||||
context: path.resolve(configs.folders.PROJECT_DIR),
|
||||
|
||||
entry: {
|
||||
app: [
|
||||
path.resolve(configs.folders.CLI_DIR, './node_modules/babel-polyfill'),
|
||||
// path.resolve(configs.folders.CLI_DIR, './build/utils/create-nonce'),
|
||||
path.resolve(configs.folders.FES_DIR, './src/app.js')
|
||||
]
|
||||
},
|
||||
|
||||
resolve: {
|
||||
extensions: ['.js', '.fes', '.vue', '.json'],
|
||||
modules: nodeModulesDir,
|
||||
alias: {
|
||||
projectRoot: configs.folders.PROJECT_DIR,
|
||||
'@': path.resolve(configs.folders.PROJECT_DIR, 'src'),
|
||||
'@@': path.resolve(configs.folders.FES_DIR, 'src'),
|
||||
assets: path.resolve(
|
||||
configs.folders.PROJECT_DIR,
|
||||
'./src/assets/'
|
||||
),
|
||||
vue$: 'vue/dist/vue.esm.js'
|
||||
}
|
||||
},
|
||||
|
||||
resolveLoader: {
|
||||
modules: nodeModulesDir
|
||||
},
|
||||
|
||||
output: {
|
||||
globalObject: 'this',
|
||||
filename: isDev ? 'js/[name].js' : 'js/[name].[contenthash:8].js',
|
||||
chunkFilename: isDev ? 'js/[name].chunk.js' : 'js/[name].[contenthash:8].js',
|
||||
path: configs.folders.PROJECT_DIST_DIR,
|
||||
publicPath: isDev ? '/' : './'
|
||||
},
|
||||
|
||||
module: {
|
||||
noParse: /^(vue|vue-router|vuex|vuex-router-sync|axios|@webank\/fes-ui)$/,
|
||||
rules: [
|
||||
|
||||
/* config.module.rule('vue') */
|
||||
{
|
||||
test: /\.vue|fes$/,
|
||||
use: [
|
||||
{
|
||||
loader: 'cache-loader',
|
||||
options: {
|
||||
cacheDirectory: path.resolve(configs.folders.PROJECT_DIR, 'node_modules/.cache/vue-loader')
|
||||
}
|
||||
},
|
||||
{
|
||||
loader: 'vue-loader',
|
||||
options: {
|
||||
compilerOptions: {
|
||||
preserveWhitespace: false
|
||||
},
|
||||
cacheDirectory: path.resolve(configs.folders.PROJECT_DIR, 'node_modules/.cache/vue-loader')
|
||||
}
|
||||
}
|
||||
]
|
||||
},
|
||||
|
||||
/* config.module.rule('images') */
|
||||
{
|
||||
test: /\.(png|jpe?g|gif|webp)(\?.*)?$/,
|
||||
use: [
|
||||
{
|
||||
loader: 'url-loader',
|
||||
options: {
|
||||
limit: 4096,
|
||||
fallback: {
|
||||
loader: 'file-loader',
|
||||
options: {
|
||||
name: isDev ? 'img/[name].[ext]' : 'img/[name].[hash:8].[ext]'
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
},
|
||||
|
||||
/* config.module.rule('svg') */
|
||||
{
|
||||
test: /\.(svg)(\?.*)?$/,
|
||||
use: [
|
||||
{
|
||||
loader: 'file-loader',
|
||||
options: {
|
||||
name: isDev ? 'img/[name].[ext]' : 'img/[name].[hash:8].[ext]'
|
||||
}
|
||||
}
|
||||
]
|
||||
},
|
||||
|
||||
/* config.module.rule('media') */
|
||||
{
|
||||
test: /\.(mp4|webm|ogg|mp3|wav|flac|aac)(\?.*)?$/,
|
||||
use: [
|
||||
{
|
||||
loader: 'url-loader',
|
||||
options: {
|
||||
limit: 4096,
|
||||
fallback: {
|
||||
loader: 'file-loader',
|
||||
options: {
|
||||
name: isDev ? 'media/[name].[ext]' : 'media/[name].[hash:8].[ext]'
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
},
|
||||
|
||||
/* config.module.rule('fonts') */
|
||||
{
|
||||
test: /\.(woff2?|eot|ttf|otf)(\?.*)?$/i,
|
||||
use: [
|
||||
{
|
||||
loader: 'url-loader',
|
||||
options: {
|
||||
limit: 4096,
|
||||
fallback: {
|
||||
loader: 'file-loader',
|
||||
options: {
|
||||
name: isDev ? 'fonts/[name].[ext]' : 'fonts/[name].[hash:8].[ext]'
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
},
|
||||
|
||||
/* config.module.rule('css') */
|
||||
{
|
||||
test: /\.css$/,
|
||||
use: cssloaders
|
||||
},
|
||||
|
||||
/* config.module.rule('postcss') */
|
||||
{
|
||||
test: /\.p(ost)?css$/,
|
||||
use: cssloaders
|
||||
},
|
||||
|
||||
/* config.module.rule('scss') */
|
||||
{
|
||||
test: /\.scss$/,
|
||||
use: cssloaders.concat([
|
||||
{
|
||||
loader: 'sass-loader',
|
||||
options: {
|
||||
sourceMap: false
|
||||
}
|
||||
}
|
||||
])
|
||||
},
|
||||
|
||||
/* config.module.rule('sass') */
|
||||
{
|
||||
test: /\.sass$/,
|
||||
use: cssloaders.concat([
|
||||
{
|
||||
loader: 'sass-loader',
|
||||
options: {
|
||||
sourceMap: false,
|
||||
indentedSyntax: true
|
||||
}
|
||||
}
|
||||
])
|
||||
},
|
||||
|
||||
/* config.module.rule('less') */
|
||||
{
|
||||
test: /\.less$/,
|
||||
use: cssloaders.concat([
|
||||
{
|
||||
loader: 'less-loader',
|
||||
options: {
|
||||
sourceMap: false,
|
||||
javascriptEnabled: true
|
||||
}
|
||||
}
|
||||
])
|
||||
},
|
||||
|
||||
/* config.module.rule('stylus') */
|
||||
{
|
||||
test: /\.styl(us)?$/,
|
||||
use: cssloaders.concat([
|
||||
{
|
||||
loader: 'stylus-loader',
|
||||
options: {
|
||||
sourceMap: false,
|
||||
preferPathResolver: 'webpack'
|
||||
}
|
||||
}
|
||||
])
|
||||
},
|
||||
|
||||
/* config.module.rule('js') */
|
||||
{
|
||||
test: /\.m?jsx?$/,
|
||||
use: [
|
||||
{
|
||||
loader: 'cache-loader',
|
||||
options: {
|
||||
cacheDirectory: path.resolve(configs.folders.PROJECT_DIR, 'node_modules/.cache/babel-loader')
|
||||
}
|
||||
},
|
||||
{
|
||||
loader: 'thread-loader'
|
||||
},
|
||||
{
|
||||
loader: 'babel-loader',
|
||||
options: {
|
||||
presets,
|
||||
plugins
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
|
||||
]
|
||||
},
|
||||
|
||||
devtool: isDev && 'cheap-module-eval-source-map',
|
||||
|
||||
plugins: [
|
||||
|
||||
/* config.plugin('progress') */
|
||||
new webpack.ProgressPlugin(),
|
||||
|
||||
/* config.plugin('vue-loader') */
|
||||
new VueLoaderPlugin(),
|
||||
|
||||
/* config.plugin('define') */
|
||||
new webpack.DefinePlugin({
|
||||
'process.privateFesEnv': {
|
||||
env: `"${configs.env}"`
|
||||
},
|
||||
'process.env': {
|
||||
env: JSON.stringify(configs.env),
|
||||
command: JSON.stringify(configs.command)
|
||||
}
|
||||
}),
|
||||
|
||||
/* config.plugin('clean dist') */
|
||||
isBuild && new CleanWebpackPlugin(),
|
||||
|
||||
/* config.plugin('extract-css') */
|
||||
isBuild
|
||||
&& new MiniCssExtractPlugin({
|
||||
filename: 'css/[name].[contenthash:8].css',
|
||||
chunkFilename: 'css/[name].[contenthash:8].css'
|
||||
}),
|
||||
|
||||
/* config.plugin('Copy static') */
|
||||
isBuild
|
||||
&& new CopyPlugin([
|
||||
{
|
||||
from: configs.folders.PROJECT_STATIC_DIR,
|
||||
to: path.resolve(
|
||||
configs.folders.PROJECT_DIST_DIR,
|
||||
'static'
|
||||
)
|
||||
}
|
||||
]),
|
||||
|
||||
/* config.plugin('optimize-css') */
|
||||
isBuild
|
||||
&& new OptimizeCssnanoPlugin({
|
||||
sourceMap: false,
|
||||
cssnanoOptions: {
|
||||
preset: [
|
||||
'default',
|
||||
{
|
||||
mergeLonghand: false,
|
||||
cssDeclarationSorter: false
|
||||
}
|
||||
]
|
||||
}
|
||||
}),
|
||||
|
||||
/* config.plugin('hash-module-ids') */
|
||||
isBuild
|
||||
&& new webpack.HashedModuleIdsPlugin({
|
||||
hashDigest: 'hex'
|
||||
}),
|
||||
|
||||
/* config.plugin('固定一下 chunk id') */
|
||||
isBuild
|
||||
&& new webpack.NamedChunksPlugin((chunk) => {
|
||||
if (chunk.name) {
|
||||
return chunk.name;
|
||||
}
|
||||
// eslint-disable-next-line
|
||||
const hash = require('hash-sum');
|
||||
const joinedHash = hash(
|
||||
Array.from(chunk.modulesIterable, m => m.id).join('_')
|
||||
);
|
||||
return `chunk-${joinedHash}`;
|
||||
}),
|
||||
|
||||
/* config.plugin('Copyright') */
|
||||
isBuild
|
||||
&& new webpack.BannerPlugin(
|
||||
'Created By MumbleFe. Copyright © 2015 - 2018 WeBank.'
|
||||
),
|
||||
|
||||
/* config.plugin('case-sensitive-paths') */
|
||||
new CaseSensitivePathsPlugin(),
|
||||
|
||||
/* config.plugin('friendly-errors') */
|
||||
new FriendlyErrorsPlugin(),
|
||||
|
||||
/* config.plugin('index.html') */
|
||||
new HtmlPlugin({
|
||||
template,
|
||||
minify: isBuild && {
|
||||
removeComments: true,
|
||||
collapseWhitespace: true,
|
||||
removeAttributeQuotes: true,
|
||||
collapseBooleanAttributes: true,
|
||||
removeScriptTypeAttributes: true
|
||||
}
|
||||
})
|
||||
|
||||
]
|
||||
};
|
||||
|
||||
if (isBuild) {
|
||||
baseConfig.optimization = {
|
||||
minimizer: [
|
||||
new TerserPlugin({
|
||||
test: /\.m?js(\?.*)?$/i,
|
||||
chunkFilter: () => true,
|
||||
warningsFilter: () => true,
|
||||
extractComments: false,
|
||||
sourceMap: true,
|
||||
cache: true,
|
||||
cacheKeys: defaultCacheKeys => defaultCacheKeys,
|
||||
parallel: true,
|
||||
include: undefined,
|
||||
exclude: undefined,
|
||||
minify: undefined,
|
||||
terserOptions: {
|
||||
output: {
|
||||
comments: /^\**!|@preserve|@license|@cc_on/i
|
||||
},
|
||||
compress: {
|
||||
arrows: false,
|
||||
collapse_vars: false,
|
||||
comparisons: false,
|
||||
computed_props: false,
|
||||
hoist_funs: false,
|
||||
hoist_props: false,
|
||||
hoist_vars: false,
|
||||
inline: false,
|
||||
loops: false,
|
||||
negate_iife: false,
|
||||
properties: false,
|
||||
reduce_funcs: false,
|
||||
reduce_vars: false,
|
||||
switches: false,
|
||||
toplevel: false,
|
||||
typeofs: false,
|
||||
booleans: true,
|
||||
if_return: true,
|
||||
sequences: true,
|
||||
unused: true,
|
||||
conditionals: true,
|
||||
dead_code: true,
|
||||
evaluate: true
|
||||
},
|
||||
mangle: {
|
||||
safari10: true
|
||||
}
|
||||
}
|
||||
})
|
||||
],
|
||||
splitChunks: {
|
||||
cacheGroups: {
|
||||
vendors: {
|
||||
name: 'chunk-vendors',
|
||||
test: /[\\/]node_modules[\\/]/,
|
||||
priority: -10,
|
||||
chunks: 'initial'
|
||||
},
|
||||
common: {
|
||||
name: 'chunk-common',
|
||||
minChunks: 2,
|
||||
priority: -20,
|
||||
chunks: 'initial',
|
||||
reuseExistingChunk: true
|
||||
}
|
||||
}
|
||||
},
|
||||
runtimeChunk: true
|
||||
};
|
||||
}
|
||||
|
||||
baseConfig.plugins = baseConfig.plugins.filter(plu => plu !== false);
|
||||
|
||||
let advancedConfig = {};
|
||||
const projectWebpackConfigFile = path.resolve(configs.folders.PROJECT_DIR, 'webpack.config.js');
|
||||
if (fs.existsSync(projectWebpackConfigFile)) {
|
||||
console.log('[init] 加载项目个性webpack配置文件');
|
||||
// eslint-disable-next-line
|
||||
advancedConfig = require(projectWebpackConfigFile)(mode, configs, webpack);
|
||||
}
|
||||
|
||||
return merge(baseConfig, advancedConfig);
|
||||
};
|
1
packages/fes-cli/build/helpers/browser.js
Normal file
@ -0,0 +1 @@
|
||||
module.exports = ['>1%', 'last 2 versions', 'safari >= 7', 'ie >= 9'];
|
62
packages/fes-cli/build/helpers/config.js
Normal file
@ -0,0 +1,62 @@
|
||||
const path = require('path');
|
||||
const fs = require('fs');
|
||||
const log = require('./log');
|
||||
|
||||
function generateConfig(command, env) {
|
||||
// cli目录
|
||||
const CLI_DIR = path.dirname(path.dirname(fs.realpathSync(process.argv[1])));
|
||||
// 解决git-bash目录问题
|
||||
const PROJECT_DIR = process.env.PWD || process.cwd();
|
||||
const FES_DIR = path.resolve(PROJECT_DIR, './node_modules/@webank/fes-core');
|
||||
|
||||
const PROJECT_DIST_DIR = path.resolve(PROJECT_DIR, 'dist');
|
||||
const PROJECT_CACHE_DIR = path.resolve(PROJECT_DIR, './.cache');
|
||||
const PROJECT_PAGE_DIR = path.resolve(PROJECT_DIR, './src/pages');
|
||||
const PROJECT_CPN_DIR = path.resolve(PROJECT_DIR, './src/components');
|
||||
const PROJECT_STATIC_DIR = path.join(PROJECT_DIR, './src/static');
|
||||
const projectName = path.basename(PROJECT_DIR);
|
||||
|
||||
const fesConfigFile = path.join(PROJECT_DIR, 'fes.config.js');
|
||||
|
||||
const config = {
|
||||
command,
|
||||
env,
|
||||
ports: {
|
||||
server: 5000,
|
||||
liveReload: 35729
|
||||
},
|
||||
projectName,
|
||||
folders: {
|
||||
CLI_DIR,
|
||||
FES_DIR,
|
||||
PROJECT_DIR,
|
||||
PROJECT_STATIC_DIR,
|
||||
PROJECT_DIST_DIR,
|
||||
PROJECT_CACHE_DIR,
|
||||
PROJECT_PAGE_DIR,
|
||||
PROJECT_CPN_DIR
|
||||
}
|
||||
};
|
||||
|
||||
if (fs.existsSync(fesConfigFile)) {
|
||||
try {
|
||||
// eslint-disable-next-line
|
||||
const fesCofig = require(path.join(config.folders.PROJECT_DIR, 'fes.config.js'));
|
||||
config.CDN = fesCofig.env[config.env].cdn;
|
||||
config.needCDN = !!config.CDN;
|
||||
} catch (e) {
|
||||
config.needCDN = false;
|
||||
}
|
||||
}
|
||||
|
||||
if (!config.needCDN) {
|
||||
if (config.command === 'dev' || config.command === 'build') {
|
||||
log.warn('项目没有配置cdn,打包之后将不会请求cdn的地址,请开发者注意!!');
|
||||
}
|
||||
}
|
||||
|
||||
return config;
|
||||
}
|
||||
|
||||
|
||||
module.exports = generateConfig;
|
44
packages/fes-cli/build/helpers/createDevServer.js
Normal file
@ -0,0 +1,44 @@
|
||||
const
|
||||
http = require('http');
|
||||
const webpack = require('webpack');
|
||||
const express = require('express');
|
||||
const opn = require('opn');
|
||||
const webpackHotMiddleware = require('webpack-hot-middleware');
|
||||
const webpackDevMiddleware = require('webpack-dev-middleware');
|
||||
const initMock = require('../mock/init.js');
|
||||
|
||||
|
||||
module.exports = function createDevServer(port, defaultConfig) {
|
||||
defaultConfig.entry.app.unshift('webpack-hot-middleware/client?reload=true');
|
||||
defaultConfig.plugins.push(new webpack.HotModuleReplacementPlugin());
|
||||
defaultConfig.plugins.push(new webpack.NamedModulesPlugin());
|
||||
|
||||
const app = express();
|
||||
const compiler = webpack(defaultConfig);
|
||||
|
||||
// devServer 自带支持,添加自定义插件。
|
||||
app.use(webpackDevMiddleware(compiler, {
|
||||
lazy: false,
|
||||
watchOptions: {
|
||||
aggregateTimeout: 300,
|
||||
poll: 1000
|
||||
},
|
||||
stats: {
|
||||
colors: true,
|
||||
chunks: false,
|
||||
timings: true
|
||||
},
|
||||
publicPath: defaultConfig.output.publicPath
|
||||
}));
|
||||
|
||||
app.use(webpackHotMiddleware(compiler));
|
||||
app.use('/static', express.static('src/static'));
|
||||
|
||||
|
||||
// 初始化Mock数据
|
||||
initMock(app);
|
||||
|
||||
opn(`http://localhost:${port}`);
|
||||
|
||||
http.createServer(app).listen(port);
|
||||
};
|
25
packages/fes-cli/build/helpers/getPort.js
Normal file
@ -0,0 +1,25 @@
|
||||
const net = require('net');
|
||||
|
||||
function checkout(port) {
|
||||
return new Promise((resolve) => {
|
||||
const server = net.createServer();
|
||||
server.once('error', (err) => {
|
||||
if (err.code === 'EADDRINUSE') {
|
||||
resolve(checkout(port + 1));
|
||||
}
|
||||
});
|
||||
|
||||
server.once('listening', () => {
|
||||
server.close(() => {
|
||||
resolve(port);
|
||||
});
|
||||
});
|
||||
|
||||
server.listen(port);
|
||||
});
|
||||
}
|
||||
|
||||
module.exports = function getPort(basePort) {
|
||||
basePort = basePort || 5000;
|
||||
return checkout(basePort);
|
||||
};
|
13
packages/fes-cli/build/helpers/log.js
Normal file
@ -0,0 +1,13 @@
|
||||
const chalk = require('chalk');
|
||||
|
||||
module.exports = {
|
||||
error(msg) {
|
||||
return console.log(chalk.red(msg));
|
||||
},
|
||||
warn(msg) {
|
||||
return console.log(chalk.yellow(msg));
|
||||
},
|
||||
message(msg) {
|
||||
return console.log(chalk.cyan(msg));
|
||||
}
|
||||
};
|
142
packages/fes-cli/build/mock/cgiMock.js
Normal file
@ -0,0 +1,142 @@
|
||||
const express = require('express');
|
||||
const fs = require('fs');
|
||||
const path = require('path');
|
||||
const httpProxy = require('http-proxy');
|
||||
const url = require('url');
|
||||
|
||||
const util = require('./util');
|
||||
|
||||
const proxy = httpProxy.createProxyServer();
|
||||
global.router = express.Router();
|
||||
|
||||
/**
|
||||
* 数据模拟函数
|
||||
*/
|
||||
function cgiMock() {
|
||||
// eslint-disable-next-line
|
||||
const option = getOption(arguments);
|
||||
|
||||
if (!option.url || !option.result) {
|
||||
return;
|
||||
}
|
||||
|
||||
// option.method is one of ['get','post','delete','put'...]
|
||||
const method = option.method || 'use';
|
||||
|
||||
global.router[method.toLowerCase()](option.url, (req, res) => {
|
||||
setTimeout(() => {
|
||||
// set header
|
||||
res.set(option.headers);
|
||||
|
||||
// set Content-Type
|
||||
option.type && res.type(option.type);
|
||||
|
||||
// set status code
|
||||
res.status(option.statusCode);
|
||||
|
||||
// set cookie
|
||||
util.each(option.cookies, (item) => {
|
||||
const name = item.name;
|
||||
const value = item.value;
|
||||
delete item.name;
|
||||
delete item.value;
|
||||
res.cookie(name, value, item);
|
||||
});
|
||||
|
||||
// do result
|
||||
if (util.isFunction(option.result)) {
|
||||
option.result(req, res);
|
||||
} else if (util.isArray(option.result) || util.isObject(option.result)) {
|
||||
!option.type && res.type('json');
|
||||
res.json(option.result);
|
||||
} else {
|
||||
!option.type && res.type('text');
|
||||
res.send(option.result.toString());
|
||||
}
|
||||
}, option.timeout);
|
||||
});
|
||||
}
|
||||
|
||||
// 根据参数个数获取配置
|
||||
function getOption(arg) {
|
||||
const len = arg.length;
|
||||
// 默认配置
|
||||
const option = {
|
||||
headers: {
|
||||
'Cache-Control': 'no-cache'
|
||||
},
|
||||
statusCode: 200,
|
||||
cookies: [],
|
||||
timeout: 0
|
||||
};
|
||||
if (len === 0) {
|
||||
return cgiMock;
|
||||
} if (len === 1) {
|
||||
const newOption = arg[0];
|
||||
if (util.isObject(newOption)) {
|
||||
util.each(newOption, (value, key) => {
|
||||
if (key === 'headers') {
|
||||
util.each(newOption.headers, (headervalue, headerkey) => {
|
||||
option.headers[headerkey] = newOption.headers[headerkey];
|
||||
});
|
||||
} else {
|
||||
option[key] = newOption[key];
|
||||
}
|
||||
});
|
||||
}
|
||||
} else {
|
||||
option.url = arg[0];
|
||||
option.result = arg[1];
|
||||
}
|
||||
return option;
|
||||
}
|
||||
|
||||
// 把基于 cgiMockfile 的相对绝对转成绝对路径
|
||||
function parsePath(value) {
|
||||
return path.join(global.cgiMockFilePath, value);
|
||||
}
|
||||
|
||||
|
||||
// log proxy data
|
||||
proxy.on('open', (proxySocket) => {
|
||||
proxySocket.on('data', (chunk) => {
|
||||
console.log(chunk.toString());
|
||||
});
|
||||
});
|
||||
proxy.on('proxyRes', (proxyRes) => {
|
||||
console.log('RAW Response from the target', JSON.stringify(proxyRes.headers, true, 2));
|
||||
const cookie = proxyRes.headers['set-cookie'];
|
||||
if (cookie && cookie.length > 0) {
|
||||
for (let i = 0; i < cookie.length; i++) {
|
||||
cookie[i] = cookie[i].replace('Secure', '');
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
proxy.on('error', (e) => {
|
||||
console.log(e);
|
||||
});
|
||||
|
||||
// 规则之外的请求转发
|
||||
cgiMock.proxy = function (host) {
|
||||
process.nextTick(() => {
|
||||
global.router.use((req, res) => {
|
||||
proxy.web(req, res, {
|
||||
target: host,
|
||||
secure: false
|
||||
});
|
||||
});
|
||||
});
|
||||
proxy.on('proxyReq', (proxyReq) => {
|
||||
proxyReq.setHeader('Host', url.parse(host).host);
|
||||
});
|
||||
};
|
||||
|
||||
// 读取文件内容
|
||||
cgiMock.file = function (file) {
|
||||
return fs.readFileSync(parsePath(file));
|
||||
};
|
||||
|
||||
cgiMock.prefix = '/';
|
||||
|
||||
module.exports = cgiMock;
|
112
packages/fes-cli/build/mock/init.js
Normal file
@ -0,0 +1,112 @@
|
||||
const Mock = require('mockjs');
|
||||
const faker = require('faker');
|
||||
const path = require('path');
|
||||
const fs = require('fs');
|
||||
|
||||
const logger = require('morgan');
|
||||
const cookieParser = require('cookie-parser');
|
||||
const bodyParser = require('body-parser');
|
||||
const onFinished = require('on-finished');
|
||||
|
||||
const util = require('./util');
|
||||
const cgiMock = require('./cgiMock');
|
||||
const log = require('../helpers/log');
|
||||
|
||||
const main = {
|
||||
init(app, argv, cwd) {
|
||||
this.app = app;
|
||||
this.argv = argv;
|
||||
this.cwd = cwd;
|
||||
|
||||
app.use(logger('dev'));
|
||||
app.use(
|
||||
bodyParser.urlencoded({
|
||||
extended: false
|
||||
})
|
||||
);
|
||||
app.use(cookieParser());
|
||||
|
||||
this.customRoute();
|
||||
this.defaultRoute();
|
||||
},
|
||||
|
||||
customRoute() {
|
||||
const argv = this.argv;
|
||||
const defaultCgiMockFile = path.join(process.cwd(), 'mock.js');
|
||||
const home = process.env[process.platform === 'win32' ? 'USERPROFILE' : 'HOME'];
|
||||
|
||||
let cgiMockFile;
|
||||
if (argv) {
|
||||
if (argv.f) {
|
||||
if (process.platform === 'win32') {
|
||||
cgiMockFile = path.resolve(this.cwd, this.argv.f);
|
||||
} else if (argv.f[0] === '~') {
|
||||
cgiMockFile = path.resolve(
|
||||
home,
|
||||
argv.f.replace(/^~\//, '')
|
||||
);
|
||||
} else {
|
||||
cgiMockFile = path.resolve(this.cwd, this.argv.f);
|
||||
}
|
||||
} else {
|
||||
cgiMockFile = defaultCgiMockFile;
|
||||
}
|
||||
} else {
|
||||
cgiMockFile = defaultCgiMockFile;
|
||||
}
|
||||
global.cgiMockFilePath = path.resolve(cgiMockFile, '..');
|
||||
|
||||
const loadRouteConfig = function () {
|
||||
util.cleanCache(cgiMockFile);
|
||||
try {
|
||||
if (!fs.existsSync(cgiMockFile)) {
|
||||
log.error('[WARN] 不存在mock.js文件');
|
||||
} else {
|
||||
// eslint-disable-next-line
|
||||
const projectMock = require(cgiMockFile);
|
||||
if (util.isFunction(projectMock)) {
|
||||
global.router.stack = [];
|
||||
projectMock(cgiMock, Mock, faker);
|
||||
log.message('[SUCCESS] mock.js 加载成功');
|
||||
} else {
|
||||
log.error(
|
||||
`[ERROR] mock.js cannot be ${typeof projectMock}`
|
||||
);
|
||||
}
|
||||
}
|
||||
} catch (e) {
|
||||
log.error('[ERROR] mock.js 有误,请检查');
|
||||
log.error(JSON.stringify(e));
|
||||
}
|
||||
};
|
||||
|
||||
loadRouteConfig();
|
||||
this.app.use(cgiMock.prefix, (req, res, next) => {
|
||||
onFinished(res, () => {
|
||||
loadRouteConfig();
|
||||
});
|
||||
global.router(req, res, next);
|
||||
});
|
||||
|
||||
util.watchFile(cgiMockFile, () => {
|
||||
log.message('[INFO] mock.js 发生变化');
|
||||
loadRouteConfig();
|
||||
});
|
||||
},
|
||||
|
||||
defaultRoute() {
|
||||
const app = this.app;
|
||||
|
||||
setTimeout(() => {
|
||||
app.use((err, req, res) => {
|
||||
res.json({
|
||||
status: err.status || 500,
|
||||
message: err.message,
|
||||
err
|
||||
});
|
||||
});
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
module.exports = main.init.bind(main);
|
17
packages/fes-cli/build/mock/task/cgiMock.js
Normal file
@ -0,0 +1,17 @@
|
||||
|
||||
|
||||
const express = require('express');
|
||||
const argv = require('yargs').argv;
|
||||
|
||||
const port = argv.p || 8888;
|
||||
const cwd = process.cwd();
|
||||
|
||||
const app = express();
|
||||
const init = require('../init');
|
||||
|
||||
init(app, argv, cwd);
|
||||
|
||||
app.set('port', port);
|
||||
app.listen(port, () => {
|
||||
console.log(`cgiMock server listening on ${port}`);
|
||||
});
|
64
packages/fes-cli/build/mock/test/app.js
Normal file
@ -0,0 +1,64 @@
|
||||
module.exports = function mock(cgiMock, Mock) {
|
||||
const Random = Mock.Random;
|
||||
|
||||
// 前缀,全局(可选)
|
||||
cgiMock.prefix = '/prefix';
|
||||
|
||||
// 返回一个数字
|
||||
cgiMock('/number', 123);
|
||||
|
||||
// 返回一个json
|
||||
cgiMock({
|
||||
url: '/json',
|
||||
result: {
|
||||
code: '400101', msg: "不合法的请求:Missing cookie 'wb_app_id' for method parameter of type String", transactionTime: '20170309171146', success: false
|
||||
}
|
||||
});
|
||||
|
||||
// 利用mock.js 产生随机文本
|
||||
cgiMock('/text', Random.cparagraph());
|
||||
|
||||
// 返回一个字符串 利用mock.js 产生随机字符
|
||||
cgiMock('/string', Mock.mock({
|
||||
'string|1-10': '★'
|
||||
}));
|
||||
|
||||
|
||||
// 正则匹配url, 返回一个字符串
|
||||
cgiMock(/\/abc|\/xyz/, 'regexp test!');
|
||||
|
||||
// option.result 参数如果是一个函数, 可以实现自定义返回内容, 接收的参数是是经过 express 封装的 req 和 res 对象.
|
||||
cgiMock(/\/function$/, (req, res) => {
|
||||
res.send('function test');
|
||||
});
|
||||
|
||||
// 返回文本 fs.readFileSync
|
||||
cgiMock('/file', cgiMock.file('./test.json'));
|
||||
|
||||
// 更复杂的规则配置
|
||||
cgiMock({
|
||||
url: /\/who/,
|
||||
method: 'GET',
|
||||
result(req, res) {
|
||||
if (req.query.name === 'kwan') {
|
||||
res.json({ kwan: '孤独患者' });
|
||||
} else {
|
||||
res.send('Nooooooooooo');
|
||||
}
|
||||
},
|
||||
headers: {
|
||||
'Content-Type': 'text/plain',
|
||||
'Content-Length': '123',
|
||||
ETag: '12345'
|
||||
},
|
||||
cookies: [
|
||||
{
|
||||
name: 'myname', value: 'kwan', maxAge: 900000, httpOnly: true
|
||||
}
|
||||
],
|
||||
// 接口随机延迟
|
||||
timeout: Mock.mock({
|
||||
'number|1000-5000': 1000
|
||||
}).number
|
||||
});
|
||||
};
|
1
packages/fes-cli/build/mock/test/myfile.txt
Normal file
@ -0,0 +1 @@
|
||||
file test
|
62
packages/fes-cli/build/mock/util.js
Normal file
@ -0,0 +1,62 @@
|
||||
const fs = require('fs');
|
||||
|
||||
const toString = Object.prototype.toString;
|
||||
module.exports = {
|
||||
|
||||
isArray(value) {
|
||||
return toString.call(value) === '[object Array]';
|
||||
},
|
||||
|
||||
isObject(value) {
|
||||
return toString.call(value) === '[object Object]';
|
||||
},
|
||||
|
||||
isFunction(value) {
|
||||
return toString.call(value) === '[object Function]';
|
||||
},
|
||||
|
||||
each(val, callback) {
|
||||
if (this.isArray(val)) {
|
||||
val.forEach(callback);
|
||||
}
|
||||
if (this.isObject(val)) {
|
||||
Object.keys(val).forEach((key) => {
|
||||
callback(val[key], key);
|
||||
});
|
||||
}
|
||||
},
|
||||
|
||||
watchFile(filename, callback) {
|
||||
const isWin = (process.platform === 'win32');
|
||||
if (isWin) {
|
||||
return fs.watch(filename, (event) => {
|
||||
if (event === 'change') {
|
||||
return callback(filename);
|
||||
}
|
||||
return null;
|
||||
});
|
||||
}
|
||||
return fs.watchFile(filename, {
|
||||
interval: 200
|
||||
}, (curr, prev) => {
|
||||
if (curr.mtime > prev.mtime) {
|
||||
return callback(filename);
|
||||
}
|
||||
return null;
|
||||
});
|
||||
},
|
||||
|
||||
unwatchFile(watcher, filename) {
|
||||
if (watcher) {
|
||||
watcher.close && watcher.close();
|
||||
} else {
|
||||
fs.unwatchFile(filename);
|
||||
}
|
||||
},
|
||||
|
||||
cleanCache(modulePath) {
|
||||
if (require.cache[modulePath]) {
|
||||
delete require.cache[modulePath];
|
||||
}
|
||||
}
|
||||
};
|
26
packages/fes-cli/build/preComplie/components.js
Normal file
@ -0,0 +1,26 @@
|
||||
// 全局注册common目录下的组件
|
||||
const fs = require('fs');
|
||||
const Path = require('path');
|
||||
const stringUtil = require('node-plus-string');
|
||||
|
||||
function addComp(path, outputCommonDir, components) {
|
||||
const dirList = fs.readdirSync(path);
|
||||
dirList.forEach((item) => {
|
||||
if (fs.statSync(`${path}/${item}`).isFile()
|
||||
&& item[0] !== '.'
|
||||
&& ['.fes', '.vue'].indexOf(Path.extname(item)) !== -1) {
|
||||
const fileName = Path.basename(item, Path.extname(item));
|
||||
const tagName = stringUtil.capitalize(fileName);
|
||||
components.push({
|
||||
tagName,
|
||||
path: Path.resolve(outputCommonDir, item).replace(/\\/g, '\\\\')
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
module.exports = function genComponents(commonDir, outputCommonDir) {
|
||||
const components = [];
|
||||
addComp(commonDir, outputCommonDir, components);
|
||||
return components;
|
||||
};
|
121
packages/fes-cli/build/preComplie/route.js
Normal file
@ -0,0 +1,121 @@
|
||||
// pages
|
||||
// ├── index.fes # 根路由页面 路径 index.html#/
|
||||
// ├── a.fes # 路径 /a
|
||||
// ├── b
|
||||
// │ ├── index.fes # 路径 /b
|
||||
// │ ├── @id.fes # 动态路由 /b/:id
|
||||
// │ └── c.fes # 路径 /b/c
|
||||
// └── layout.fes # 根路由下所有page共用的外层
|
||||
|
||||
const fs = require('fs');
|
||||
const Path = require('path');
|
||||
|
||||
let pagesDir;
|
||||
let outputPageDir;
|
||||
let components = [];
|
||||
|
||||
function checkHasLayout(path) {
|
||||
const dirList = fs.readdirSync(path);
|
||||
let hasLayout = false;
|
||||
dirList.forEach((item) => {
|
||||
if (fs.statSync(`${path}/${item}`).isFile()
|
||||
&& item[0] !== '.' && ['.fes', '.vue'].indexOf(Path.extname(item)) !== -1
|
||||
&& Path.basename(item, Path.extname(item)) === 'layout') {
|
||||
hasLayout = true;
|
||||
}
|
||||
});
|
||||
return hasLayout;
|
||||
}
|
||||
|
||||
function routeUrlFormmter(str) {
|
||||
return str.replace(/@/g, ':');
|
||||
}
|
||||
|
||||
function genRoute(path, prePathUrl, preRoutes) {
|
||||
const hasLayout = checkHasLayout(path);
|
||||
const dirList = fs.readdirSync(path);
|
||||
let childRoutes = {};
|
||||
const parentRoutes = {};
|
||||
const preRouteUrl = routeUrlFormmter(prePathUrl);
|
||||
if (hasLayout) {
|
||||
parentRoutes[preRouteUrl] = {
|
||||
subRoutes: childRoutes
|
||||
};
|
||||
} else {
|
||||
childRoutes = parentRoutes;
|
||||
}
|
||||
dirList.forEach((item) => {
|
||||
if (fs.statSync(`${path}/${item}`).isFile()
|
||||
&& item[0] !== '.' && ['.fes', '.vue'].indexOf(Path.extname(item)) !== -1) {
|
||||
const fileName = Path.basename(item, Path.extname(item));
|
||||
const preRouteName = path.slice(pagesDir.length + 1);
|
||||
let routePath = Path.posix.join(preRouteUrl, (fileName === 'index' ? '' : fileName.replace(/@/g, ':')));
|
||||
const filePath = Path.resolve(path.replace(pagesDir, outputPageDir), item);
|
||||
let routeName = preRouteName ? `${preRouteName}_${fileName}` : fileName;
|
||||
routeName = routeName.replace(/\//g, '_').replace(/@/g, '');
|
||||
routePath = routePath.replace(/@/g, ':');
|
||||
|
||||
if (hasLayout && fileName === 'index') {
|
||||
routePath = '/';
|
||||
} else if (hasLayout && fileName !== 'index') {
|
||||
routePath = routePath.split(preRouteUrl)[1];
|
||||
}
|
||||
components.push({
|
||||
name: routeName,
|
||||
path: filePath.replace(/\\/g, '\\\\')
|
||||
});
|
||||
if (fileName === 'layout') {
|
||||
parentRoutes[preRouteUrl].component = routeName;
|
||||
return;
|
||||
}
|
||||
childRoutes[routePath] = {
|
||||
name: routeName || 'index',
|
||||
component: routeName
|
||||
};
|
||||
}
|
||||
});
|
||||
preRoutes = Object.assign(preRoutes, parentRoutes);
|
||||
const toNextRoutes = hasLayout ? childRoutes : preRoutes;
|
||||
|
||||
dirList.forEach((item) => {
|
||||
if (fs.statSync(`${path}/${item}`).isDirectory()) {
|
||||
let toNextPreRouteUrl = Path.posix.join(prePathUrl, item);
|
||||
if (hasLayout) {
|
||||
toNextPreRouteUrl = Path.posix.join('/', item);
|
||||
}
|
||||
genRoute(`${path}/${item}`, toNextPreRouteUrl, toNextRoutes, outputPageDir, components);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
function fixRouter2(routes, newRoutes, f) {
|
||||
Object.keys(routes).forEach((p) => {
|
||||
const item = {
|
||||
path: f ? p.slice(1) : p,
|
||||
component: routes[p].component
|
||||
};
|
||||
if (routes[p].name) {
|
||||
item.name = routes[p].name;
|
||||
}
|
||||
if (routes[p].subRoutes) {
|
||||
item.children = [];
|
||||
fixRouter2(routes[p].subRoutes, item.children, true);
|
||||
}
|
||||
newRoutes.push(item);
|
||||
});
|
||||
}
|
||||
|
||||
module.exports = function (_pagesDir, _outputPageDir) {
|
||||
const routes = {};
|
||||
const newRoutes = [];
|
||||
components = [];
|
||||
pagesDir = _pagesDir;
|
||||
outputPageDir = _outputPageDir;
|
||||
genRoute(pagesDir, '/', routes);
|
||||
fixRouter2(routes, newRoutes);
|
||||
return {
|
||||
routes,
|
||||
newRoutes,
|
||||
components
|
||||
};
|
||||
};
|
25
packages/fes-cli/build/tasks/build.js
Normal file
@ -0,0 +1,25 @@
|
||||
const webpack = require('webpack');
|
||||
const log = require('../helpers/log');
|
||||
const createProdConfig = require('../configs/webpack.config');
|
||||
|
||||
const generateRoute = require('./route');
|
||||
const generateComponent = require('./components');
|
||||
|
||||
function startBuild(config) {
|
||||
try {
|
||||
generateRoute(config);
|
||||
generateComponent(config);
|
||||
const webpackConfig = createProdConfig(config, webpack, 'build');
|
||||
webpack(webpackConfig, (err) => {
|
||||
if (err) {
|
||||
log.error(err);
|
||||
return;
|
||||
}
|
||||
console.log('[build] success');
|
||||
});
|
||||
} catch (e) {
|
||||
log.error(e);
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = startBuild;
|
45
packages/fes-cli/build/tasks/components.js
Normal file
@ -0,0 +1,45 @@
|
||||
const render = require('json-templater/string');
|
||||
const path = require('path');
|
||||
const endOfLine = require('os').EOL;
|
||||
const fs = require('fs-extra');
|
||||
const getCommonComponent = require('../preComplie/components');
|
||||
|
||||
function generateComponent(config) {
|
||||
const OUTPUT_PATH = path.resolve(config.folders.PROJECT_CACHE_DIR, 'commonComp.js');
|
||||
const IMPORT_TEMPLATE = 'import {{name}} from \'{{path}}\';';
|
||||
const LIST_TEMPLATE = ' {{name}}';
|
||||
|
||||
const MAIN_TEMPLATE = `
|
||||
/**
|
||||
* 全局组件配置输出
|
||||
*/
|
||||
{{include}}
|
||||
|
||||
export default {
|
||||
{{list}}
|
||||
};
|
||||
`;
|
||||
|
||||
const components = getCommonComponent(config.folders.PROJECT_CPN_DIR, config.folders.PROJECT_CPN_DIR);
|
||||
|
||||
const componentsTemplate = [];
|
||||
const listTemplate = [];
|
||||
components.forEach((item) => {
|
||||
componentsTemplate.push(render(IMPORT_TEMPLATE, {
|
||||
name: item.tagName,
|
||||
path: item.path
|
||||
}));
|
||||
listTemplate.push(render(LIST_TEMPLATE, {
|
||||
name: item.tagName
|
||||
}));
|
||||
});
|
||||
|
||||
const template = render(MAIN_TEMPLATE, {
|
||||
include: componentsTemplate.join(endOfLine),
|
||||
list: listTemplate.join(`,${endOfLine}`)
|
||||
});
|
||||
|
||||
fs.outputFileSync(OUTPUT_PATH, template);
|
||||
}
|
||||
|
||||
module.exports = generateComponent;
|
63
packages/fes-cli/build/tasks/dev.js
Normal file
@ -0,0 +1,63 @@
|
||||
const path = require('path');
|
||||
const chokidar = require('chokidar');
|
||||
const webpack = require('webpack');
|
||||
const createDevServer = require('../helpers/createDevServer');
|
||||
const getPort = require('../helpers/getPort');
|
||||
const log = require('../helpers/log');
|
||||
const createDevConfig = require('../configs/webpack.config');
|
||||
|
||||
const generateRoute = require('./route');
|
||||
const generateComponent = require('./components');
|
||||
|
||||
|
||||
function routeHandle(config) {
|
||||
generateRoute(config);
|
||||
// 监听pages变化重新生成路由
|
||||
const pagesWatcher = chokidar.watch(path.resolve(config.folders.PROJECT_DIR, './src/pages'));
|
||||
pagesWatcher.on('ready', () => {
|
||||
pagesWatcher.on('add', (filePath) => {
|
||||
if (path.extname(filePath) === '.fes' || path.extname(filePath) === '.vue') {
|
||||
generateRoute(config);
|
||||
}
|
||||
}).on('unlink', (filePath) => {
|
||||
if (path.extname(filePath) === '.fes' || path.extname(filePath) === '.vue') {
|
||||
generateRoute(config);
|
||||
}
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
function globalComponentHandle(config) {
|
||||
generateComponent(config);
|
||||
// 监听components变化重新生成组件注入文件
|
||||
const compWatcher = chokidar.watch(path.resolve(config.folders.PROJECT_DIR, './src/components'));
|
||||
compWatcher.on('ready', () => {
|
||||
compWatcher.on('add', (filePath) => {
|
||||
if (path.extname(filePath) === '.fes' || path.extname(filePath) === '.vue') {
|
||||
generateComponent(config);
|
||||
}
|
||||
}).on('unlink', (filePath) => {
|
||||
if (path.extname(filePath) === '.fes' || path.extname(filePath) === '.vue') {
|
||||
generateComponent(config);
|
||||
}
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
function startDev(config) {
|
||||
routeHandle(config);
|
||||
globalComponentHandle(config);
|
||||
const webpackConfig = createDevConfig(config, webpack, 'dev');
|
||||
if (!webpackConfig) return;
|
||||
|
||||
getPort(config.ports.server)
|
||||
.then((port) => {
|
||||
log.message(`------------ find port success. port: ${port}`);
|
||||
createDevServer(port, webpackConfig);
|
||||
}).catch((err) => {
|
||||
log.message('------------ build error.');
|
||||
log.error(err);
|
||||
});
|
||||
}
|
||||
|
||||
module.exports = startDev;
|
15
packages/fes-cli/build/tasks/index.js
Normal file
@ -0,0 +1,15 @@
|
||||
const init = require('./init.js');
|
||||
const route = require('./route.js');
|
||||
const components = require('./components.js');
|
||||
const build = require('./build.js');
|
||||
const dev = require('./dev.js');
|
||||
const update = require('./update.js');
|
||||
|
||||
module.exports = {
|
||||
init,
|
||||
route,
|
||||
components,
|
||||
build,
|
||||
dev,
|
||||
update
|
||||
};
|
54
packages/fes-cli/build/tasks/init.js
Normal file
@ -0,0 +1,54 @@
|
||||
const path = require('path');
|
||||
const fs = require('fs-extra');
|
||||
const prompts = require('prompts');
|
||||
const { exec } = require('child_process');
|
||||
const log = require('../helpers/log');
|
||||
|
||||
|
||||
function createProject(config, projectName) {
|
||||
log.message('正在初始化项目...');
|
||||
const projectDir = path.resolve(config.folders.PROJECT_DIR, projectName);
|
||||
if (fs.pathExistsSync(projectDir)) {
|
||||
log.error('该项目已存在,请重新输入!');
|
||||
return Promise.reject();
|
||||
}
|
||||
return new Promise((resolve, reject) => {
|
||||
fs.copy(`${config.folders.CLI_DIR}/template`, `${config.folders.PROJECT_DIR}/${projectName}`).then(() => {
|
||||
exec(`cd ${config.folders.PROJECT_DIR}/${projectName} && git init && npm i @webank/fes-core @webank/fes-ui && npm i`, (err) => {
|
||||
if (err) {
|
||||
log.error(err);
|
||||
reject(err);
|
||||
return;
|
||||
}
|
||||
log.message(`项目 ${projectName} 创建完成,请执行下面的命令进行使用:`);
|
||||
log.message(`$ cd ${projectName}`);
|
||||
log.message('$ npm run dev');
|
||||
resolve();
|
||||
});
|
||||
}).catch((err) => {
|
||||
log.error(err);
|
||||
reject(err);
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
async function initProject(config, projectName) {
|
||||
if (projectName) {
|
||||
await createProject(config, projectName);
|
||||
} else {
|
||||
const response = await prompts([
|
||||
{
|
||||
type: 'text',
|
||||
name: 'name',
|
||||
message: '请输入项目名称: '
|
||||
}
|
||||
]);
|
||||
if (!response.name) {
|
||||
await initProject(config, projectName);
|
||||
} else {
|
||||
await createProject(config, response.name);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = initProject;
|
35
packages/fes-cli/build/tasks/route.js
Normal file
@ -0,0 +1,35 @@
|
||||
const render = require('json-templater/string');
|
||||
const path = require('path');
|
||||
const endOfLine = require('os').EOL;
|
||||
const fs = require('fs-extra');
|
||||
const getRoute = require('../preComplie/route');
|
||||
|
||||
function generateRoute(config) {
|
||||
const OUTPUT_PATH = path.resolve(config.folders.PROJECT_CACHE_DIR, 'routeConfig.js');
|
||||
const IMPORT_TEMPLATE = 'import {{name}} from \'{{path}}\';';
|
||||
|
||||
const MAIN_TEMPLATE = `
|
||||
{{include}}
|
||||
|
||||
export default {{routes}};
|
||||
`;
|
||||
|
||||
const routes = getRoute(config.folders.PROJECT_PAGE_DIR, config.folders.PROJECT_PAGE_DIR);
|
||||
|
||||
const componentsTemplate = [];
|
||||
routes.components.forEach((item) => {
|
||||
componentsTemplate.push(render(IMPORT_TEMPLATE, {
|
||||
name: item.name,
|
||||
path: item.path
|
||||
}));
|
||||
});
|
||||
|
||||
const template = render(MAIN_TEMPLATE, {
|
||||
include: componentsTemplate.join(endOfLine),
|
||||
routes: JSON.stringify(routes.newRoutes).replace(/"component":"(.+?)"/g, '"component": $1')
|
||||
});
|
||||
|
||||
fs.outputFileSync(OUTPUT_PATH, template);
|
||||
}
|
||||
|
||||
module.exports = generateRoute;
|
15
packages/fes-cli/build/tasks/update.js
Normal file
@ -0,0 +1,15 @@
|
||||
const { exec } = require('child_process');
|
||||
const log = require('../helpers/log');
|
||||
|
||||
function update(config) {
|
||||
log.message('安装@webank/fes-core @webank/fes-ui...');
|
||||
exec(`cd ${config.folders.PROJECT_DIR} && npm i @webank/fes-core @webank/fes-ui --save && npm i`, (err) => {
|
||||
if (err) {
|
||||
console.error(err);
|
||||
return;
|
||||
}
|
||||
log.message('升级完毕');
|
||||
});
|
||||
}
|
||||
|
||||
module.exports = update;
|
7
packages/fes-cli/change.md
Normal file
@ -0,0 +1,7 @@
|
||||
# 变更
|
||||
|
||||
* 改用 commander 描述命令行参数
|
||||
* 删除命令选项 https,(实际上代码中没有用到)
|
||||
* 更改 configs >> config
|
||||
* init 模版使用 cli 自带模版,fes-core/fes-ui 创建项目时安装,新项目都采用最新的版本
|
||||
* 该更 route | components 生成方式(使其不依赖 gulp 的编译)
|
98
packages/fes-cli/package.json
Normal file
@ -0,0 +1,98 @@
|
||||
{
|
||||
"name": "@webank/fes-cli",
|
||||
"version": "0.1.1",
|
||||
"description": "一个好用的前端管理台快速开发框架",
|
||||
"preferGlobal": true,
|
||||
"scripts": {
|
||||
"test": "echo \"Error: no test specified\" && exit 1"
|
||||
},
|
||||
"bin": {
|
||||
"fes": "./bin/index.js"
|
||||
},
|
||||
"author": "harrywan,qlin",
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/WeBankFinTech/fes.js.git"
|
||||
},
|
||||
"license": "MIT",
|
||||
"keywords": [
|
||||
"管理端",
|
||||
"fes",
|
||||
"fast",
|
||||
"easy",
|
||||
"strong"
|
||||
],
|
||||
"dependencies": {
|
||||
"@babel/core": "^7.6.4",
|
||||
"@babel/helper-module-imports": "^7.7.0",
|
||||
"@babel/plugin-proposal-class-properties": "^7.7.0",
|
||||
"@babel/plugin-proposal-decorators": "^7.7.0",
|
||||
"@babel/plugin-proposal-object-rest-spread": "^7.6.2",
|
||||
"@babel/plugin-syntax-dynamic-import": "^7.2.0",
|
||||
"@babel/plugin-transform-runtime": "^7.6.2",
|
||||
"@babel/preset-env": "^7.6.3",
|
||||
"@babel/runtime": "^7.7.2",
|
||||
"@intervolga/optimize-cssnano-plugin": "^1.0.6",
|
||||
"autoprefixer": "^8.1.0",
|
||||
"babel-loader": "^8.0.6",
|
||||
"babel-polyfill": "^6.26.0",
|
||||
"body-parser": "^1.5.2",
|
||||
"cache-loader": "^4.1.0",
|
||||
"case-sensitive-paths-webpack-plugin": "^2.2.0",
|
||||
"chalk": "^1.1.1",
|
||||
"chokidar": "^1.7.0",
|
||||
"clean-webpack-plugin": "^3.0.0",
|
||||
"commander": "^4.1.0",
|
||||
"cookie-parser": "^1.4.3",
|
||||
"copy-webpack-plugin": "^5.0.4",
|
||||
"cross-spawn": "^2.1.0",
|
||||
"css-loader": "^3.1.0",
|
||||
"execa": "^0.8.0",
|
||||
"express": "^4.14.0",
|
||||
"express-http-proxy": "^0.10.0",
|
||||
"express-session": "^1.7.2",
|
||||
"faker": "^4.1.0",
|
||||
"file-loader": "^4.2.0",
|
||||
"friendly-errors-webpack-plugin": "^1.7.0",
|
||||
"fs": "0.0.2",
|
||||
"fs-extra": "^8.1.0",
|
||||
"hash-sum": "^2.0.0",
|
||||
"html-webpack-plugin": "^3.2.0",
|
||||
"http-proxy": "^1.12.0",
|
||||
"is-ci": "^1.0.10",
|
||||
"json-templater": "^1.2.0",
|
||||
"lodash": "^4.17.4",
|
||||
"mini-css-extract-plugin": "^0.8.0",
|
||||
"mockjs": "^1.1.0",
|
||||
"morgan": "^1.2.2",
|
||||
"node-plus-string": "^1.0.1",
|
||||
"node-sass": "^4.13.0",
|
||||
"normalize-path": "^1.0.0",
|
||||
"on-finished": "^2.3.0",
|
||||
"opn": "^4.0.2",
|
||||
"path": "^0.12.7",
|
||||
"postcss-loader": "^3.0.0",
|
||||
"prompts": "^2.3.0",
|
||||
"request": "^2.81.0",
|
||||
"require-dir": "^0.3.0",
|
||||
"sass-loader": "^8.0.0",
|
||||
"shelljs": "^0.5.3",
|
||||
"string-replace-loader": "^2.2.0",
|
||||
"strip-indent": "^2.0.0",
|
||||
"style-loader": "^1.0.0",
|
||||
"tar-fs": "^1.16.0",
|
||||
"terser-webpack-plugin": "^2.2.1",
|
||||
"thread-loader": "^2.1.3",
|
||||
"url-loader": "^2.2.0",
|
||||
"vue-loader": "^15.7.2",
|
||||
"vue-style-loader": "^4.1.2",
|
||||
"vue-template-compiler": "^2.6.10",
|
||||
"webpack": "^4.41.2",
|
||||
"webpack-cli": "^3.3.9",
|
||||
"webpack-dev-middleware": "^3.7.2",
|
||||
"webpack-dev-server": "^3.9.0",
|
||||
"webpack-hot-middleware": "^2.25.0",
|
||||
"webpack-merge": "^4.2.2",
|
||||
"yargs": "^3.31.0"
|
||||
}
|
||||
}
|
17
packages/fes-cli/template/.eslintrc.js
Normal file
@ -0,0 +1,17 @@
|
||||
|
||||
module.exports = {
|
||||
extends: [
|
||||
'@webank/eslint-config-webank/vue',
|
||||
],
|
||||
globals: {
|
||||
// 这里填入你的项目需要的全局变量
|
||||
// 这里值为 false 表示这个全局变量不允许被重新赋值,比如:
|
||||
//
|
||||
// Vue: false
|
||||
},
|
||||
rules: {
|
||||
'no-plusplus': 'off',
|
||||
'no-bitwise': 'off',
|
||||
'vue/comment-directive': 'off'
|
||||
}
|
||||
};
|
8
packages/fes-cli/template/.gitignore
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
.DS_Store
|
||||
.idea
|
||||
.git
|
||||
.vscode
|
||||
.cache
|
||||
/dist
|
||||
.history
|
||||
/node_modules
|
21
packages/fes-cli/template/LICENSE
Normal file
@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2020-present harrywan
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
11
packages/fes-cli/template/README.md
Normal file
@ -0,0 +1,11 @@
|
||||
# 项目名称
|
||||
|
||||
## 运行
|
||||
```bash
|
||||
npm run dev
|
||||
```
|
||||
|
||||
## 编译
|
||||
```bash
|
||||
npm run build
|
||||
```
|
140
packages/fes-cli/template/fes.config.js
Normal file
@ -0,0 +1,140 @@
|
||||
module.exports = {
|
||||
mode: 'vertical', // 可选有vertical、horizontal,默认vertical
|
||||
theme: 'blue', // 可选有blue、dark,默认blue
|
||||
fesName: 'xx 运营平台', // 项目名称
|
||||
favicon: 'static/favicon.ico', // 图标
|
||||
// 环境变量配置, 默认使用local环境
|
||||
env: {
|
||||
// 本地开发环境
|
||||
local: {
|
||||
api: ''
|
||||
},
|
||||
// 测试环境 --env=sit 触发使用
|
||||
develop: {
|
||||
api: ''
|
||||
},
|
||||
// 生产环境 --env=prod 触发使用
|
||||
prod: {
|
||||
api: ''
|
||||
}
|
||||
},
|
||||
// 配置角色-路由访问权限,使用FesApp.setRole('unLogin')来修改当前用户的角色,控制路由访问权限
|
||||
roles: {
|
||||
unLogin: ['/home'],
|
||||
service: ['/list', '/home'],
|
||||
admin: ['/list', '/api/fes']
|
||||
},
|
||||
// map
|
||||
map: {
|
||||
status: [['1', '成功'], ['2', '失败']]
|
||||
},
|
||||
// 左侧菜单配置
|
||||
menu: [
|
||||
{
|
||||
title: '$i18n.menu.interface',
|
||||
path: '/api',
|
||||
subMenu: [
|
||||
{
|
||||
title: 'Fes',
|
||||
path: '/api/fes'
|
||||
},
|
||||
{
|
||||
title: 'FesApp',
|
||||
path: '/api/fesApp'
|
||||
},
|
||||
{
|
||||
title: 'FesApi',
|
||||
path: '/api/fesApi'
|
||||
},
|
||||
{
|
||||
title: 'FesMenu',
|
||||
path: '/api/fesMenu'
|
||||
},
|
||||
{
|
||||
title: 'FesMap',
|
||||
path: '/api/fesMap'
|
||||
},
|
||||
{
|
||||
title: 'FesFesx',
|
||||
path: '/api/fesFesx'
|
||||
},
|
||||
{
|
||||
title: 'FesStorage',
|
||||
path: '/api/fesStorage'
|
||||
},
|
||||
{
|
||||
title: 'FesUtil',
|
||||
path: '/api/fesUtil'
|
||||
}
|
||||
]
|
||||
},
|
||||
{
|
||||
title: '路由',
|
||||
path: '/route'
|
||||
},
|
||||
{
|
||||
icon: 'static/bell.png',
|
||||
title: '列表页',
|
||||
path: '/list'
|
||||
},
|
||||
{
|
||||
title: '内容很多的编辑',
|
||||
path: '/list/edit'
|
||||
},
|
||||
{
|
||||
title: '显示头部',
|
||||
path: '/header'
|
||||
},
|
||||
{
|
||||
title: '静态资源',
|
||||
path: '/static'
|
||||
},
|
||||
{
|
||||
title: '子路由',
|
||||
path: '/layout',
|
||||
subMenu: [
|
||||
{
|
||||
title: '子路由A',
|
||||
path: '/layout/a'
|
||||
},
|
||||
{
|
||||
title: '子路由B',
|
||||
path: '/layout/b'
|
||||
}
|
||||
]
|
||||
},
|
||||
{
|
||||
title: '国际化',
|
||||
path: '/i18n'
|
||||
}
|
||||
],
|
||||
i18n: {
|
||||
locale: 'en', // default zh-cn
|
||||
messages: {
|
||||
'zh-cn': {
|
||||
menu: {
|
||||
interface: '接口'
|
||||
},
|
||||
overview: '概述',
|
||||
i18n: {
|
||||
internationalization: '国际化,基于',
|
||||
achieve: '实现。',
|
||||
ui: 'UI组件'
|
||||
},
|
||||
title: '标题'
|
||||
},
|
||||
en: {
|
||||
menu: {
|
||||
interface: 'interface'
|
||||
},
|
||||
overview: 'Overview',
|
||||
i18n: {
|
||||
internationalization: 'internationalization,base on',
|
||||
achieve: 'to achieve.',
|
||||
ui: 'UI components'
|
||||
},
|
||||
title: 'title'
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
113
packages/fes-cli/template/mock.js
Normal file
@ -0,0 +1,113 @@
|
||||
module.exports = (cgiMock, Mock) => {
|
||||
const { Random } = Mock;
|
||||
|
||||
// 前缀,全局(可选)
|
||||
// cgiMock.prefix = '';
|
||||
|
||||
// 返回一个数字
|
||||
cgiMock('/number', 123);
|
||||
|
||||
// 返回一个json
|
||||
cgiMock({
|
||||
url: '/json',
|
||||
result: {
|
||||
code: '400101', msg: "不合法的请求:Missing cookie 'wb_app_id' for method parameter of type String", transactionTime: '20170309171146', success: false
|
||||
}
|
||||
});
|
||||
|
||||
// 利用mock.js 产生随机文本
|
||||
cgiMock('/text', Random.cparagraph());
|
||||
|
||||
// 返回一个字符串 利用mock.js 产生随机字符
|
||||
cgiMock('/string', Mock.mock({
|
||||
'string|1-10': '★'
|
||||
}));
|
||||
|
||||
|
||||
// 正则匹配url, 返回一个字符串
|
||||
// cgiMock(/\/abc|\/xyz/, 'regexp test!');
|
||||
|
||||
// option.result 参数如果是一个函数, 可以实现自定义返回内容, 接收的参数是是经过 express 封装的 req 和 res 对象.
|
||||
// cgiMock(/\/function$/, function (req, res) {
|
||||
// res.send('function test');
|
||||
// });
|
||||
|
||||
// 返回文本 fs.readFileSync
|
||||
// cgiMock('/file', cgiMock.file('./test.json'));
|
||||
|
||||
// 更复杂的规则配置
|
||||
cgiMock({
|
||||
url: /\/who/,
|
||||
method: 'GET',
|
||||
result(req, res) {
|
||||
if (req.query.name === 'kwan') {
|
||||
res.json({ kwan: '孤独患者' });
|
||||
} else {
|
||||
res.send('Nooooooooooo');
|
||||
}
|
||||
},
|
||||
headers: {
|
||||
'Content-Type': 'text/plain',
|
||||
'Content-Length': '123',
|
||||
ETag: '12345'
|
||||
},
|
||||
cookies: [
|
||||
{
|
||||
name: 'myname', value: 'kwan', maxAge: 900000, httpOnly: true
|
||||
}
|
||||
],
|
||||
// 接口随机延迟
|
||||
timeout: Mock.mock({
|
||||
'number|1000-5000': 1000
|
||||
}).number
|
||||
});
|
||||
// 登录
|
||||
cgiMock('/login', (req, res) => {
|
||||
res.send(JSON.stringify({
|
||||
code: '0',
|
||||
msg: '',
|
||||
result: {
|
||||
username: '万纯(harrywan)',
|
||||
roleName: '管理员'
|
||||
}
|
||||
}));
|
||||
});
|
||||
|
||||
cgiMock('/getTestList', (req, res) => {
|
||||
const list = [];
|
||||
for (let i = 0; i < req.body.pageSize; i++) {
|
||||
list.push({
|
||||
a: i
|
||||
});
|
||||
}
|
||||
res.send(JSON.stringify({
|
||||
code: '0',
|
||||
msg: 'this is message',
|
||||
result: {
|
||||
list,
|
||||
page: {
|
||||
pageSize: req.body.pageSize,
|
||||
currentPage: req.body.currentPage,
|
||||
totalPage: 1000
|
||||
}
|
||||
}
|
||||
|
||||
}));
|
||||
});
|
||||
|
||||
cgiMock('/getNumber', (req, res) => {
|
||||
res.send(JSON.stringify({
|
||||
code: '0',
|
||||
msg: 'this is message',
|
||||
result: 4
|
||||
}));
|
||||
});
|
||||
|
||||
cgiMock('/getRoleName', (req, res) => {
|
||||
res.send(JSON.stringify({
|
||||
code: '0',
|
||||
msg: 'this is message',
|
||||
result: 'admin'
|
||||
}));
|
||||
});
|
||||
};
|
57
packages/fes-cli/template/package.json
Normal file
@ -0,0 +1,57 @@
|
||||
{
|
||||
"name": "@webank/fes-template",
|
||||
"version": "0.1.0",
|
||||
"description": "fes项目模版",
|
||||
"main": "index.js",
|
||||
"scripts": {
|
||||
"build": "fes build",
|
||||
"dev": "fes dev"
|
||||
},
|
||||
"husky": {
|
||||
"hooks": {
|
||||
"pre-commit": "lint-staged"
|
||||
}
|
||||
},
|
||||
"lint-staged": {
|
||||
"*.js": [
|
||||
"eslint",
|
||||
"git add"
|
||||
],
|
||||
"*.fes": [
|
||||
"eslint",
|
||||
"git add"
|
||||
],
|
||||
"*.vue": [
|
||||
"eslint",
|
||||
"git add"
|
||||
]
|
||||
},
|
||||
"keywords": [
|
||||
"管理端",
|
||||
"fes",
|
||||
"fast",
|
||||
"easy",
|
||||
"strong"
|
||||
],
|
||||
"files": [
|
||||
".eslintrc.js",
|
||||
".gitignore",
|
||||
"fes.config.js",
|
||||
"mock.js",
|
||||
"package.json",
|
||||
"README.md",
|
||||
"/src"
|
||||
],
|
||||
"repository": {
|
||||
"type": "git"
|
||||
},
|
||||
"author": "harrywan qlin",
|
||||
"license": "MIT",
|
||||
"devDependencies": {
|
||||
"@webank/eslint-config-webank": "^0.1.4",
|
||||
"husky": "^3.0.9",
|
||||
"lint-staged": "^9.4.2"
|
||||
},
|
||||
"dependencies": {},
|
||||
"peerDependencies": {}
|
||||
}
|
45
packages/fes-cli/template/src/app.js
Normal file
@ -0,0 +1,45 @@
|
||||
import './assets/styles/main.scss';
|
||||
|
||||
export default function () {
|
||||
this.FesApp.set('FesName', '$i18n.title');
|
||||
|
||||
setTimeout(() => {
|
||||
this.FesApp.setRole('admin', false);
|
||||
}, 1000);
|
||||
|
||||
// 设置退出逻辑
|
||||
this.on('fes_logout', () => {
|
||||
this.FesApp.setRole('unLogin');
|
||||
this.FesStorage.set('userLogin', false);
|
||||
});
|
||||
|
||||
// 设置logo点击事件
|
||||
this.on('fes_logo_click', () => {
|
||||
window.Toast('你点击了LOGO');
|
||||
});
|
||||
|
||||
// 设置路由钩子
|
||||
this.FesApp.setBeforeRouter((from, to, next) => {
|
||||
next();
|
||||
});
|
||||
this.FesApp.setAfterRouter((route) => {
|
||||
console.log(`您浏览到了${route.path}`);
|
||||
});
|
||||
|
||||
// // 设置当前角色
|
||||
// if (!this.FesStorage.get('userLogin') === true) {
|
||||
// this.setRole('unLogin')
|
||||
// }
|
||||
|
||||
// 设置AJAX配置
|
||||
this.FesApi.option({
|
||||
});
|
||||
|
||||
// 设置响应结构
|
||||
this.FesApi.setResponse({
|
||||
successCode: '0',
|
||||
codePath: 'code',
|
||||
messagePath: 'msg',
|
||||
resultPath: 'result'
|
||||
});
|
||||
}
|
BIN
packages/fes-cli/template/src/assets/images/bg.png
Normal file
After Width: | Height: | Size: 5.9 KiB |
BIN
packages/fes-cli/template/src/assets/images/logo.png
Normal file
After Width: | Height: | Size: 2.7 KiB |
131
packages/fes-cli/template/src/assets/styles/login.scss
Normal file
@ -0,0 +1,131 @@
|
||||
.login-panel {
|
||||
position: absolute;
|
||||
left: 0;
|
||||
top: 0;
|
||||
margin: 0;
|
||||
right: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
background-color: $body-background;
|
||||
background-image: url("../images/bg.png");
|
||||
background-position: left bottom;
|
||||
background-repeat: no-repeat;
|
||||
background-size: 100% auto;
|
||||
font-size: 18px;
|
||||
color: $black-text-color;
|
||||
}
|
||||
|
||||
.login-panel .login-panel-swap {
|
||||
position: relative;
|
||||
width: 945px;
|
||||
height: 324px;
|
||||
margin: 10% auto;
|
||||
border-radius: $border-radius-small;
|
||||
}
|
||||
|
||||
.login-panel .logo {
|
||||
display: inline-block;
|
||||
vertical-align: middle;
|
||||
width: 400px;
|
||||
height: 100%;
|
||||
text-align: center;
|
||||
.logo-text {
|
||||
display: inline-block;
|
||||
vertical-align: middle;
|
||||
margin-top: 130px;
|
||||
font-size: 28px;
|
||||
font-weight: bold;
|
||||
}
|
||||
}
|
||||
|
||||
.login-panel .split {
|
||||
display: inline-block;
|
||||
background: $border-color-base;
|
||||
width: 1px;
|
||||
height: 180px;
|
||||
vertical-align: middle;
|
||||
}
|
||||
|
||||
.login-panel .error {
|
||||
margin-top: 10px;
|
||||
width: 350px;
|
||||
color: $error-color;
|
||||
font-size: 14px;
|
||||
.ui-icon-exclamation-circle{
|
||||
margin-right: 6px;
|
||||
}
|
||||
}
|
||||
|
||||
.login-panel .login-form {
|
||||
position: relative;
|
||||
display: inline-block;
|
||||
vertical-align: middle;
|
||||
padding: 0 75px 0;
|
||||
height: 270px;
|
||||
input::-webkit-input-placeholder{
|
||||
color: $black-text-color;
|
||||
}
|
||||
input::-moz-placeholder {
|
||||
color: $black-text-color;
|
||||
}
|
||||
/* ie */
|
||||
input:-ms-input-placeholder {
|
||||
color: $black-text-color;
|
||||
}
|
||||
/* firefox 19+ */
|
||||
input:-moz-placeholder {
|
||||
color: $black-text-color;
|
||||
}
|
||||
|
||||
input:-webkit-autofill {
|
||||
-webkit-box-shadow: 0 0 0px 1000px white inset !important;
|
||||
-webkit-text-fill-color: $black-text-color !important;
|
||||
}
|
||||
.line{
|
||||
padding-top: 30px;
|
||||
border-bottom: 1px solid $border-color-split;
|
||||
}
|
||||
input[type="text"],
|
||||
input[type="password"]{
|
||||
margin: 0;
|
||||
width: 350px;
|
||||
height: 40px;
|
||||
line-height: 40px;
|
||||
padding: 0 15px;
|
||||
vertical-align: middle;
|
||||
background: 0 0;
|
||||
outline: 0;
|
||||
border: none;
|
||||
font-size: 16px;
|
||||
color: $black-text-color;
|
||||
}
|
||||
button {
|
||||
margin-top: 10px;
|
||||
width: 350px;
|
||||
height: 48px;
|
||||
border-radius: 4px;
|
||||
line-height: 46px;
|
||||
border: none;
|
||||
outline: 0;
|
||||
background-color: $primary-color;
|
||||
color: #ffffff;
|
||||
cursor: pointer;
|
||||
text-align: center;
|
||||
font-size: 18px;
|
||||
&:hover {
|
||||
background-color: $selected-color;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.ie-palceholder::after{
|
||||
content: '用户名';
|
||||
position: absolute;
|
||||
left: 88px;
|
||||
top: 38px;
|
||||
font-size: 16px;
|
||||
}
|
||||
.ie-palceholder-password::after{
|
||||
content: '密码';
|
||||
top: 110px;
|
||||
}
|
50
packages/fes-cli/template/src/assets/styles/main.scss
Normal file
@ -0,0 +1,50 @@
|
||||
@import "variables";
|
||||
@import "login";
|
||||
|
||||
.article {
|
||||
padding: 20px;
|
||||
h1 {
|
||||
font-size: 26px;
|
||||
font-weight: 400;
|
||||
margin: 12px 0;
|
||||
}
|
||||
|
||||
h2 {
|
||||
margin: 25px 0 12px;
|
||||
font-size: 20px;
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
h3 {
|
||||
font-size: 16px;
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
p {
|
||||
font-size: 14px;
|
||||
margin: 5px;
|
||||
}
|
||||
|
||||
ul {
|
||||
padding-left: 40px;
|
||||
}
|
||||
|
||||
li {
|
||||
list-style-type: disc;
|
||||
margin-bottom: 5px;
|
||||
font-size: 14px;
|
||||
}
|
||||
table{
|
||||
border-collapse: collapse;
|
||||
border-spacing: 0;
|
||||
empty-cells: show;
|
||||
border: 1px solid #e9e9e9;
|
||||
width: 500px;
|
||||
margin-bottom: 24px;
|
||||
}
|
||||
th,td {
|
||||
border: 1px solid #e9e9e9;
|
||||
padding: 8px 16px;
|
||||
text-align: left;
|
||||
}
|
||||
}
|
90
packages/fes-cli/template/src/assets/styles/variables.scss
Normal file
@ -0,0 +1,90 @@
|
||||
// Color
|
||||
$primary-color : #3399ff;
|
||||
$info-color : #2db7f5;
|
||||
$success-color : #00cc66;
|
||||
$warning-color : #ff9900;
|
||||
$error-color : #ff5500;
|
||||
$link-color : #3399ff;
|
||||
$link-hover-color : #5cadff;
|
||||
$link-focus-color : rgba(51,153,255, .2);
|
||||
$link-active-color : #3091f2;
|
||||
$selected-color : rgba($primary-color, .9);
|
||||
$tooltip-color : #fff;
|
||||
//辅助/图标
|
||||
$subsidiary-color : #9ea7b4;
|
||||
$disabled-color : #f3f3f3;
|
||||
|
||||
// Base
|
||||
$body-background : #fff;
|
||||
$component-background : #fff;
|
||||
$font-family : "Helvetica Neue",Helvetica,"PingFang SC","Hiragino Sans GB","Microsoft YaHei","微软雅黑",Arial,sans-serif;
|
||||
$code-family : Consolas,Menlo,Courier,monospace;
|
||||
$title-color : #464c5b;
|
||||
$text-color : #657180;
|
||||
$black-text-color : #333333;
|
||||
$sub-text-color : #999;
|
||||
$dark-color : #333;
|
||||
|
||||
//失效 Disabled
|
||||
$tip-color : #c3cbd6;
|
||||
$font-size-lg : 16px;
|
||||
$font-size-base : 14px;
|
||||
$font-size-small : 12px;
|
||||
$line-height-base : 1.5;
|
||||
$line-height-computed : floor(($font-size-base * $line-height-base));
|
||||
$border-radius-base : 6px;
|
||||
$border-radius-small : 4px;
|
||||
$cursor-disabled : not-allowed;
|
||||
|
||||
// Border color
|
||||
$border-color-base : #d7dde4; // outside
|
||||
$border-color-split : #e3e8ee; // inside
|
||||
|
||||
|
||||
// Background color
|
||||
$background-color-base : #f7f7f7; // base
|
||||
$background-color-select-hover: #f3f3f3;
|
||||
$tooltip-bg : rgba(70, 76, 91, .9);
|
||||
$head-bg : #f9fafc;
|
||||
$table-thead-bg : #f5f7f9;
|
||||
$table-td-stripe-bg : #f5f7f9;
|
||||
$table-td-hover-bg : #ebf7ff;
|
||||
$table-td-highlight-bg : #ebf7ff;
|
||||
|
||||
// Z-index
|
||||
$zindex-spin : 8;
|
||||
$zindex-affix : 10;
|
||||
$zindex-back-top : 10;
|
||||
$zindex-select : 900;
|
||||
$zindex-modal : 1000;
|
||||
$zindex-message : 1010;
|
||||
$zindex-notification : 1010;
|
||||
$zindex-tooltip : 1060;
|
||||
$zindex-loading-bar : 2000;
|
||||
|
||||
// Animation
|
||||
$animation-time : .3s;
|
||||
$transition-time : .2s;
|
||||
$ease-out : cubic-bezier(0.215, 0.61, 0.355, 1);
|
||||
$ease-in : cubic-bezier(0.55, 0.055, 0.675, 0.19);
|
||||
$ease-in-out : cubic-bezier(0.645, 0.045, 0.355, 1);
|
||||
$ease-out-back : cubic-bezier(0.12, 0.4, 0.29, 1.46);
|
||||
$ease-in-back : cubic-bezier(0.71, -0.46, 0.88, 0.6);
|
||||
$ease-in-out-back : cubic-bezier(0.71, -0.46, 0.29, 1.46);
|
||||
$ease-out-circ : cubic-bezier(0.08, 0.82, 0.17, 1);
|
||||
$ease-in-circ : cubic-bezier(0.6, 0.04, 0.98, 0.34);
|
||||
$ease-in-out-circ : cubic-bezier(0.78, 0.14, 0.15, 0.86);
|
||||
$ease-out-quint : cubic-bezier(0.23, 1, 0.32, 1);
|
||||
$ease-in-quint : cubic-bezier(0.755, 0.05, 0.855, 0.06);
|
||||
$ease-in-out-quint : cubic-bezier(0.86, 0, 0.07, 1);
|
||||
|
||||
// Shadow
|
||||
$shadow-color : rgba(0, 0, 0, .2);
|
||||
$shadow-1-up : 0 -1px 6px $shadow-color;
|
||||
$shadow-1-down : 0 1px 6px $shadow-color;
|
||||
$shadow-1-left : -1px 0 6px $shadow-color;
|
||||
$shadow-1-right : 1px 0 6px $shadow-color;
|
||||
$shadow-2 : 0 2px 8px $shadow-color;
|
||||
$box-shadow-base : $shadow-1-down;
|
||||
|
||||
$mask-color: rgba(55, 55, 55, .6);
|
14
packages/fes-cli/template/src/components/fesHeader.vue
Normal file
@ -0,0 +1,14 @@
|
||||
<template>
|
||||
<div style="text-align: center">
|
||||
我是头部
|
||||
</div>
|
||||
</template>
|
||||
<script>
|
||||
// components文件夹中的*.fes会根据文件名自动注册为组件,可直接使用
|
||||
// 请勿删除 fesHeader.fes
|
||||
export default {
|
||||
FesReady() {
|
||||
// do something
|
||||
}
|
||||
};
|
||||
</script>
|
9
packages/fes-cli/template/src/components/fesLeft.vue
Normal file
@ -0,0 +1,9 @@
|
||||
<template>
|
||||
<div />
|
||||
</template>
|
||||
<script>
|
||||
// 我是左侧菜单下方的自定义内容
|
||||
// 请勿删除fesLeft.fes
|
||||
export default {
|
||||
};
|
||||
</script>
|
74
packages/fes-cli/template/src/pages/api/fes/index.vue
Normal file
@ -0,0 +1,74 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>对Vue做了一层封装。给Vue添加了一些属性,方便使用。</p>
|
||||
|
||||
<h2>属性</h2>
|
||||
<ul>
|
||||
<li><router-link to="/api/fesApp">this.FesApp App实例</router-link></li>
|
||||
<li><router-link to="/api/fesUtil">this.FesUtil 工具函数</router-link></li>
|
||||
<li><router-link to="/api/fesStorage">this.FesStorage 操作存储: cookie、sessionStorage、localStorage</router-link></li>
|
||||
<li><router-link to="/api/fesApi">this.FesApi 操作Ajax</router-link></li>
|
||||
</ul>
|
||||
|
||||
<h2>响应式数据FesData</h2>
|
||||
<ul>
|
||||
<li>
|
||||
<p>双向绑定的值,类似绑定到ng里面scope的值</p>
|
||||
<p>
|
||||
<input v-model="data1"> {{data1}}
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p><router-link to="/api/fesMap"><strong>数据字典FesMap</strong></router-link></p>
|
||||
<p>FesMap会自动挂载到FesData</p>
|
||||
<p>FesMap: {{FesMap}}</p>
|
||||
</li>
|
||||
<li>
|
||||
<p><router-link to="/api/fesFesx"><strong>全局状态FesFesx</strong></router-link></p>
|
||||
<p>FesFesx会自动挂载到FesData </p>
|
||||
<p>FesFesx: {{FesFesx}}</p>
|
||||
</li>
|
||||
</ul>
|
||||
|
||||
|
||||
<h2>异步响应数据FesSyncData</h2>
|
||||
<p>进入路由时根据接口查询到的值,并且设置成响应数据</p>
|
||||
<p>
|
||||
<input v-model="data2">
|
||||
</p>
|
||||
<p>
|
||||
data2: {{data2}}
|
||||
</p>
|
||||
|
||||
|
||||
<h2>过滤器</h2>
|
||||
<p>日期: {{date}} => {{date | date}}</p>
|
||||
<p>日期(带参数): {{date}} => {{date | date('yyyy-MM-dd')}}</p>
|
||||
<p>金钱: {{money}} => {{money | money}}</p>
|
||||
<p>银行卡: {{card}} => {{card | card}}</p>
|
||||
<p>加密: {{safety}} => {{safety | safety(4,3)}}</p>
|
||||
<p>数据字典: {{map}} => {{map | map(FesMap.status)}}</p>
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
FesData: {
|
||||
data1: '1',
|
||||
money: 100000,
|
||||
card: '11122233242323',
|
||||
safety: '42222991018719191',
|
||||
map: '1',
|
||||
date: new Date().getTime()
|
||||
},
|
||||
FesSyncData: {
|
||||
data2: ['/getNumber', {
|
||||
max: 100,
|
||||
min: 1
|
||||
}]
|
||||
},
|
||||
FesReady() {
|
||||
// do something
|
||||
}
|
||||
};
|
||||
</script>
|
50
packages/fes-cli/template/src/pages/api/fesApi/index.vue
Normal file
@ -0,0 +1,50 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>对vue-resource的封装。公共的异常处理,响应处理等。</p>
|
||||
|
||||
<h2>函数</h2>
|
||||
<ul>
|
||||
<li>
|
||||
<p>get(url, data, option)</p>
|
||||
<p>
|
||||
发起一个get请求,返回值是promise对象
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>post(url, data, option)</p>
|
||||
<p>
|
||||
发起一个post请求,返回值是promise对象
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>fetch(url, data, option)</p>
|
||||
<p>
|
||||
默认调用post请求,返回值是promise对象
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>option(option)</p>
|
||||
<p>
|
||||
设置ajax的公共配置,比如root(根路径)、timeout(超时时间)、xhr(xhr对象)
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>setError(errors)</p>
|
||||
<p>
|
||||
设置当响应状态是非200时触发的事件钩子,比如401啊等
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>setResponse(constructionOfResponse)</p>
|
||||
<p>
|
||||
设置响应结构。响应一般会由状态码、错误消息、数据组成。通过此函数设置一个路径,当响应回来是来解析响应。
|
||||
</p>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
};
|
||||
</script>
|
94
packages/fes-cli/template/src/pages/api/fesApp/index.vue
Normal file
@ -0,0 +1,94 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>一个App表示一个应用。管理一些应用级别的状态。</p>
|
||||
|
||||
<h2>属性</h2>
|
||||
<ul>
|
||||
<li>
|
||||
<router-link to="/api/fesApi">
|
||||
<p>this.FesApi</p>
|
||||
<p>操作ajax的对象</p>
|
||||
</router-link>
|
||||
</li>
|
||||
<li>
|
||||
<router-link to="/api/fesMap">
|
||||
<p>this.FesMap</p>
|
||||
<p>数据字典,保存数据字典的容器,提供查找函数</p>
|
||||
</router-link>
|
||||
</li>
|
||||
<li>
|
||||
<router-link to="/api/fesUtil">
|
||||
<p>this.FesUtil</p>
|
||||
<p>工具函数,操作dom、对象等</p>
|
||||
</router-link>
|
||||
</li>
|
||||
<li>
|
||||
<router-link to="/api/fesFesx">
|
||||
<p>this.FesFesx</p>
|
||||
<p>存储全局状态的容器</p>
|
||||
</router-link>
|
||||
</li>
|
||||
<li>
|
||||
<router-link to="/api/fesStorage">
|
||||
<p>this.FesStorage</p>
|
||||
<p>操作存储: cookie、sessionStorage、localStorage</p>
|
||||
</router-link>
|
||||
</li>
|
||||
<li>
|
||||
<p>this.router</p>
|
||||
<p>当前路由对象,具体api查询vue-router v0.7</p>
|
||||
</li>
|
||||
</ul>
|
||||
|
||||
<h2>函数</h2>
|
||||
<ul>
|
||||
<li>
|
||||
<p>init</p>
|
||||
<p>
|
||||
初始化整个应用,我们只需要在app.js写入自定义的初始化内容,比如设置菜单、项目名等
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>get(prop)</p>
|
||||
<p>
|
||||
根据prop获取在App保存的应用层面的状态值
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>set(prop, value)</p>
|
||||
<p>
|
||||
设置应用层面的状态,包括FesName、FesMenu、FesUserName、FesRoleName、FesLogout
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>setRole(role)</p>
|
||||
<p>
|
||||
我们需要在common/rolesConfig.js中配置角色所属的菜单权限,可以把未登录状态也当作一种角色。根据当前登录状态或者用户设置不同的角色。菜单也会根据角色展示。
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>getAllowPage()</p>
|
||||
<p>
|
||||
返回当前能访问的页面
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>setBeforeRouter(beforeRouter)</p>
|
||||
<p>
|
||||
设置路由切换之前的事件钩子,可以根据条件判断是否阻止切换
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>setAfterRouter(afterRouter)</p>
|
||||
<p>
|
||||
路由切换之后的事件钩子
|
||||
</p>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
};
|
||||
</script>
|
26
packages/fes-cli/template/src/pages/api/fesFesx/index.vue
Normal file
@ -0,0 +1,26 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>全局的状态管理容器</p>
|
||||
|
||||
<h2>函数</h2>
|
||||
<ul>
|
||||
<li>
|
||||
<p>get(prop)</p>
|
||||
<p>
|
||||
根据prop获取值
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>set(prop, value)</p>
|
||||
<p>
|
||||
设置一个全局的状态
|
||||
</p>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
};
|
||||
</script>
|
20
packages/fes-cli/template/src/pages/api/fesMap/index.vue
Normal file
@ -0,0 +1,20 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>数据字典管理对象。当程序加载时,从common/map.js中读取原始数据,转换成{value: value, text: text},存入FesMap</p>
|
||||
|
||||
<h2>函数</h2>
|
||||
<ul>
|
||||
<li>
|
||||
<p>getNameByValue(name, value)</p>
|
||||
<p>
|
||||
从FesMap[name]中找出值是value的那一条数据
|
||||
</p>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
};
|
||||
</script>
|
129
packages/fes-cli/template/src/pages/api/fesMenu/index.vue
Normal file
@ -0,0 +1,129 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>
|
||||
配置公共左侧菜单
|
||||
<br> 在调用FesApp.setRole(...)之后,FesMenu只显示当前用户可以访问的菜单
|
||||
</p>
|
||||
<h2>API</h2>
|
||||
<p>
|
||||
<table>
|
||||
<thead>
|
||||
<tr>
|
||||
<th style="text-align:left">
|
||||
属性
|
||||
</th>
|
||||
<th style="text-align:left">
|
||||
说明
|
||||
</th>
|
||||
<th style="text-align:left">
|
||||
类型
|
||||
</th>
|
||||
<th style="text-align:left">
|
||||
默认值
|
||||
</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<tr>
|
||||
<td style="text-align:left">
|
||||
title
|
||||
</td>
|
||||
<td style="text-align:left">
|
||||
菜单标题
|
||||
</td>
|
||||
<td style="text-align:left">
|
||||
String
|
||||
</td>
|
||||
<td style="text-align:left">
|
||||
空
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td style="text-align:left">
|
||||
path
|
||||
</td>
|
||||
<td style="text-align:left">
|
||||
菜单项点击后,跳转路径
|
||||
</td>
|
||||
<td style="text-align:left">
|
||||
String
|
||||
</td>
|
||||
<td style="text-align:left">
|
||||
空
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td style="text-align:left">
|
||||
subMenu
|
||||
</td>
|
||||
<td style="text-align:left">
|
||||
子菜单,每项也有path和title
|
||||
</td>
|
||||
<td style="text-align:left">
|
||||
Array
|
||||
</td>
|
||||
<td style="text-align:left">
|
||||
空
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</p>
|
||||
<h2>示例</h2>
|
||||
<pre>
|
||||
//在app.js文件中设置菜单
|
||||
this.set("FesMenu", [{
|
||||
title: "接口",
|
||||
path: '/api',
|
||||
subMenu: [{
|
||||
title: "Fes",
|
||||
path: '/api/fes',
|
||||
},{
|
||||
title: "FesApp",
|
||||
path: '/api/fesApp',
|
||||
},{
|
||||
title: "FesApi",
|
||||
path: '/api/fesApi',
|
||||
},{
|
||||
title: "FesMenu",
|
||||
path: '/api/fesMenu',
|
||||
},{
|
||||
title: "FesMap",
|
||||
path: '/api/fesMap',
|
||||
},{
|
||||
title: "FesFesx",
|
||||
path: '/api/fesFesx',
|
||||
},{
|
||||
title: "FesStorage",
|
||||
path: '/api/fesStorage',
|
||||
},{
|
||||
title: "FesUtil",
|
||||
path: '/api/fesUtil',
|
||||
}]
|
||||
},{
|
||||
title: "简单的列表页",
|
||||
path: '/list'
|
||||
},{
|
||||
title: "自定义内容列表页",
|
||||
path: '/list1'
|
||||
},{
|
||||
title: "内容很多的编辑",
|
||||
path: '/list/edit'
|
||||
}]
|
||||
);
|
||||
</pre>
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
FesData: {
|
||||
data1: '1',
|
||||
money: 100000,
|
||||
card: '11122233242323',
|
||||
safety: '42222991018719191',
|
||||
map: '1',
|
||||
date: new Date().getTime()
|
||||
}
|
||||
};
|
||||
</script>
|
38
packages/fes-cli/template/src/pages/api/fesStorage/index.vue
Normal file
@ -0,0 +1,38 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>操作Storage,包含cookie、sessionStorage、localStorage。 category值SESSION对应sessionStorage,LOCAL对应localStorage,COOKIE对应cookie</p>
|
||||
|
||||
<h2>函数</h2>
|
||||
<ul>
|
||||
<li>
|
||||
<p>set(key, value, category = SESSION, expired)</p>
|
||||
<p>
|
||||
往Storage中存入一个值
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>get(key, category = SESSION)</p>
|
||||
<p>
|
||||
从Storage中取得值
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>clear(category = SESSION)</p>
|
||||
<p>
|
||||
清除所有值
|
||||
</p>
|
||||
</li>
|
||||
<li>
|
||||
<p>remove(key, category = SESSION)</p>
|
||||
<p>
|
||||
删除key对应的值
|
||||
</p>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
};
|
||||
</script>
|
33
packages/fes-cli/template/src/pages/api/fesUtil/index.vue
Normal file
@ -0,0 +1,33 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>工具集合</p>
|
||||
|
||||
<h2>函数</h2>
|
||||
<ul>
|
||||
<li>getClass</li>
|
||||
<li>hasClass</li>
|
||||
<li>setClass</li>
|
||||
<li>addClass</li>
|
||||
<li>removeClass</li>
|
||||
<li>contains</li>
|
||||
<li>addEvent</li>
|
||||
<li>triggerEvent</li>
|
||||
<li>merge</li>
|
||||
<li>extend</li>
|
||||
<li>isPlainObject</li>
|
||||
<li>isNumber</li>
|
||||
<li>isDate</li>
|
||||
<li>isFunction</li>
|
||||
<li>isObject</li>
|
||||
<li>isArray</li>
|
||||
<li>isObjectLike</li>
|
||||
<li>isString</li>
|
||||
<li>isNull</li>
|
||||
</ul>
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
};
|
||||
</script>
|
77
packages/fes-cli/template/src/pages/header/index.vue
Normal file
@ -0,0 +1,77 @@
|
||||
<template>
|
||||
<div>
|
||||
<Fes-search-panel>
|
||||
<Wb-form :label-width="150" type="query">
|
||||
<Form-item label="姓名:">
|
||||
<wb-input placeholder="请输入" />
|
||||
</Form-item>
|
||||
<Form-item label="身份证:">
|
||||
<wb-input placeholder="请输入" />
|
||||
</Form-item>
|
||||
</Wb-form>
|
||||
<div slot="button">
|
||||
<Wb-button @click="search" type="primary" icon="search">
|
||||
查询
|
||||
</Wb-button>
|
||||
</div>
|
||||
</Fes-search-panel>
|
||||
<Fes-list-panel>
|
||||
<Wb-table :data="data">
|
||||
<Column prop="date" name="日期" />
|
||||
<Column prop="name" name="姓名" />
|
||||
<Column prop="age" name="年龄" />
|
||||
<Column prop="adr" name="地址" />
|
||||
<Column prop="status" name="等级" />
|
||||
</Wb-table>
|
||||
<Pagination :size="paginationOption.pageSize" :current="paginationOption.currentPage" :total="paginationOption.totalPage" @on-change="changePage" />
|
||||
</Fes-list-panel>
|
||||
</div>
|
||||
</template>
|
||||
<script>
|
||||
export default {
|
||||
FesHeader: true,
|
||||
FesData() {
|
||||
return {
|
||||
data: [{
|
||||
name: '张晓刚',
|
||||
age: 24,
|
||||
date: new Date(2016, 9, 10),
|
||||
adr: '北京市海淀区西二旗',
|
||||
status: 1
|
||||
}, {
|
||||
name: '李晓红',
|
||||
age: 26,
|
||||
date: new Date(2016, 9, 11),
|
||||
adr: '北京市海淀区西二旗',
|
||||
status: 2
|
||||
}, {
|
||||
name: '隔壁老王',
|
||||
age: 22,
|
||||
date: new Date(2016, 9, 12),
|
||||
adr: '北京市海淀区西二旗',
|
||||
status: 3
|
||||
}, {
|
||||
name: '我爸是李刚',
|
||||
age: 19,
|
||||
date: new Date(2016, 9, 13),
|
||||
adr: '北京市海淀区西二旗',
|
||||
status: 4
|
||||
}],
|
||||
paginationOption: {
|
||||
pageSize: 10,
|
||||
currentPage: 1,
|
||||
totalPage: 1
|
||||
}
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
search() {
|
||||
// do something
|
||||
},
|
||||
changePage() {
|
||||
// do something
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
</script>
|
142
packages/fes-cli/template/src/pages/home/index.vue
Normal file
@ -0,0 +1,142 @@
|
||||
<template>
|
||||
<div class="login-panel">
|
||||
<div class="login-panel-swap">
|
||||
<div class="logo">
|
||||
<span class="logo-text">
|
||||
xx运营平台
|
||||
</span>
|
||||
</div>
|
||||
<div class="split" />
|
||||
<div class="login-form">
|
||||
<div :class="getStyle('userFocus')" class="line">
|
||||
<input
|
||||
ref="username"
|
||||
v-model="username"
|
||||
@input="input"
|
||||
@keydown.enter="login"
|
||||
@focus="focusHandler('userFocus')"
|
||||
@blur="blurHandler('userFocus')"
|
||||
type="text"
|
||||
name="username"
|
||||
autocomplete="off"
|
||||
autofocus
|
||||
placeholder="用户名"
|
||||
>
|
||||
</div>
|
||||
<div :class="getStyle('passwordFocus')" class="line">
|
||||
<input
|
||||
ref="password"
|
||||
v-model="password"
|
||||
@input="input"
|
||||
@keydown.enter="login"
|
||||
@focus="focusHandler('passwordFocus')"
|
||||
@blur="blurHandler('passwordFocus')"
|
||||
type="password"
|
||||
name="password"
|
||||
autocomplete="off"
|
||||
placeholder="密码"
|
||||
>
|
||||
</div>
|
||||
<div class="line">
|
||||
<button @click="login">
|
||||
登录
|
||||
</button>
|
||||
</div>
|
||||
<div v-show="error" class="error">
|
||||
<Icon type="exclamation-circle" />{{error}}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
<script>
|
||||
export default {
|
||||
FesLeft: false,
|
||||
FesData() {
|
||||
return {
|
||||
userFocus: false,
|
||||
passwordFocus: false,
|
||||
username: '',
|
||||
password: '',
|
||||
error: '' // 请输入正确的密码,8-16位
|
||||
};
|
||||
},
|
||||
FesReady() {
|
||||
this.$nextTick(() => {
|
||||
this.$refs.username.focus();
|
||||
});
|
||||
this.initStyle();
|
||||
|
||||
if (this.FesStorage.get('userLogin') === true) {
|
||||
this.getRole();
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
login() {
|
||||
if (this.validate()) {
|
||||
this.FesApi.fetch('login').then(() => {
|
||||
// 设置用户、角色等
|
||||
this.FesApp.set('FesUserName', '万纯(harrywan)');
|
||||
this.FesApp.set('FesRoleName', '管理员');
|
||||
this.FesStorage.set('userLogin', true);
|
||||
this.getRole();
|
||||
});
|
||||
}
|
||||
},
|
||||
getRole() {
|
||||
this.FesApi.fetch('getRoleName').then((res) => {
|
||||
// 默认跳入rolesConfig的第一项
|
||||
this.FesApp.setRole(res);
|
||||
});
|
||||
},
|
||||
input() {
|
||||
this.error = '';
|
||||
},
|
||||
validate() {
|
||||
const { username } = this;
|
||||
const { password } = this;
|
||||
if (username === '' || username == null) {
|
||||
this.error = '请输入用户名';
|
||||
return false;
|
||||
}
|
||||
if (!/^[a-zA-Z0-9]+([._\\-]*[a-zA-Z0-9])*@([a-zA-Z0-9]+[-a-zA-Z0-9]*[a-zA-Z0-9]+.){1,63}[a-zA-Z0-9]+$/.test(username)) {
|
||||
this.error = '请输入正确邮箱账号';
|
||||
return false;
|
||||
}
|
||||
if (password === '' || password == null) {
|
||||
this.error = '请输入密码';
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
},
|
||||
getStyle(type) {
|
||||
let style = '';
|
||||
if (this[type] && this.isIE(9)) {
|
||||
if (type === 'passwordFocus' && !this.password) {
|
||||
style += ' ie-palceholder ie-palceholder-password';
|
||||
} else if (type === 'userFocus' && !this.username) {
|
||||
style += ' ie-palceholder';
|
||||
}
|
||||
}
|
||||
return style;
|
||||
},
|
||||
initStyle() {
|
||||
if (this.isIE(9)) {
|
||||
!this.password && (this.passwordFocus = true);
|
||||
!this.username && (this.userFocus = true);
|
||||
}
|
||||
},
|
||||
focusHandler(type) {
|
||||
this.isIE(9) && (this[type] = false);
|
||||
},
|
||||
blurHandler(type) {
|
||||
this[type] = true;
|
||||
},
|
||||
isIE(ver) {
|
||||
const b = document.createElement('b');
|
||||
b.innerHTML = `<!--[if IE ${ver}]><i></i><![endif]-->`;
|
||||
return b.getElementsByTagName('i').length === 1;
|
||||
}
|
||||
}
|
||||
};
|
||||
</script>
|
48
packages/fes-cli/template/src/pages/i18n/index.vue
Normal file
@ -0,0 +1,48 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>
|
||||
{{$t('i18n.internationalization')}}<a target="_blank" href="https://kazupon.github.io/vue-i18n">
|
||||
vue-i18n
|
||||
</a>{{$t('i18n.achieve')}}
|
||||
<Wb-select v-model="locale" @on-change="change" class="select">
|
||||
<wb-option value="zh-cn">
|
||||
zh-cn
|
||||
</wb-option>
|
||||
<wb-option value="en">
|
||||
en
|
||||
</wb-option>
|
||||
</Wb-select>
|
||||
</p>
|
||||
|
||||
<h2>{{$t('i18n.ui')}}</h2>
|
||||
<Date-picker :value="value" />
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
FesData() {
|
||||
const local = this.FesApp.i18n.locale;
|
||||
return {
|
||||
locale: local,
|
||||
value: +new Date()
|
||||
};
|
||||
},
|
||||
FesReady() {
|
||||
// do something
|
||||
},
|
||||
methods: {
|
||||
change() {
|
||||
this.FesApp.setLocale(this.locale);
|
||||
}
|
||||
}
|
||||
};
|
||||
</script>
|
||||
<style>
|
||||
.select{
|
||||
width: 200px;
|
||||
}
|
||||
.article ul{
|
||||
padding: 0;
|
||||
}
|
||||
</style>
|
12
packages/fes-cli/template/src/pages/layout/a.vue
Normal file
@ -0,0 +1,12 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
A子页面
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
FesReady() {
|
||||
// do something
|
||||
}
|
||||
};
|
||||
</script>
|
12
packages/fes-cli/template/src/pages/layout/b.vue
Normal file
@ -0,0 +1,12 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
B子页面
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
FesReady() {
|
||||
// do something
|
||||
}
|
||||
};
|
||||
</script>
|
16
packages/fes-cli/template/src/pages/layout/layout.vue
Normal file
@ -0,0 +1,16 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>
|
||||
根路由下所有page共用的外层
|
||||
</p>
|
||||
<router-view />
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
FesReady() {
|
||||
// do something
|
||||
}
|
||||
};
|
||||
</script>
|
89
packages/fes-cli/template/src/pages/list/edit/index.vue
Normal file
@ -0,0 +1,89 @@
|
||||
<template>
|
||||
<Panel title="新增XXXX">
|
||||
<Icon slot="action" type="check" title="保存" size="20" />
|
||||
<Icon slot="action" type="rollback" title="返回到列表页面" size="20" />
|
||||
<Wb-form :label-width="150">
|
||||
<Row>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
</Row>
|
||||
<Row>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
</Row>
|
||||
<Row>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
</Row>
|
||||
<Row>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
</Row>
|
||||
<Row>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
</Row>
|
||||
<Row>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
<Cell span="12">
|
||||
<Form-item label="应用名称:">
|
||||
<wb-input placeholder="我是文本哦" />
|
||||
</Form-item>
|
||||
</Cell>
|
||||
</Row>
|
||||
</Wb-form>
|
||||
</Panel>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
FesData() {
|
||||
return {
|
||||
};
|
||||
},
|
||||
methods: {}
|
||||
};
|
||||
</script>
|
84
packages/fes-cli/template/src/pages/list/index.vue
Normal file
@ -0,0 +1,84 @@
|
||||
<template>
|
||||
<div>
|
||||
<fes-search-panel>
|
||||
<Wb-form :label-width="150" type="query">
|
||||
<Form-item label="姓名:">
|
||||
<wb-input v-model="query.name" placeholder="请输入" />
|
||||
</Form-item>
|
||||
<Form-item label="身份证:">
|
||||
<wb-input v-model="query.id" placeholder="请输入" />
|
||||
</Form-item>
|
||||
</Wb-form>
|
||||
<div slot="button">
|
||||
<Wb-button @click="search" type="primary" icon="search">
|
||||
查询
|
||||
</Wb-button>
|
||||
</div>
|
||||
</fes-search-panel>
|
||||
<fes-list-panel>
|
||||
<Wb-table :data="data">
|
||||
<Column prop="date" name="日期" />
|
||||
<Column prop="name" name="姓名" />
|
||||
<Column prop="age" name="年龄" />
|
||||
<Column prop="adr" name="地址" />
|
||||
<Column prop="status" name="等级" />
|
||||
</Wb-table>
|
||||
<Pagination :size="paginationOption.pageSize" :current="paginationOption.currentPage" :total="paginationOption.totalPage" @on-change="changePage" />
|
||||
</fes-list-panel>
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
FesDataCache: 'test',
|
||||
FesData() {
|
||||
return {
|
||||
query: {
|
||||
name: '',
|
||||
id: ''
|
||||
},
|
||||
data: [{
|
||||
name: '张晓刚',
|
||||
age: 24,
|
||||
date: new Date(2016, 9, 10),
|
||||
adr: '北京市海淀区西二旗',
|
||||
status: 1
|
||||
}, {
|
||||
name: '李晓红',
|
||||
age: 26,
|
||||
date: new Date(2016, 9, 11),
|
||||
adr: '北京市海淀区西二旗',
|
||||
status: 2
|
||||
}, {
|
||||
name: '隔壁老王',
|
||||
age: 22,
|
||||
date: new Date(2016, 9, 12),
|
||||
adr: '北京市海淀区西二旗',
|
||||
status: 3
|
||||
}, {
|
||||
name: '我爸是李刚',
|
||||
age: 19,
|
||||
date: new Date(2016, 9, 13),
|
||||
adr: '北京市海淀区西二旗',
|
||||
status: 4
|
||||
}],
|
||||
paginationOption: {
|
||||
pageSize: 10,
|
||||
currentPage: 1,
|
||||
totalPage: 2
|
||||
}
|
||||
};
|
||||
},
|
||||
FesReady() {
|
||||
console.log(this);
|
||||
},
|
||||
methods: {
|
||||
changePage({ current, size }) {
|
||||
this.paginationOption.currentPage = current;
|
||||
this.paginationOption.pageSize = size;
|
||||
},
|
||||
search() {
|
||||
this.FesApp.router.push('/test');
|
||||
}
|
||||
}
|
||||
};
|
||||
</script>
|
28
packages/fes-cli/template/src/pages/route.vue
Normal file
@ -0,0 +1,28 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>
|
||||
Fes的路由是根据src/pages目录自动生成的<br>
|
||||
</p>
|
||||
<h2>生成规则</h2>
|
||||
<pre>
|
||||
pages
|
||||
├── index.fes # 根路由页面 路径 http://localhost:5000/index.html#!/
|
||||
├── a.fes # 路径 /a
|
||||
├── b
|
||||
│ ├── index.fes # 路径 /b
|
||||
│ ├── @id.fes # 动态路由 /b/:id
|
||||
│ └── c.fes # 路径 /b/c(优先于/b/:id进行匹配)
|
||||
└── layout.fes # 根路由下所有page共用的外层
|
||||
</pre>
|
||||
</div>
|
||||
</template>
|
||||
<script type="text/ecmascript-6">
|
||||
export default {
|
||||
FesData() {
|
||||
return {
|
||||
value: +new Date()
|
||||
};
|
||||
}
|
||||
};
|
||||
</script>
|
18
packages/fes-cli/template/src/pages/static/index.vue
Normal file
@ -0,0 +1,18 @@
|
||||
<template>
|
||||
<div class="article">
|
||||
<h2>{{$t('overview')}}</h2>
|
||||
<p>存放静态资源文件,不需要经过打包处理的,跟vue体系无关的前端资源可以放在这里</p>
|
||||
|
||||
<a target="_blank" href="static/1.txt">
|
||||
点击下载 1.txt
|
||||
</a>
|
||||
</div>
|
||||
</template>
|
||||
<script>
|
||||
export default {
|
||||
FesData() {
|
||||
return {
|
||||
};
|
||||
}
|
||||
};
|
||||
</script>
|
1
packages/fes-cli/template/src/static/1.txt
Normal file
@ -0,0 +1 @@
|
||||
111
|
BIN
packages/fes-cli/template/src/static/bell.png
Normal file
After Width: | Height: | Size: 443 B |
BIN
packages/fes-cli/template/src/static/favicon.ico
Normal file
After Width: | Height: | Size: 5.3 KiB |
21
packages/fes-core/LICENSE
Normal file
@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2020-present harrywan
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
8
packages/fes-core/README.md
Normal file
@ -0,0 +1,8 @@
|
||||
# fes-core
|
||||
`fes-core`是框架核心,对Vue的API做了一些增强。建议先阅读学习[Vue2.0](https://cn.vuejs.org/v2/guide/)。
|
||||
|
||||
## 安装:
|
||||
npm install @webank/fes-core --save
|
||||
|
||||
## 文档
|
||||
详细使用请查看[文档](https://webankfintech.github.io/fes.js/)
|
33
packages/fes-core/package.json
Normal file
@ -0,0 +1,33 @@
|
||||
{
|
||||
"name": "@webank/fes-core",
|
||||
"version": "0.1.0",
|
||||
"description": "一个好用的前端管理台快速开发框架",
|
||||
"scripts": {
|
||||
"test": "echo \"Error: no test specified\" && exit 1"
|
||||
},
|
||||
"author": "harrywan,qlin",
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/WeBankFinTech/fes.js.git"
|
||||
},
|
||||
"license": "MIT",
|
||||
"keywords": [
|
||||
"管理端",
|
||||
"fes",
|
||||
"fast",
|
||||
"easy",
|
||||
"strong"
|
||||
],
|
||||
"dependencies": {
|
||||
"axios": "^0.16.2",
|
||||
"lodash": "^4.17.15",
|
||||
"vue": "^2.6.10",
|
||||
"vue-i18n": "^8.4.0",
|
||||
"vue-router": "^2.6.0",
|
||||
"vue-template-compiler": "^2.6.10"
|
||||
},
|
||||
"peerDependencies": {
|
||||
"@webank/fes-core": "^0.1.0",
|
||||
"@webank/fes-ui": "^0.1.0"
|
||||
}
|
||||
}
|
395
packages/fes-core/src/api/index.js
Normal file
@ -0,0 +1,395 @@
|
||||
/**
|
||||
* 操作Api
|
||||
*/
|
||||
import axios from 'axios';
|
||||
import util from '../util';
|
||||
import env from '../env';
|
||||
import storage from '../storage';
|
||||
|
||||
const trim = function (obj) {
|
||||
Object.keys(obj).forEach((p) => {
|
||||
if (util.isString(obj[p])) {
|
||||
obj[p] = obj[p].trim();
|
||||
} else if (util.isPlainObject(obj[p])) {
|
||||
trim(obj[p]);
|
||||
} else if (util.isArray(obj[p])) {
|
||||
trim(obj[p]);
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
const requsetLog = {
|
||||
data: storage.get('FES_AJAX_LOG') || [],
|
||||
importantApi: {},
|
||||
creatLog(url, data, status) {
|
||||
let _data;
|
||||
if (data) {
|
||||
_data = JSON.stringify(data);
|
||||
}
|
||||
if (_data && _data.length > 1000) {
|
||||
data = _data.slice(0, 1000); // 大约1K
|
||||
}
|
||||
const now = new Date().getTime();
|
||||
const obj = {
|
||||
url,
|
||||
data,
|
||||
timestamp: now,
|
||||
status: status || 'send'
|
||||
};
|
||||
if (this.data.length >= 500) {
|
||||
this.data.shift();
|
||||
}
|
||||
this.data.push(obj);
|
||||
try {
|
||||
storage.set('FES_AJAX_LOG', this.data);
|
||||
} catch (e) {
|
||||
storage.remove('FES_AJAX_LOG');
|
||||
this.data = [obj];
|
||||
storage.set('FES_AJAX_LOG', this.data);
|
||||
}
|
||||
return obj;
|
||||
},
|
||||
changeLogStatus(log, newStatus) {
|
||||
const logs = this.data.filter(obj => obj.timestamp === log.timestamp);
|
||||
if (logs.length > 0) {
|
||||
logs[0].status = newStatus;
|
||||
storage.set('FES_AJAX_LOG', this.data);
|
||||
}
|
||||
},
|
||||
getLogByURL(url, data) {
|
||||
return this.data.filter(obj => obj.url === url && JSON.stringify(data) === JSON.stringify(obj.data));
|
||||
}
|
||||
};
|
||||
|
||||
const instance = axios.create({
|
||||
method: 'post',
|
||||
baseURL: env.api,
|
||||
timeout: 10000,
|
||||
withCredentials: true
|
||||
});
|
||||
|
||||
const api = {
|
||||
instance,
|
||||
error: {},
|
||||
constructionOfResponse: {
|
||||
codePath: 'code',
|
||||
successCode: '0',
|
||||
messagePath: 'msg',
|
||||
resultPath: 'result'
|
||||
}
|
||||
};
|
||||
|
||||
const getData = function (data, resultFormat) {
|
||||
const _arr = ['codePath', 'messagePath', 'resultPath'];
|
||||
const arr = []; const
|
||||
rst = {};
|
||||
for (let i = 0; i < _arr.length; i++) {
|
||||
const pathArray = resultFormat[_arr[i]].split('.');
|
||||
const pathLength = pathArray.length;
|
||||
let result;
|
||||
if (pathLength === 1 && pathArray[0] === '*') {
|
||||
result = data;
|
||||
} else {
|
||||
result = data[pathArray[0]];
|
||||
}
|
||||
for (let j = 1; j < pathLength; j++) {
|
||||
result = result[pathArray[j]];
|
||||
if (!result) {
|
||||
if (j < pathLength - 1) {
|
||||
console.error(`【FEX】ConstructionOfResponse配置错误:${_arr[i]}拿到的值是undefined,请检查配置`);
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
arr.push(result);
|
||||
}
|
||||
rst.code = arr[0];
|
||||
rst.message = arr[1];
|
||||
rst.result = arr[2];
|
||||
return rst;
|
||||
};
|
||||
|
||||
const success = function (response) {
|
||||
// 响应结构
|
||||
const resultFormat = (response.config && response.config.resultFormat) || api.constructionOfResponse;
|
||||
// 哪些code不处理错误
|
||||
const ignoreCode = (response.config && response.config.ignoreCode) || [];
|
||||
if (util.isNull(resultFormat.codePath) || util.isNull(resultFormat.successCode)
|
||||
|| util.isNull(resultFormat.messagePath) || util.isNull(resultFormat.resultPath)) {
|
||||
console.error('【FEX】Api配置错误: 请调用setConstructionOfResponse来设置API的响应结构');
|
||||
return null;
|
||||
}
|
||||
|
||||
let data;
|
||||
if (util.isString(response.data)) {
|
||||
data = JSON.parse(response.data);
|
||||
} else if (util.isObject(response.data)) {
|
||||
data = response.data;
|
||||
} else {
|
||||
throw new Error(util.format('fesMessages.defaultError'));
|
||||
}
|
||||
|
||||
const { code, message, result } = getData(data, resultFormat);
|
||||
|
||||
if (code !== resultFormat.successCode) {
|
||||
let _message = '';
|
||||
if (api.error[code]) {
|
||||
api.error[code].forEach(fn => fn(response));
|
||||
} else if (!ignoreCode.includes(code) && ignoreCode !== '*') {
|
||||
_message = message || util.format('fesMessages.defaultError');
|
||||
}
|
||||
const error = new Error(_message);
|
||||
error.response = response;
|
||||
throw error;
|
||||
}
|
||||
return result || {};
|
||||
};
|
||||
|
||||
const fail = function (error) {
|
||||
let _message = '';
|
||||
const response = error.response;
|
||||
if (response && api.error[response.status]) {
|
||||
api.error[response.status].forEach(fn => fn(response));
|
||||
} else {
|
||||
_message = util.format('fesMessages.defaultError');
|
||||
try {
|
||||
if (response && response.data) {
|
||||
let data;
|
||||
if (util.isString(response.data)) {
|
||||
data = JSON.parse(response.data);
|
||||
} else if (util.isObject(response.data)) {
|
||||
data = response.data;
|
||||
}
|
||||
if (data) {
|
||||
const { message } = getData(data, (response.config && response.config.resultFormat) || api.constructionOfResponse);
|
||||
_message = message;
|
||||
}
|
||||
}
|
||||
} catch (e) {
|
||||
// 可以啥都不做
|
||||
}
|
||||
}
|
||||
error.message = _message;
|
||||
throw error;
|
||||
};
|
||||
|
||||
const param = function (url, data, option) {
|
||||
const method = instance.defaults.method || 'post';
|
||||
if (util.isNull(url)) {
|
||||
return console.error('请传入URL');
|
||||
} if (!util.isNull(url) && util.isNull(data) && util.isNull(option)) {
|
||||
option = {
|
||||
method
|
||||
};
|
||||
} else if (!util.isNull(url) && !util.isNull(data) && util.isNull(option)) {
|
||||
option = {
|
||||
method
|
||||
};
|
||||
if (util.isString(data)) {
|
||||
option.method = data;
|
||||
} else if (util.isObject(data)) {
|
||||
option.data = data;
|
||||
}
|
||||
} else if (!util.isNull(url) && !util.isNull(data) && !util.isNull(option)) {
|
||||
if (!util.isObject(data)) {
|
||||
data = {};
|
||||
}
|
||||
if (util.isString(option)) {
|
||||
option = {
|
||||
method: option
|
||||
};
|
||||
} else if (util.isObject(option)) {
|
||||
option.method = option.method || method;
|
||||
} else {
|
||||
option = {
|
||||
method
|
||||
};
|
||||
}
|
||||
if (option.method === 'get' || option.method === 'delete' || option.method === 'head' || option.method === 'options') {
|
||||
option.params = data;
|
||||
}
|
||||
if (option.method === 'post' || option.method === 'put' || option.method === 'patch') {
|
||||
option.data = data;
|
||||
}
|
||||
}
|
||||
// 过滤参数中的空格
|
||||
const _data = option.params || option.data;
|
||||
if (_data && util.isObject(_data) && option.trim !== false) {
|
||||
trim(_data);
|
||||
}
|
||||
|
||||
option.url = url;
|
||||
|
||||
// 如果传了button
|
||||
if (option.button) {
|
||||
option.button.currentDisabled = true;
|
||||
}
|
||||
|
||||
return instance.request(option);
|
||||
};
|
||||
|
||||
const action = function (url, data, option) {
|
||||
// 记录日志
|
||||
const log = requsetLog.creatLog(url, data);
|
||||
|
||||
return param(url, data, option)
|
||||
.then(success, fail)
|
||||
.then((response) => {
|
||||
requsetLog.changeLogStatus(log, 'success');
|
||||
if (option && option.button) {
|
||||
option.button.currentDisabled = false;
|
||||
}
|
||||
return response;
|
||||
})
|
||||
.catch((error) => {
|
||||
requsetLog.changeLogStatus(log, 'fail');
|
||||
if (option && option.button) {
|
||||
option.button.currentDisabled = false;
|
||||
}
|
||||
error.message && window.Toast.error(error.message);
|
||||
throw error;
|
||||
});
|
||||
};
|
||||
|
||||
api.fetch = function (url, data, option) {
|
||||
if (requsetLog.importantApi[url]) {
|
||||
const logs = requsetLog.getLogByURL(url, data);
|
||||
if (logs.length > 0) {
|
||||
const compareLog = logs[logs.length - 1];
|
||||
if (compareLog.status === 'compare') {
|
||||
requsetLog.creatLog(url, data, 'notAllowed');
|
||||
return {
|
||||
then: () => {}
|
||||
};
|
||||
}
|
||||
const importantApiOption = requsetLog.importantApi[url];
|
||||
const control = importantApiOption.control || 10000;
|
||||
const message = importantApiOption.message || util.format('fesMessages.importInterfaceTip', { s: control / 1000 });
|
||||
if (new Date().getTime() - compareLog.timestamp < control) {
|
||||
const oldStatus = compareLog.status;
|
||||
requsetLog.changeLogStatus(compareLog, 'compare');
|
||||
return new Promise(((resolve, reject) => {
|
||||
window.Message.confirm(util.format('fesMessages.tip'), message).then((index) => {
|
||||
if (compareLog.status === 'compare') {
|
||||
requsetLog.changeLogStatus(compareLog, oldStatus);
|
||||
}
|
||||
if (index === 0) {
|
||||
resolve(action(url, data, option));
|
||||
} else {
|
||||
reject(new Error('不允许相同操作间隔过小'));
|
||||
}
|
||||
});
|
||||
}));
|
||||
}
|
||||
return action(url, data, option);
|
||||
}
|
||||
return action(url, data, option);
|
||||
}
|
||||
return action(url, data, option);
|
||||
};
|
||||
|
||||
/**
|
||||
* 设置 request Header
|
||||
* @param headers Object
|
||||
*/
|
||||
api.setHeader = function (headers = {}) {
|
||||
Object.keys(headers).forEach((p) => {
|
||||
if (['delete', 'get', 'head', 'post', 'put', 'patch', 'common'].includes(p)) {
|
||||
instance.defaults.headers[p] = Object.assign({}, instance.defaults.headers[p], headers[p]);
|
||||
} else {
|
||||
instance.defaults.headers.common[p] = headers[p];
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* 配置ajax请求参数
|
||||
* @param option
|
||||
*/
|
||||
api.option = function (option = {}) {
|
||||
const {
|
||||
root,
|
||||
baseURL,
|
||||
timeout,
|
||||
headers,
|
||||
config,
|
||||
...others
|
||||
} = option;
|
||||
if (root || baseURL) {
|
||||
instance.defaults.baseURL = root || baseURL;
|
||||
}
|
||||
if (timeout && util.isNumber(timeout)) {
|
||||
instance.defaults.timeout = timeout;
|
||||
}
|
||||
if (headers) {
|
||||
api.setHeader(headers);
|
||||
}
|
||||
const otherPropertys = Object.assign({}, others, config);
|
||||
Object.keys(otherPropertys).forEach((p) => {
|
||||
instance.defaults[p] = otherPropertys[p];
|
||||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* 请求拦截器
|
||||
* @param before function 请求之前的拦截器
|
||||
*/
|
||||
api.setReqInterceptor = function (before, error) {
|
||||
if (Array.isArray(before)) {
|
||||
return instance.interceptors.request.use(...before);
|
||||
}
|
||||
return instance.interceptors.request.use(before, error);
|
||||
};
|
||||
api.ejectReqInterceptor = function (interceptor) {
|
||||
return instance.interceptors.request.eject(interceptor);
|
||||
};
|
||||
|
||||
/**
|
||||
* 响应拦截器
|
||||
* @param after function 响应之后的拦截器
|
||||
*/
|
||||
api.setResInterceptor = function (after, error) {
|
||||
if (Array.isArray(after)) {
|
||||
return instance.interceptors.response.use(...after);
|
||||
}
|
||||
return instance.interceptors.response.use(after, error);
|
||||
};
|
||||
api.ejectResInterceptor = function (interceptor) {
|
||||
return instance.interceptors.response.eject(interceptor);
|
||||
};
|
||||
|
||||
/**
|
||||
* 配置错误响应
|
||||
* @param option
|
||||
*/
|
||||
api.setError = function (option) {
|
||||
if (option && util.isObject(option)) {
|
||||
Object.keys(option).forEach((key) => {
|
||||
if (!util.isArray(api.error[key])) {
|
||||
api.error[key] = [];
|
||||
}
|
||||
api.error[key].push(option[key]);
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* 设置响应结构
|
||||
* @param constructionOfResponse
|
||||
*/
|
||||
api.setResponse = function (constructionOfResponse) {
|
||||
this.constructionOfResponse = constructionOfResponse;
|
||||
};
|
||||
|
||||
/**
|
||||
* 配置重要请求
|
||||
*/
|
||||
api.setImportant = function (option) {
|
||||
if (option && util.isObject(option)) {
|
||||
requsetLog.importantApi = option;
|
||||
} else {
|
||||
console.error('【FEX】ImportantApi配置错误: 参数必须是对象{"/get": { message:"xxx", control: 10000 } }');
|
||||
}
|
||||
};
|
||||
|
||||
export default api;
|
9
packages/fes-core/src/app.js
Normal file
@ -0,0 +1,9 @@
|
||||
import './polyfill';
|
||||
// eslint-disable-next-line
|
||||
import '@webank/fes-ui/dist/styles/fes-ui.css';
|
||||
import './views/styles/index.scss';
|
||||
// eslint-disable-next-line
|
||||
import init from 'projectRoot/src/app.js';
|
||||
import App from './instance/app';
|
||||
|
||||
App.init(init);
|
61
packages/fes-core/src/config/index.js
Normal file
@ -0,0 +1,61 @@
|
||||
// eslint-disable-next-line
|
||||
import fesConfig from 'projectRoot/fes.config.js';
|
||||
|
||||
// 设置默认
|
||||
if (!fesConfig.mode) {
|
||||
fesConfig.mode = 'vertical';
|
||||
}
|
||||
|
||||
if (!fesConfig.theme) {
|
||||
fesConfig.theme = 'blue';
|
||||
}
|
||||
|
||||
if (!fesConfig.env) {
|
||||
fesConfig.env = {};
|
||||
}
|
||||
|
||||
if (!fesConfig.roles) {
|
||||
fesConfig.roles = {};
|
||||
}
|
||||
|
||||
if (!fesConfig.menu) {
|
||||
fesConfig.menu = [];
|
||||
}
|
||||
|
||||
if (!fesConfig.i18n) {
|
||||
fesConfig.i18n = {
|
||||
locale: 'zh-cn',
|
||||
messages: {
|
||||
}
|
||||
};
|
||||
}
|
||||
if (!fesConfig.i18n.locale) {
|
||||
fesConfig.i18n.locale = 'zh-cn';
|
||||
}
|
||||
if (!fesConfig.i18n.messages) {
|
||||
fesConfig.i18n.messages = {};
|
||||
}
|
||||
if (!fesConfig.i18n.messages['zh-cn']) {
|
||||
fesConfig.i18n.messages['zh-cn'] = {};
|
||||
}
|
||||
if (!fesConfig.i18n.messages.en) {
|
||||
fesConfig.i18n.messages.en = {};
|
||||
}
|
||||
Object.assign(fesConfig.i18n.messages['zh-cn'], {
|
||||
fesMessages: {
|
||||
defaultError: '后台接口异常,请联系开发处理!',
|
||||
importInterfaceTip: '两个相同请求间隔小于 {s} 秒,是否继续?',
|
||||
tip: '提示',
|
||||
noPermission: '您没有访问当前路径的权限'
|
||||
}
|
||||
});
|
||||
Object.assign(fesConfig.i18n.messages.en, {
|
||||
fesMessages: {
|
||||
defaultError: 'Server-end API error, please contact the admin.',
|
||||
importInterfaceTip: 'Repetitive request in {s} seconds, continue anyway?',
|
||||
tip: 'Tips',
|
||||
noPermission: 'You don’t have the authority to access.'
|
||||
}
|
||||
});
|
||||
|
||||
export default fesConfig;
|
22
packages/fes-core/src/directive/index.js
Normal file
@ -0,0 +1,22 @@
|
||||
// TODO runtime 实例和具体功能解耦
|
||||
/*eslint-disable */
|
||||
import app from '../instance/app';
|
||||
|
||||
/**
|
||||
* 常用的指令
|
||||
*/
|
||||
export const permission = {
|
||||
bind(el, binding) {
|
||||
const dispaly = el.style.display;
|
||||
const setDispaly = () => {
|
||||
const urls = app.getAllowPage() || [];
|
||||
if (urls.indexOf(binding.value) === -1) {
|
||||
el.style.display = 'none';
|
||||
} else {
|
||||
el.style.display = dispaly;
|
||||
}
|
||||
};
|
||||
setDispaly();
|
||||
app.FesUtil.event.on('fes_allowPage_change', setDispaly);
|
||||
}
|
||||
};
|
3
packages/fes-core/src/env/index.js
vendored
Normal file
@ -0,0 +1,3 @@
|
||||
import fesConfig from '../config';
|
||||
|
||||
export default fesConfig.env[process.privateFesEnv.env] || {};
|
4
packages/fes-core/src/fesx/_fesx.js
Normal file
@ -0,0 +1,4 @@
|
||||
import FesxClass from './fesx';
|
||||
|
||||
const insideName = `inside_${window.location.pathname.replace(/\//g, '_')}`;
|
||||
export default new FesxClass(insideName);
|
56
packages/fes-core/src/fesx/fesx.js
Normal file
@ -0,0 +1,56 @@
|
||||
/**
|
||||
* 全局状态管理
|
||||
*/
|
||||
import Vue from 'vue';
|
||||
import storage from '../storage';
|
||||
import util from '../util';
|
||||
|
||||
class Fesx {
|
||||
constructor(name) {
|
||||
Object.defineProperty(this, 'name', {
|
||||
value: name,
|
||||
enumerable: false
|
||||
});
|
||||
Object.defineProperty(this, 'pre', {
|
||||
value: `FesFesx_${this.name}_`,
|
||||
enumerable: false
|
||||
});
|
||||
const keys = Object.keys(sessionStorage);
|
||||
const len = keys.length;
|
||||
for (let i = 0; i < len; i++) {
|
||||
const key = keys[i];
|
||||
if (key.indexOf(this.pre) === 0) {
|
||||
Vue.set(this, key.slice(this.pre.length), storage.get(key));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
get(prop) {
|
||||
if (!this[prop]) {
|
||||
this.set(prop, storage.get(this.pre + prop));
|
||||
}
|
||||
return this[prop];
|
||||
}
|
||||
|
||||
set(prop, value) {
|
||||
Vue.set(this, prop, value);
|
||||
if (!util.isFunction(value)) {
|
||||
storage.set(this.pre + prop, value);
|
||||
}
|
||||
return this;
|
||||
}
|
||||
|
||||
clear() {
|
||||
const keys = Object.keys(sessionStorage);
|
||||
const len = keys.length;
|
||||
for (let i = 0; i < len; i++) {
|
||||
const key = keys[i];
|
||||
if (key.indexOf(this.pre) === 0) {
|
||||
storage.remove(key);
|
||||
Vue.set(this, key.slice(this.pre.length), undefined);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export default Fesx;
|
9
packages/fes-core/src/fesx/index.js
Normal file
@ -0,0 +1,9 @@
|
||||
/**
|
||||
* 全局状态管理
|
||||
*/
|
||||
|
||||
import Fesx from './fesx';
|
||||
|
||||
const collection = new Fesx('outside');
|
||||
|
||||
export default collection;
|
154
packages/fes-core/src/filter/index.js
Normal file
@ -0,0 +1,154 @@
|
||||
/**
|
||||
* 常用的过滤器
|
||||
*/
|
||||
import util from '../util';
|
||||
|
||||
/**
|
||||
* 日期格式化
|
||||
* @param _date
|
||||
* @param format
|
||||
* @returns {*}
|
||||
*/
|
||||
export function date(timestap, format) {
|
||||
if (!timestap) return '';
|
||||
format = format || 'yyyy-MM-dd hh:mm:ss';
|
||||
timestap = Number(timestap);
|
||||
const time = new Date(timestap);
|
||||
const obj = {
|
||||
'y+': time.getFullYear(),
|
||||
'M+': time.getMonth() + 1,
|
||||
'd+': time.getDate(),
|
||||
'h+': time.getHours(),
|
||||
'm+': time.getMinutes(),
|
||||
's+': time.getSeconds()
|
||||
};
|
||||
|
||||
if (new RegExp('(y+)').test(format)) {
|
||||
format = format.replace(RegExp.$1, obj['y+']);
|
||||
}
|
||||
Object.keys(obj).forEach((j) => {
|
||||
if (new RegExp(`(${j})`).test(format)) {
|
||||
format = format.replace(RegExp.$1, (RegExp.$1.length === 1) ? (obj[j]) : ((`00${obj[j]}`).substr((`${obj[j]}`).length)));
|
||||
}
|
||||
});
|
||||
return format;
|
||||
}
|
||||
|
||||
/**
|
||||
* 资金格式化插件
|
||||
* @param value
|
||||
* @returns {string|*}
|
||||
*/
|
||||
export function money(value) {
|
||||
const m = [];
|
||||
value = Number(value).toFixed(2);
|
||||
// 获取小数部分
|
||||
const decimals = value.match(/\.[0-9]*/g);
|
||||
// 获取整数部分
|
||||
const integer = parseInt(value, 10).toString();
|
||||
const temp = integer.split('');
|
||||
const length = temp.length;
|
||||
|
||||
// 添加","分隔符
|
||||
function formart() {
|
||||
let count = 0;
|
||||
for (let n = length; n > 0; n--, count++) {
|
||||
if (count && count % 3 === 0) {
|
||||
m.unshift(',');
|
||||
count = 0;
|
||||
}
|
||||
m.unshift(temp.pop());
|
||||
}
|
||||
const result = m.join('');
|
||||
return decimals ? result.concat(decimals) : result;
|
||||
}
|
||||
|
||||
return length > 3 ? formart() : value;
|
||||
}
|
||||
|
||||
/**
|
||||
* 银行卡,四位加一空格
|
||||
* @param value
|
||||
* @returns {*}
|
||||
*/
|
||||
export function card(value) {
|
||||
value = `${value}`;
|
||||
const reg = /([0-9]{4})/g;
|
||||
if (value) {
|
||||
value = value.replace(reg, '$1 ');
|
||||
}
|
||||
return value;
|
||||
}
|
||||
|
||||
/**
|
||||
* 给字符串中间加***
|
||||
* @param value
|
||||
* @param frontLen
|
||||
* @param backLen
|
||||
* @returns {*}
|
||||
*/
|
||||
export function safety(value, frontLen, backLen) {
|
||||
if (value) {
|
||||
const len = value.length;
|
||||
let front = '';
|
||||
let back = '';
|
||||
if (frontLen && len > frontLen) {
|
||||
front = value.slice(0, frontLen);
|
||||
}
|
||||
if (backLen && len > (frontLen + backLen)) {
|
||||
back = value.slice(len - backLen);
|
||||
}
|
||||
return `${front}***${back}`;
|
||||
}
|
||||
return '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 把数据字典中的值转换成text
|
||||
* @param value
|
||||
* @param arr
|
||||
* @returns {string}
|
||||
*/
|
||||
export function map(value, arr) {
|
||||
let name = '';
|
||||
if (arr && util.isArray(arr)) {
|
||||
arr.forEach((item) => {
|
||||
if (item.value === value) {
|
||||
name = item.text;
|
||||
}
|
||||
});
|
||||
}
|
||||
return name;
|
||||
}
|
||||
|
||||
/**
|
||||
* 过滤掉数据中的值
|
||||
* @param value
|
||||
* @param arr
|
||||
* @returns {string}
|
||||
*/
|
||||
export function allow(value, arr) {
|
||||
const _arr = [];
|
||||
if (util.isArray(value)) {
|
||||
value.forEach((obj) => {
|
||||
if (util.isArray(arr)) {
|
||||
if (arr.indexOf(obj.value) !== -1) {
|
||||
_arr.push(obj);
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
return _arr;
|
||||
}
|
||||
|
||||
export function capitalize(text) {
|
||||
return text[0].toUpperCase() + text.slice(1);
|
||||
}
|
||||
|
||||
export function uppercase(text) {
|
||||
return text.toUpperCase();
|
||||
}
|
||||
|
||||
export function lowercase(text) {
|
||||
return text.toLowerCase();
|
||||
}
|
16
packages/fes-core/src/index.html
Normal file
@ -0,0 +1,16 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<title>Fes</title>
|
||||
<meta charset="utf-8">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
|
||||
<meta name="renderer" content="webkit|ie-stand">
|
||||
<meta name="applicable-device" content="pc">
|
||||
<meta http-equiv="pragma" content="no-cache">
|
||||
<meta http-equiv="cache-control" content="no-cache">
|
||||
<meta http-equiv="expires" content="0">
|
||||
</head>
|
||||
<body>
|
||||
<div id="app"></div>
|
||||
</body>
|
||||
</html>
|
328
packages/fes-core/src/instance/app.js
Normal file
@ -0,0 +1,328 @@
|
||||
import Vue from 'vue';
|
||||
import VueRouter from 'vue-router';
|
||||
import VueI18n from 'vue-i18n';
|
||||
// eslint-disable-next-line
|
||||
import UiWebank from '@webank/fes-ui';
|
||||
// eslint-disable-next-line
|
||||
import routerConfig from 'projectRoot/.cache/routeConfig.js';
|
||||
// eslint-disable-next-line
|
||||
import commonCompConfig from 'projectRoot/.cache/commonComp.js';
|
||||
import Page from './page';
|
||||
import fesComponents from '../views/components';
|
||||
import root from '../views/layout/root.vue';
|
||||
|
||||
import * as filters from '../filter';
|
||||
// eslint-disable-next-line
|
||||
import * as directives from '../directive';
|
||||
|
||||
|
||||
import util from '../util';
|
||||
import _fesx from '../fesx/_fesx';
|
||||
import storage from '../storage';
|
||||
import api from '../api';
|
||||
import map from '../map';
|
||||
import fesx from '../fesx';
|
||||
import fesConfig from '../config';
|
||||
import env from '../env';
|
||||
import permission from './permission';
|
||||
|
||||
|
||||
if (process.privateFesEnv.env !== 'prod') {
|
||||
Vue.config.debug = true;
|
||||
Vue.config.devtools = true;
|
||||
}
|
||||
|
||||
const rolesConfig = fesConfig.roles;
|
||||
|
||||
class App {
|
||||
constructor() {
|
||||
this.FesApp = this;
|
||||
this.FesApi = api;
|
||||
this.FesStorage = storage;
|
||||
this.FesMap = map;
|
||||
this.FesFesx = fesx;
|
||||
this.FesUtil = util;
|
||||
this.FesEnv = env;
|
||||
|
||||
// 允许的路由
|
||||
// 默认可以访问所有路由,第一次addAllowPage的时候需要删除'*'
|
||||
this._roleId = _fesx.get('FesRoleId');
|
||||
if (this._roleId) {
|
||||
permission.set(rolesConfig[this._roleId] || ['*']);
|
||||
} else {
|
||||
permission.set(_fesx.get('FesAllowPageList') || ['*']);
|
||||
}
|
||||
|
||||
this.router = null;
|
||||
this.beforeRouter = null;
|
||||
this.afterRouter = null;
|
||||
|
||||
this.i18n = null;
|
||||
}
|
||||
|
||||
init(func) {
|
||||
window.Vue = Vue;
|
||||
|
||||
// ======================安装插件====================
|
||||
Vue.use(VueRouter);
|
||||
Vue.use(UiWebank);
|
||||
Vue.use(VueI18n);
|
||||
Vue.use(fesComponents);
|
||||
Vue.use(Page, this);
|
||||
|
||||
// =====================注册全局过滤器================
|
||||
Object.keys(filters).forEach((p) => {
|
||||
Vue.filter(p, filters[p]);
|
||||
});
|
||||
|
||||
// =====================注册全局组件==================
|
||||
Object.keys(commonCompConfig).forEach((p) => {
|
||||
Vue.component(p, commonCompConfig[p]);
|
||||
});
|
||||
|
||||
// =====================注册全局指令==================
|
||||
Object.keys(directives).forEach((p) => {
|
||||
Vue.directive(p, directives[p]);
|
||||
});
|
||||
|
||||
// 设置系统名称
|
||||
if (fesConfig.fesName) {
|
||||
this.set('FesName', fesConfig.fesName);
|
||||
}
|
||||
|
||||
// 设置系统名称
|
||||
if (fesConfig.favicon) {
|
||||
this.setFavicon(fesConfig.favicon);
|
||||
}
|
||||
|
||||
if (util.isFunction(func)) {
|
||||
func.call(this);
|
||||
}
|
||||
|
||||
this.run();
|
||||
}
|
||||
|
||||
run() {
|
||||
this.creatRouter();
|
||||
this.creatI18n();
|
||||
// eslint-disable-next-line
|
||||
new Vue({
|
||||
el: '#app',
|
||||
extends: root,
|
||||
router: this.router,
|
||||
i18n: this.i18n
|
||||
});
|
||||
}
|
||||
|
||||
creatI18n() {
|
||||
this.i18n = new VueI18n(fesConfig.i18n);
|
||||
this.setLocale(fesConfig.i18n.locale);
|
||||
}
|
||||
|
||||
creatRouter() {
|
||||
this.router = new VueRouter({
|
||||
routes: routerConfig,
|
||||
scrollBehavior(to, from, savedPosition) {
|
||||
if (savedPosition) {
|
||||
return savedPosition;
|
||||
}
|
||||
return {
|
||||
x: 0,
|
||||
y: 0
|
||||
};
|
||||
}
|
||||
});
|
||||
this.handleRouter();
|
||||
}
|
||||
|
||||
handleRouter() {
|
||||
this.router.beforeEach(async (to, from, next) => {
|
||||
util.history.record(to.path);
|
||||
let path;
|
||||
if (to.matched.length === 1) {
|
||||
path = to.matched[0].path;
|
||||
} else {
|
||||
path = to.path;
|
||||
}
|
||||
// 只有允许的路由才能进
|
||||
const canRoute = await permission.match(path);
|
||||
if (canRoute) {
|
||||
if (this.beforeRouter && util.isFunction(this.beforeRouter)) {
|
||||
this.beforeRouter(to, from, next);
|
||||
} else {
|
||||
next();
|
||||
}
|
||||
} else {
|
||||
window.Toast.error(util.format('fesMessages.noPermission'));
|
||||
if (from.path) {
|
||||
next(false);
|
||||
} else {
|
||||
next('/');
|
||||
}
|
||||
}
|
||||
});
|
||||
this.router.afterEach((route) => {
|
||||
// 更新页面的title
|
||||
let title;
|
||||
fesConfig.menu.forEach((parent) => {
|
||||
if (parent.path === route.path) {
|
||||
title = parent.title;
|
||||
} else if (parent.subMenu && parent.subMenu.length > 0) {
|
||||
parent.subMenu.forEach((son) => {
|
||||
if (son.path === route.path) {
|
||||
title = son.title;
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
// 设置切换路由时页面的标题
|
||||
let fesName = this.get('FesName');
|
||||
if (fesName.slice(0, 6) === '$i18n.') {
|
||||
fesName = util.format(fesName.slice(6));
|
||||
}
|
||||
document.title = title ? `${fesName} | ${title}` : fesName;
|
||||
|
||||
if (this.afterRouter && util.isFunction(this.afterRouter)) {
|
||||
this.afterRouter(route);
|
||||
}
|
||||
});
|
||||
this.setDefaultPage();
|
||||
}
|
||||
|
||||
async getDefaultPage(update) {
|
||||
// 如果router已初始化,通过当前链接来找路由,返回当前链接对应的路由
|
||||
if (this.router && !update) {
|
||||
const currentPath = this.router.history.getCurrentLocation();
|
||||
const isMatchCurrentPath = await permission.match(currentPath);
|
||||
if (isMatchCurrentPath) {
|
||||
return currentPath;
|
||||
}
|
||||
}
|
||||
// 返回权限列表第一个 > 路由表第一个
|
||||
const allAllowedRoute = await permission.get();
|
||||
return allAllowedRoute.length > 0 ? allAllowedRoute[0] : routerConfig[0];
|
||||
}
|
||||
|
||||
async setDefaultPage() {
|
||||
const defaultPage = await this.getDefaultPage(true);
|
||||
this.router.addRoutes([{
|
||||
path: '/',
|
||||
redirect: () => defaultPage
|
||||
}]);
|
||||
}
|
||||
|
||||
async setRole(roleId, redirect = true, update = true) {
|
||||
if (_fesx.get('FesRoleId') !== roleId) {
|
||||
if (rolesConfig[roleId] instanceof Array) {
|
||||
permission.set(rolesConfig[roleId]);
|
||||
this.set('FesRoleId', roleId);
|
||||
if (this.router && redirect) {
|
||||
const defaultPage = await this.getDefaultPage(update);
|
||||
this.router.push(defaultPage);
|
||||
}
|
||||
util.event.trigger('fes_allowPage_change');
|
||||
} else {
|
||||
console.error(`rolesConfig配置错误,不存在角色${roleId}`);
|
||||
}
|
||||
}
|
||||
return this;
|
||||
}
|
||||
|
||||
async setAllowPage(pageList, redirect = true, update = true) {
|
||||
if (pageList instanceof Array) {
|
||||
permission.set(pageList);
|
||||
this.set('FesRoleId', ''); // 通过角色控制权限和通过路由控制权限互斥,只能使用一种
|
||||
this.set('FesAllowPageList', pageList);
|
||||
if (this.router && redirect) {
|
||||
const defaultPage = await this.getDefaultPage(update);
|
||||
this.router.push(defaultPage);
|
||||
}
|
||||
util.event.trigger('fes_allowPage_change');
|
||||
}
|
||||
return this;
|
||||
}
|
||||
|
||||
// 废弃 API
|
||||
getAllowPage() {
|
||||
// 异步的这里会造成 break;
|
||||
return permission.getSync();
|
||||
}
|
||||
|
||||
getAllowPageAsync() {
|
||||
return permission.get();
|
||||
}
|
||||
|
||||
get(prop) {
|
||||
return _fesx.get(prop);
|
||||
}
|
||||
|
||||
set(prop, value) {
|
||||
_fesx.set(prop, value);
|
||||
return this;
|
||||
}
|
||||
|
||||
// 添加过滤器
|
||||
addFilter(name, func) {
|
||||
Vue.filter(name, func);
|
||||
return this;
|
||||
}
|
||||
|
||||
// 添加指令
|
||||
addDirective(name, option) {
|
||||
Vue.directive(name, option);
|
||||
return this;
|
||||
}
|
||||
|
||||
// 添加组件
|
||||
addComponent(name, c) {
|
||||
Vue.component(name, c);
|
||||
return this;
|
||||
}
|
||||
|
||||
// 第三方插件
|
||||
addThrid(name, option) {
|
||||
Vue.use(name, option);
|
||||
return this;
|
||||
}
|
||||
|
||||
setBeforeRouter(beforeRouter) {
|
||||
this.beforeRouter = beforeRouter;
|
||||
return this;
|
||||
}
|
||||
|
||||
setAfterRouter(afterRouter) {
|
||||
this.afterRouter = afterRouter;
|
||||
return this;
|
||||
}
|
||||
|
||||
// 添加favicon
|
||||
setFavicon(url) {
|
||||
let favicon = document.querySelector('#favicon');
|
||||
if (!favicon) {
|
||||
favicon = document.createElement('link');
|
||||
favicon.id = 'favicon';
|
||||
favicon.rel = 'shortcut icon';
|
||||
favicon.type = 'image/png';
|
||||
favicon.href = url;
|
||||
document.head.appendChild(favicon);
|
||||
} else {
|
||||
favicon.href = url;
|
||||
}
|
||||
return this;
|
||||
}
|
||||
|
||||
setLocale(lang) {
|
||||
// 修改vue-i18n的语言
|
||||
this.i18n.locale = lang;
|
||||
// 修改组件库的语言
|
||||
UiWebank.i18n.setLocale(lang);
|
||||
return this;
|
||||
}
|
||||
}
|
||||
|
||||
util.merge(App.prototype, util.event);
|
||||
// 暂时去掉package.json引入,安全检测不通过
|
||||
// App.prototype.version = packageConfig.version;
|
||||
App.prototype.engine = 'Vue';
|
||||
|
||||
export default new App();
|
122
packages/fes-core/src/instance/page.js
Normal file
@ -0,0 +1,122 @@
|
||||
import util from '../util';
|
||||
import storage from '../storage';
|
||||
import api from '../api';
|
||||
import map from '../map';
|
||||
import fesx from '../fesx';
|
||||
import env from '../env';
|
||||
import fesConfig from '../config';
|
||||
|
||||
const fesDataCache = {};
|
||||
|
||||
const Page = {
|
||||
install(Vue, App) {
|
||||
Vue.mixin({
|
||||
data() {
|
||||
const data = {
|
||||
FesMap: map,
|
||||
FesFesx: fesx
|
||||
};
|
||||
|
||||
// 如果存在页面缓存
|
||||
const cacheName = this.$options.FesDataCache;
|
||||
if (cacheName && fesDataCache[cacheName] && util.history.current.type !== 'forward') {
|
||||
return fesDataCache[cacheName];
|
||||
}
|
||||
if (this.$options.FesSyncData) {
|
||||
Object.keys(this.$options.FesSyncData).forEach((p) => {
|
||||
data[p] = null;
|
||||
});
|
||||
}
|
||||
let fesData;
|
||||
if (util.isFunction(this.$options.FesData)) {
|
||||
this.FesFesx = fesx;
|
||||
this.FesMap = map;
|
||||
fesData = this.$options.FesData.call(this);
|
||||
} else {
|
||||
// 直接等于,是对象的引用,会导致下次进入页面,FesData的值没变
|
||||
fesData = this.$options.FesData;
|
||||
}
|
||||
if (fesData) {
|
||||
Object.keys(fesData).forEach((p) => {
|
||||
data[p] = fesData[p];
|
||||
});
|
||||
}
|
||||
return data;
|
||||
},
|
||||
created() {
|
||||
// route切换时,重新设置为初始值
|
||||
const comp = (this.$route && this.$route.matched) || [];
|
||||
if (comp.length > 0) {
|
||||
const matchPage = comp[comp.length - 1].components.default;
|
||||
if (this.$options.__file === matchPage.__file) {
|
||||
const defaultHeader = fesConfig.FesHeader === undefined ? false : fesConfig.FesHeader;
|
||||
const defaultLeft = fesConfig.FesLeft === undefined ? true : fesConfig.FesLeft;
|
||||
if (typeof matchPage.FesHeader === 'boolean') {
|
||||
this.$root.header = matchPage.FesHeader;
|
||||
} else {
|
||||
this.$root.header = defaultHeader;
|
||||
}
|
||||
if (typeof matchPage.FesLeft === 'boolean') {
|
||||
this.$root.left = matchPage.FesLeft;
|
||||
} else {
|
||||
this.$root.left = defaultLeft;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
const syncData = this.$options.FesSyncData;
|
||||
if (syncData) {
|
||||
const arr = [];
|
||||
Object.keys(syncData).forEach((p) => {
|
||||
if (util.isArray(syncData[p])) {
|
||||
arr.push([p, syncData[p][0], syncData[p][1]]);
|
||||
} else {
|
||||
console.error(`【FEX】异步参数【${p}】配置错误:值不是数组`, syncData[p]);
|
||||
}
|
||||
});
|
||||
const requests = [];
|
||||
for (let i = 0; i < arr.length; i++) {
|
||||
requests.push(api.fetch(arr[i][1], util.merge({}, this.$route.params, this.$route.query, arr[i][2])));
|
||||
}
|
||||
Promise.all(requests).then((values) => {
|
||||
values.forEach((value, index) => {
|
||||
this[arr[index][0]] = value;
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
if (this.$options.FesCreated && util.isFunction(this.$options.FesCreated)) {
|
||||
this.$options.FesCreated.call(this);
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
if (this.$options.FesReady && util.isFunction(this.$options.FesReady)) {
|
||||
this.$options.FesReady.call(this);
|
||||
}
|
||||
},
|
||||
beforeDestroy() {
|
||||
const cacheName = this.$options.FesDataCache;
|
||||
if (cacheName) {
|
||||
fesDataCache[cacheName] = this.$data;
|
||||
}
|
||||
if (this.$options.FesBeforeDestroy && util.isFunction(this.$options.FesBeforeDestroy)) {
|
||||
this.$options.FesBeforeDestroy.call(this);
|
||||
}
|
||||
},
|
||||
destroyed() {
|
||||
if (this.$options.FesDestroy && util.isFunction(this.$options.FesDestroy)) {
|
||||
this.$options.FesDestroy.call(this);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// 注入自己的对象
|
||||
Vue.prototype.FesApp = App;
|
||||
Vue.prototype.FesUtil = util;
|
||||
Vue.prototype.FesStorage = storage;
|
||||
Vue.prototype.FesApi = api;
|
||||
Vue.prototype.FesEnv = env;
|
||||
}
|
||||
};
|
||||
|
||||
export default Page;
|
34
packages/fes-core/src/instance/permission.js
Normal file
@ -0,0 +1,34 @@
|
||||
/**
|
||||
* 用户路由控制
|
||||
*/
|
||||
import util from '../util';
|
||||
|
||||
const permission = {
|
||||
allowRoutesSync: [], // 兼容老的 API 请勿使用
|
||||
allowRoutes: [],
|
||||
format(allowRoutes) {
|
||||
if (Array.isArray(allowRoutes)) {
|
||||
return allowRoutes.map(allow => Promise.resolve(allow));
|
||||
}
|
||||
return [Promise.resolve(allowRoutes)];
|
||||
},
|
||||
set(data) {
|
||||
this.allowRoutesSync = data;
|
||||
this.allowRoutes = this.format(data);
|
||||
},
|
||||
getSync() {
|
||||
return this.allowRoutesSync;
|
||||
},
|
||||
get() {
|
||||
return Promise.all(this.allowRoutes).then(data => data.reduce((merge, cur) => merge.concat(cur), []));
|
||||
},
|
||||
merge(data) {
|
||||
this.allowRoutes = this.allowRoutes.concat(this.format(data));
|
||||
},
|
||||
async match(path) {
|
||||
const mergedAllowRoutes = await this.get();
|
||||
return util.canRoute(path, mergedAllowRoutes);
|
||||
}
|
||||
};
|
||||
|
||||
export default permission;
|
42
packages/fes-core/src/map/index.js
Normal file
@ -0,0 +1,42 @@
|
||||
/**
|
||||
* 数据字典管理
|
||||
*/
|
||||
import util from '../util';
|
||||
import fesConfig from '../config';
|
||||
|
||||
const data = fesConfig.map;
|
||||
|
||||
const $Map = Object.create({
|
||||
getValueByName(name, text) {
|
||||
// TODO 不确定这里是否需要 ===
|
||||
// eslint-disable-next-line
|
||||
const arr = this[name].filter(item => item.text == text);
|
||||
return arr[0] ? arr[0].value : '';
|
||||
},
|
||||
getNameByValue(name, value) {
|
||||
// TODO 不确定这里是否需要 ===
|
||||
// eslint-disable-next-line
|
||||
const arr = this[name].filter(item => item.value == value);
|
||||
return arr[0] ? arr[0].text : '';
|
||||
}
|
||||
});
|
||||
|
||||
Object.keys(data).forEach((name) => {
|
||||
$Map[name] = [];
|
||||
if (util.isArray(data[name])) {
|
||||
data[name].forEach((item) => {
|
||||
if (item.length >= 2) {
|
||||
$Map[name].push({
|
||||
value: item[0],
|
||||
text: item[1]
|
||||
});
|
||||
} else {
|
||||
console.error(`【FEX】Map配置错误:Name${name}的值必输是数组,类似['1', '成功']`, item);
|
||||
}
|
||||
});
|
||||
} else {
|
||||
console.error('【FEX】Map配置错误:后面的值必须是数组', data[name]);
|
||||
}
|
||||
});
|
||||
|
||||
export default $Map;
|
48
packages/fes-core/src/polyfill/index.js
Normal file
@ -0,0 +1,48 @@
|
||||
/* eslint-disable */
|
||||
class File {
|
||||
toString() {
|
||||
console.log('compatible File');
|
||||
return 'function File() { [native code] }';
|
||||
}
|
||||
}
|
||||
|
||||
if (window.File === undefined) {
|
||||
window.File = File;
|
||||
}
|
||||
|
||||
// el remove
|
||||
(function (arr) {
|
||||
arr.forEach((item) => {
|
||||
if (item.hasOwnProperty('remove')) {
|
||||
return;
|
||||
}
|
||||
Object.defineProperty(item, 'remove', {
|
||||
configurable: true,
|
||||
enumerable: true,
|
||||
writable: true,
|
||||
value: function remove() {
|
||||
if (this.parentNode !== null) this.parentNode.removeChild(this);
|
||||
}
|
||||
});
|
||||
});
|
||||
}([Element.prototype, CharacterData.prototype, DocumentType.prototype]));
|
||||
|
||||
|
||||
// Function.bind
|
||||
if (!Function.prototype.bind) {
|
||||
Function.prototype.bind = function (oThis) {
|
||||
if (typeof this !== 'function') {
|
||||
throw new TypeError('Function.prototype.bind - what is trying to be bound is not callable');
|
||||
}
|
||||
const aArgs = Array.prototype.slice.call(arguments, 1);
|
||||
const fToBind = this;
|
||||
const fNOP = function () {};
|
||||
const fBound = function () {
|
||||
return fToBind.apply(this instanceof fNOP && oThis ? this : oThis,
|
||||
aArgs.concat(Array.prototype.slice.call(arguments)));
|
||||
};
|
||||
fNOP.prototype = this.prototype;
|
||||
fBound.prototype = new fNOP();
|
||||
return fBound;
|
||||
};
|
||||
}
|
183
packages/fes-core/src/storage/index.js
Normal file
@ -0,0 +1,183 @@
|
||||
/* \
|
||||
|*|
|
||||
|*| :: cookies.js ::
|
||||
|*|
|
||||
|*| A complete cookies reader/writer framework with full unicode support.
|
||||
|*|
|
||||
|*| https://developer.mozilla.org/en-US/docs/DOM/document.cookie
|
||||
|*|
|
||||
|*| This framework is released under the GNU Public License, version 3 or later.
|
||||
|*| http://www.gnu.org/licenses/gpl-3.0-standalone.html
|
||||
|*|
|
||||
|*| Syntaxes:
|
||||
|*|
|
||||
|*| * docCookies.setItem(name, value[, end[, path[, domain[, secure]]]])
|
||||
|*| * docCookies.getItem(name)
|
||||
|*| * docCookies.removeItem(name[, path], domain)
|
||||
|*| * docCookies.hasItem(name)
|
||||
|*| * docCookies.keys()
|
||||
|*|
|
||||
\ */
|
||||
|
||||
const docCookies = {
|
||||
getItem(sKey) {
|
||||
return decodeURIComponent(document.cookie.replace(new RegExp(`(?:(?:^|.*;)\\s*${encodeURIComponent(sKey).replace(/[-.+*]/g, '\\$&')}\\s*\\=\\s*([^;]*).*$)|^.*$`), '$1')) || null;
|
||||
},
|
||||
setItem(sKey, sValue, vEnd, sPath, sDomain, bSecure) {
|
||||
if (!sKey || /^(?:expires|max-age|path|domain|secure)$/i.test(sKey)) {
|
||||
return false;
|
||||
}
|
||||
let sExpires = '';
|
||||
if (vEnd) {
|
||||
switch (vEnd.constructor) {
|
||||
case Number:
|
||||
sExpires = vEnd === Infinity ? '; expires=Fri, 31 Dec 9999 23:59:59 GMT' : `; max-age=${vEnd}`;
|
||||
break;
|
||||
case String:
|
||||
sExpires = `; expires=${vEnd}`;
|
||||
break;
|
||||
case Date:
|
||||
sExpires = `; expires=${vEnd.toUTCString()}`;
|
||||
break;
|
||||
default: break;
|
||||
}
|
||||
}
|
||||
document.cookie = `${encodeURIComponent(sKey)}=${encodeURIComponent(sValue)}${sExpires}${sDomain ? `; domain=${sDomain}` : ''}${sPath ? `; path=${sPath}` : ''}${bSecure ? '; secure' : ''}`;
|
||||
return true;
|
||||
},
|
||||
removeItem(sKey, sPath, sDomain) {
|
||||
if (!sKey || !this.hasItem(sKey)) {
|
||||
return false;
|
||||
}
|
||||
document.cookie = `${encodeURIComponent(sKey)}=; expires=Thu, 01 Jan 1970 00:00:00 GMT${sDomain ? `; domain=${sDomain}` : ''}${sPath ? `; path=${sPath}` : ''}`;
|
||||
return true;
|
||||
},
|
||||
hasItem(sKey) {
|
||||
return (new RegExp(`(?:^|;\\s*)${encodeURIComponent(sKey).replace(/[-.+*]/g, '\\$&')}\\s*\\=`)).test(document.cookie);
|
||||
},
|
||||
// eslint-disable-next-line
|
||||
keys: /* optional method: you can safely remove it! */ function () {
|
||||
const aKeys = document.cookie.replace(/((?:^|\s*;)[^=]+)(?=;|$)|^\s*|\s*(?:=[^;]*)?(?:\1|$)/g, '').split(/\s*(?:=[^;]*)?;\s*/);
|
||||
for (let nIdx = 0; nIdx < aKeys.length; nIdx++) {
|
||||
aKeys[nIdx] = decodeURIComponent(aKeys[nIdx]);
|
||||
}
|
||||
return aKeys;
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
const isProd = process.env.NODE_ENV === 'production';
|
||||
|
||||
export const storageManager = {
|
||||
set(key, value, storage) {
|
||||
try {
|
||||
window[storage].setItem(key, JSON.stringify(value));
|
||||
} catch (e) {
|
||||
!isProd && console.error(e);
|
||||
}
|
||||
},
|
||||
get(key, storage) {
|
||||
try {
|
||||
if (window[storage].getItem(key)) {
|
||||
return JSON.parse(window[storage].getItem(key));
|
||||
}
|
||||
return window[storage].getItem(key);
|
||||
} catch (e) {
|
||||
!isProd && console.error(e, key);
|
||||
return null;
|
||||
}
|
||||
},
|
||||
clear(storage) {
|
||||
window[storage].clear();
|
||||
},
|
||||
remove(key, storage) {
|
||||
window[storage].removeItem(key);
|
||||
}
|
||||
};
|
||||
|
||||
export const cookieManager = {
|
||||
set(key, value, expired) {
|
||||
if (expired) docCookies.setItem(key, value, expired);
|
||||
else docCookies.setItem(key, value);
|
||||
},
|
||||
get(key) {
|
||||
return docCookies.getItem(key);
|
||||
},
|
||||
clear() {
|
||||
docCookies.keys().forEach((key) => {
|
||||
docCookies.removeItem(key);
|
||||
});
|
||||
},
|
||||
remove(key) {
|
||||
docCookies.removeItem(key);
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
/**
|
||||
* 操作cookie、sessionStorage、localStorage、缓存
|
||||
*/
|
||||
|
||||
const
|
||||
SESSION = 'session';
|
||||
const LOCAL = 'local';
|
||||
const COOKIE = 'cookie';
|
||||
|
||||
export default {
|
||||
set(key, value, category = SESSION, expired) {
|
||||
const { storage, isWebStorage = true } = this._map(category);
|
||||
|
||||
if (isWebStorage) {
|
||||
storageManager.set(key, value, storage);
|
||||
} else {
|
||||
cookieManager.set(key, value, expired);
|
||||
}
|
||||
},
|
||||
get(key, category = SESSION) {
|
||||
const { storage, isWebStorage = true } = this._map(category);
|
||||
|
||||
if (isWebStorage) {
|
||||
return storageManager.get(key, storage);
|
||||
}
|
||||
return cookieManager.get(key);
|
||||
},
|
||||
clear(category = SESSION) {
|
||||
const { storage, isWebStorage = true } = this._map(category);
|
||||
|
||||
if (isWebStorage) {
|
||||
storageManager.clear(storage);
|
||||
} else {
|
||||
cookieManager.clear();
|
||||
}
|
||||
},
|
||||
remove(key, category = SESSION) {
|
||||
const { storage, isWebStorage = true } = this._map(category);
|
||||
|
||||
if (isWebStorage) {
|
||||
storageManager.remove(key, storage);
|
||||
} else {
|
||||
cookieManager.remove(key);
|
||||
}
|
||||
},
|
||||
_map(category) {
|
||||
let isWebStorage = true; let
|
||||
storage;
|
||||
|
||||
switch (true) {
|
||||
case category === SESSION:
|
||||
storage = 'sessionStorage';
|
||||
break;
|
||||
case category === LOCAL:
|
||||
storage = 'localStorage';
|
||||
break;
|
||||
case category === COOKIE:
|
||||
storage = 'cookie';
|
||||
isWebStorage = false;
|
||||
break;
|
||||
default:
|
||||
storage = 'sessionStorage';
|
||||
}
|
||||
|
||||
return { isWebStorage, storage };
|
||||
}
|
||||
};
|
106
packages/fes-core/src/util/dom.js
Normal file
@ -0,0 +1,106 @@
|
||||
const inBrowser = typeof window !== 'undefined'
|
||||
&& Object.prototype.toString.call(window) !== '[object Object]';
|
||||
export const UA = inBrowser && window.navigator.userAgent.toLowerCase();
|
||||
export const isIE = UA && UA.indexOf('trident') > 0;
|
||||
export const isIE9 = UA && UA.indexOf('msie 9.0') > 0;
|
||||
|
||||
/**
|
||||
* For IE9 compat: when both class and :class are present
|
||||
* getAttribute('class') returns wrong value...
|
||||
*
|
||||
* @param {Element} el
|
||||
* @return {String}
|
||||
*/
|
||||
export function getClass(el) {
|
||||
let classname = el.className;
|
||||
if (typeof classname === 'object') {
|
||||
classname = classname.baseVal || '';
|
||||
}
|
||||
return classname;
|
||||
}
|
||||
|
||||
/**
|
||||
* 判断dom节点是否有某样式
|
||||
*
|
||||
* @param {Element} el
|
||||
* @return {String}
|
||||
* @returns {boolean}
|
||||
*/
|
||||
export function hasClass(el, name) {
|
||||
if (!el) return null;
|
||||
const className = getClass(el);
|
||||
const classes = className.split(' ');
|
||||
return classes.indexOf(name) !== -1;
|
||||
}
|
||||
|
||||
/**
|
||||
* In IE9, setAttribute('class') will result in empty class
|
||||
* if the element also has the :class attribute; However in
|
||||
* PhantomJS, setting `className` does not work on SVG elements...
|
||||
* So we have to do a conditional check here.
|
||||
*
|
||||
* @param {Element} el
|
||||
* @param {String} cls
|
||||
*/
|
||||
export function setClass(el, cls) {
|
||||
/* istanbul ignore if */
|
||||
if (isIE9 && !/svg$/.test(el.namespaceURI)) {
|
||||
el.className = cls;
|
||||
} else {
|
||||
el.setAttribute('class', cls);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Add class with compatibility for IE & SVG
|
||||
*
|
||||
* @param {Element} el
|
||||
* @param {String} cls
|
||||
*/
|
||||
|
||||
export function addClass(el, cls) {
|
||||
if (el.classList) {
|
||||
el.classList.add(cls);
|
||||
} else {
|
||||
const cur = ` ${getClass(el)} `;
|
||||
if (cur.indexOf(` ${cls} `) < 0) {
|
||||
setClass(el, (cur + cls).trim());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove class with compatibility for IE & SVG
|
||||
*
|
||||
* @param {Element} el
|
||||
* @param {String} cls
|
||||
*/
|
||||
|
||||
export function removeClass(el, cls) {
|
||||
if (el.classList) {
|
||||
el.classList.remove(cls);
|
||||
} else {
|
||||
let cur = ` ${getClass(el)} `;
|
||||
const tar = ` ${cls} `;
|
||||
while (cur.indexOf(tar) >= 0) {
|
||||
cur = cur.replace(tar, ' ');
|
||||
}
|
||||
setClass(el, cur.trim());
|
||||
}
|
||||
if (!el.className) {
|
||||
el.removeAttribute('class');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 从jquery扣过来的,递归去算
|
||||
*
|
||||
* @param {Element} a
|
||||
* @param {Element} b
|
||||
* @returns {boolean}
|
||||
*/
|
||||
export function contains(a, b) {
|
||||
const adown = a.nodeType === 9 ? a.documentElement : a;
|
||||
const bup = b && b.parentNode;
|
||||
return a === bup || !!(bup && bup.nodeType === 1 && adown.contains(bup));
|
||||
}
|
51
packages/fes-core/src/util/event.js
Normal file
@ -0,0 +1,51 @@
|
||||
|
||||
const eventProxy = {
|
||||
onObj: {},
|
||||
oneObj: {},
|
||||
on(key, fn) {
|
||||
if (fn && typeof (fn) === 'function') {
|
||||
if (this.onObj[key] === undefined) {
|
||||
this.onObj[key] = [];
|
||||
}
|
||||
this.onObj[key].push(fn);
|
||||
} else {
|
||||
throw new Error('请传入正确的回调函数');
|
||||
}
|
||||
},
|
||||
one(key, fn) {
|
||||
if (this.oneObj[key] === undefined) {
|
||||
this.oneObj[key] = [];
|
||||
}
|
||||
|
||||
this.oneObj[key].push(fn);
|
||||
},
|
||||
off(key) {
|
||||
this.onObj[key] = [];
|
||||
this.oneObj[key] = [];
|
||||
},
|
||||
trigger(...args) {
|
||||
if (args.length === 0) {
|
||||
return false;
|
||||
}
|
||||
const key = args[0];
|
||||
args = [].concat(Array.prototype.slice.call(args, 1));
|
||||
|
||||
if (this.onObj[key] !== undefined
|
||||
&& this.onObj[key].length > 0) {
|
||||
Object.keys(this.onObj[key]).forEach((i) => {
|
||||
this.onObj[key][i].apply(null, args);
|
||||
});
|
||||
}
|
||||
if (this.oneObj[key] !== undefined
|
||||
&& this.oneObj[key].length > 0) {
|
||||
Object.keys(this.oneObj[key]).forEach((i) => {
|
||||
this.oneObj[key][i].apply(null, args);
|
||||
this.oneObj[key][i] = undefined;
|
||||
});
|
||||
this.oneObj[key] = [];
|
||||
}
|
||||
return null;
|
||||
}
|
||||
};
|
||||
|
||||
export default eventProxy;
|
46
packages/fes-core/src/util/format.js
Normal file
@ -0,0 +1,46 @@
|
||||
import fesConfig from '../config';
|
||||
|
||||
const hasOwnProperty = Object.prototype.hasOwnProperty;
|
||||
const hasOwn = function (obj, key) {
|
||||
return hasOwnProperty.call(obj, key);
|
||||
};
|
||||
const RE_NARGS = /(%|)\{([0-9a-zA-Z_]+)\}/g;
|
||||
|
||||
const template = function (string, ...args) {
|
||||
if (args.length === 1 && typeof args[0] === 'object') {
|
||||
args = args[0];
|
||||
}
|
||||
|
||||
if (!args || !args.hasOwnProperty) {
|
||||
args = {};
|
||||
}
|
||||
|
||||
return string.replace(RE_NARGS, (match, prefix, i, index) => {
|
||||
if (string[index - 1] === '{' && string[index + match.length] === '}') {
|
||||
return i;
|
||||
}
|
||||
const result = hasOwn(args, i) ? args[i] : null;
|
||||
if (result === null || result === undefined) {
|
||||
return '';
|
||||
}
|
||||
|
||||
return result;
|
||||
});
|
||||
};
|
||||
|
||||
export default function (path, options) {
|
||||
const array = path.split('.');
|
||||
let current = fesConfig.i18n.messages[fesConfig.i18n.locale];
|
||||
if (!current) {
|
||||
current = fesConfig.i18n.messages['zh-cn'];
|
||||
}
|
||||
let value;
|
||||
for (let i = 0, j = array.length; i < j; i++) {
|
||||
const property = array[i];
|
||||
value = current[property];
|
||||
if (i === j - 1) return template(value, options);
|
||||
if (!value) return '';
|
||||
current = value;
|
||||
}
|
||||
return '';
|
||||
}
|
38
packages/fes-core/src/util/history.js
Normal file
@ -0,0 +1,38 @@
|
||||
import storage from '../storage';
|
||||
|
||||
const history = {
|
||||
data: storage.get('Fes_History') || [],
|
||||
current: null
|
||||
};
|
||||
|
||||
history.record = function (href) {
|
||||
const length = history.data.length;
|
||||
const obj = {
|
||||
href,
|
||||
type: ''
|
||||
};
|
||||
if (length === 0) {
|
||||
obj.type = 'forward';
|
||||
} else if (length > 0 && length <= 1) {
|
||||
if (history.data[length - 1].href === href) {
|
||||
obj.type = 'refresh';
|
||||
} else {
|
||||
obj.type = 'forward';
|
||||
}
|
||||
} else if (length > 1) {
|
||||
const first = history.data[length - 1];
|
||||
const second = history.data[length - 2];
|
||||
if (first.href === href) {
|
||||
obj.type = 'refresh';
|
||||
} else if (second.href === href) {
|
||||
obj.type = 'back';
|
||||
} else {
|
||||
obj.type = 'forward';
|
||||
}
|
||||
}
|
||||
history.data.push(obj);
|
||||
history.current = obj;
|
||||
storage.set('Fes_History', history.data);
|
||||
};
|
||||
|
||||
export default history;
|
88
packages/fes-core/src/util/index.js
Normal file
@ -0,0 +1,88 @@
|
||||
import _ from 'lodash';
|
||||
import * as domUtil from './dom';
|
||||
import * as objectUtil from './object';
|
||||
import * as typeUtil from './type';
|
||||
import format from './format';
|
||||
import event from './event';
|
||||
import history from './history';
|
||||
|
||||
const util = {
|
||||
// 验证一个path是否可以访问, 空的allowPage可以访问任何路由
|
||||
canRoute(path, allowPage) {
|
||||
path = path.split('?')[0];
|
||||
if (Array.isArray(allowPage) && allowPage.length > 0) {
|
||||
if (path === '' && allowPage.includes('/')) return true;
|
||||
if (path) {
|
||||
for (let i = 0; i < allowPage.length; i++) {
|
||||
if (path === allowPage[i]) {
|
||||
return true;
|
||||
}
|
||||
// 支持*匹配
|
||||
const reg = new RegExp(`^${allowPage[i].replace('*', '.+')}$`);
|
||||
if (reg.test(path)) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
},
|
||||
getUrlParam(name) {
|
||||
const reg = new RegExp(`(^|&)${name}=([^&]*)(&|$)`);
|
||||
let r = window.location.search.substr(1).match(reg);
|
||||
const hashQuery = window.location.hash.split('?')[1];
|
||||
if (r != null) {
|
||||
return decodeURIComponent(r[2]);
|
||||
} if (hashQuery) {
|
||||
r = hashQuery.match(reg);
|
||||
return r && decodeURIComponent(r[2]);
|
||||
}
|
||||
return null;
|
||||
},
|
||||
removeParam(name, content) {
|
||||
if (typeof name !== 'string') return false;
|
||||
const prefix = encodeURIComponent(`${name}=`);
|
||||
const pars = content.split(/[&;]/g);
|
||||
let i = 0; const
|
||||
len = pars.length;
|
||||
let value = '';
|
||||
for (; i < len; i++) {
|
||||
if (encodeURIComponent(pars[i]).lastIndexOf(prefix, 0) !== -1) {
|
||||
pars.splice(i, 1);
|
||||
}
|
||||
}
|
||||
value = (pars.length > 0 ? `?${pars.join('&')}` : '');
|
||||
return value;
|
||||
},
|
||||
proxyFn(proxy, prop, apiArr) {
|
||||
proxy[prop] = {};
|
||||
const cache = {};
|
||||
if (window.Proxy) {
|
||||
proxy[prop] = new Proxy(proxy[prop], {
|
||||
get(target, name) {
|
||||
cache[name] = cache[name] ? cache[name] : [];
|
||||
if (!target[name]) {
|
||||
target[name] = function (...args) {
|
||||
cache[name].push(args);
|
||||
};
|
||||
}
|
||||
return target[name];
|
||||
}
|
||||
});
|
||||
} else {
|
||||
apiArr.forEach((api) => {
|
||||
if (!proxy[prop][api]) {
|
||||
proxy[prop][api] = function (...args) {
|
||||
cache[api] = cache[api] ? cache[api] : [];
|
||||
cache[api].push(args);
|
||||
};
|
||||
}
|
||||
});
|
||||
}
|
||||
return cache;
|
||||
},
|
||||
_
|
||||
};
|
||||
objectUtil.merge(util, domUtil, objectUtil, typeUtil, { format }, { event }, { history });
|
||||
export default util;
|
27
packages/fes-core/src/util/object.js
Normal file
@ -0,0 +1,27 @@
|
||||
export function merge(...args) {
|
||||
const base = args[0];
|
||||
if (!base) return null;
|
||||
[].forEach.call(args, (item, index) => {
|
||||
if (index > 0) {
|
||||
Object.keys(item).forEach((attrname) => {
|
||||
base[attrname] = item[attrname];
|
||||
});
|
||||
}
|
||||
});
|
||||
return base;
|
||||
}
|
||||
|
||||
export function extend(...args) {
|
||||
const base = args[0];
|
||||
if (!base) return null;
|
||||
[].forEach.call(args, (item, index) => {
|
||||
if (index > 0) {
|
||||
Object.keys(item).forEach((attrname) => {
|
||||
if (base[attrname] !== undefined) {
|
||||
base[attrname] = item[attrname];
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
return base;
|
||||
}
|