diff --git a/.eslintrc.js b/.eslintrc.js
index 054b0d92..7813edc9 100644
--- a/.eslintrc.js
+++ b/.eslintrc.js
@@ -17,5 +17,10 @@ module.exports = {
'no-restricted-syntax': 'off',
'no-undefined': 'off',
'vue/valid-template-root': 'off'
+ },
+ parserOptions: {
+ babelOptions: {
+ presets: ['@babel/preset-env']
+ }
}
};
diff --git a/.fatherrc.js b/.fatherrc.js
index 680c2f30..f0d53670 100644
--- a/.fatherrc.js
+++ b/.fatherrc.js
@@ -16,7 +16,9 @@ const headPkgs = [
"fes-plugin-icon",
"fes-plugin-locale",
"fes-plugin-enums",
- "create-fes-app"
+ "fes-plugin-jest",
+ "fes-plugin-vuex",
+ "create-fes-app",
];
const tailPkgs = [];
// const otherPkgs = readdirSync(join(__dirname, 'packages')).filter(
diff --git a/lerna.json b/lerna.json
index c80a8f92..71814b8a 100644
--- a/lerna.json
+++ b/lerna.json
@@ -1,5 +1,5 @@
{
- "version": "2.0.0-alpha.5",
+ "version": "2.0.0-alpha.8",
"changelog": {
"repo": "WeBankFinTech/fes.js",
"cacheDir": ".changelog",
diff --git a/package.json b/package.json
index 2008d6eb..3758d103 100644
--- a/package.json
+++ b/package.json
@@ -34,7 +34,7 @@
"@vuepress/plugin-pwa": "^2.0.0-alpha.18",
"@vuepress/plugin-pwa-popup": "^2.0.0-alpha.18",
"@vuepress/theme-vue": "^2.0.0-alpha.18",
- "@webank/eslint-config-webank": "^0.2.10",
+ "@webank/eslint-config-webank": "0.2.10",
"commitizen": "^4.2.1",
"cz-conventional-changelog": "^3.3.0",
"esbuild-loader": "^2.7.0",
diff --git a/packages/create-fes-app/README.md b/packages/create-fes-app/README.md
new file mode 100644
index 00000000..14704fc6
--- /dev/null
+++ b/packages/create-fes-app/README.md
@@ -0,0 +1,3 @@
+# fes
+
+一个好用的前端应用解决方案
\ No newline at end of file
diff --git a/packages/create-fes-app/package.json b/packages/create-fes-app/package.json
index 164e4cfb..d68b67cc 100644
--- a/packages/create-fes-app/package.json
+++ b/packages/create-fes-app/package.json
@@ -1,6 +1,6 @@
{
"name": "@webank/create-fes-app",
- "version": "2.0.0-alpha.3",
+ "version": "2.0.0-alpha.8",
"description": "create a app base on fes.js",
"main": "lib/index.js",
"files": [
diff --git a/packages/create-fes-app/templates/app/h5/package.json b/packages/create-fes-app/templates/app/h5/package.json
index 50c620e9..962cff3e 100644
--- a/packages/create-fes-app/templates/app/h5/package.json
+++ b/packages/create-fes-app/templates/app/h5/package.json
@@ -1,6 +1,6 @@
{
"name": "@webank/fes-template-h5",
- "version": "2.0.0-alpha.1",
+ "version": "2.0.0-alpha",
"description": "fes 移动端项目模版",
"scripts": {
"build": "fes build",
@@ -13,26 +13,41 @@
"easy",
"strong"
],
+ "files": [
+ ".eslintrc.js",
+ ".gitignore",
+ ".fes.js",
+ ".fes.prod.js",
+ "mock.js",
+ "package.json",
+ "README.md",
+ "tsconfig.json",
+ "/src",
+ "/config"
+ ],
"repository": {
"type": "git",
"url": "git+https://github.com/WeBankFinTech/fes.js.git",
"directory": "packages/fes-template-h5"
},
- "author": "harrywan qlin",
+ "author": "qlin",
"license": "MIT",
"bugs": {
"url": "https://github.com/WeBankFinTech/fes.js/issues"
},
"homepage": "https://github.com/WeBankFinTech/fes.js#readme",
+ "publishConfig": {
+ "access": "public"
+ },
"devDependencies": {
- "@vue/compiler-sfc": "^3.0.0",
- "@webank/eslint-config-webank": "0.2.7",
+ "@webank/eslint-config-webank": "0.2.10",
"postcss-px-to-viewport": "1.1.1"
},
"dependencies": {
- "@webank/fes": "^2.0.0-alpha.0",
- "@webank/fes-plugin-icon": "^2.0.0-alpha.0",
- "@webank/fes-plugin-request": "^2.0.0-alpha.0",
- "vue": "^3.0.4"
- }
+ "@webank/fes": "^2.0.0-alpha.6",
+ "@webank/fes-plugin-icon": "^2.0.0-alpha.6",
+ "@webank/fes-plugin-request": "^2.0.0-alpha.6",
+ "vue": "3.0.5"
+ },
+ "private": true
}
diff --git a/packages/create-fes-app/templates/app/h5/public/index.html b/packages/create-fes-app/templates/app/h5/public/index.html
index 7959977e..750a5c09 100644
--- a/packages/create-fes-app/templates/app/h5/public/index.html
+++ b/packages/create-fes-app/templates/app/h5/public/index.html
@@ -9,6 +9,7 @@
<%= htmlWebpackPlugin.options.title %>
+
diff --git a/packages/create-fes-app/templates/app/h5/public/logo.png b/packages/create-fes-app/templates/app/h5/public/logo.png
new file mode 100644
index 00000000..4958e971
Binary files /dev/null and b/packages/create-fes-app/templates/app/h5/public/logo.png differ
diff --git a/packages/create-fes-app/templates/app/pc/.eslintrc.js b/packages/create-fes-app/templates/app/pc/.eslintrc.js
index 74c6bf29..6300816f 100644
--- a/packages/create-fes-app/templates/app/pc/.eslintrc.js
+++ b/packages/create-fes-app/templates/app/pc/.eslintrc.js
@@ -1,19 +1,14 @@
-
module.exports = {
- extends: [
- '@webank/eslint-config-webank/vue.js'
+ extends: ['@webank/eslint-config-webank/vue.js'],
+ overrides: [
+ {
+ files: [
+ '**/__tests__/*.{j,t}s?(x)',
+ '**/tests/unit/**/*.spec.{j,t}s?(x)'
+ ]
+ }
],
- globals: {
- // 这里填入你的项目需要的全局变量
- // 这里值为 false 表示这个全局变量不允许被重新赋值,比如:
- //
- // Vue: false
- __DEV__: false
- },
- rules: {
- 'vue/comment-directive': 'off',
- 'global-require': 'off',
- 'import/no-unresolved': 'off',
- 'no-restricted-syntax': 'off'
+ env: {
+ jest: true
}
};
diff --git a/packages/create-fes-app/templates/app/pc/.fes.js b/packages/create-fes-app/templates/app/pc/.fes.js
index 05a57ed4..db77aa6f 100644
--- a/packages/create-fes-app/templates/app/pc/.fes.js
+++ b/packages/create-fes-app/templates/app/pc/.fes.js
@@ -14,7 +14,7 @@ export default {
},
layout: {
title: "Fes.js",
- logo: 'https://gw.alipayobjects.com/zos/rmsportal/KDpgvguMpGfqaHPjicRK.svg',
+ footer: 'Created by MumbelFe',
multiTabs: false,
menus: [{
name: 'index'
@@ -27,5 +27,8 @@ export default {
},
devServer: {
port: 8080
+ },
+ enums: {
+ status: [['0', '无效的'], ['1', '有效的']]
}
};
diff --git a/packages/create-fes-app/templates/app/pc/.gitignore b/packages/create-fes-app/templates/app/pc/.gitignore
index 15fc6484..df903b98 100644
--- a/packages/create-fes-app/templates/app/pc/.gitignore
+++ b/packages/create-fes-app/templates/app/pc/.gitignore
@@ -2,7 +2,7 @@
# dependencies
/node_modules
-
+/coverage
# fes
/src/.fes
diff --git a/packages/create-fes-app/templates/app/pc/__tests__/add.js b/packages/create-fes-app/templates/app/pc/__tests__/add.js
new file mode 100644
index 00000000..d5c874a0
--- /dev/null
+++ b/packages/create-fes-app/templates/app/pc/__tests__/add.js
@@ -0,0 +1,5 @@
+import sum from '@/utils/sum';
+
+test('adds 1 + 2 to equal 3', () => {
+ expect(sum(1, 2)).toBe(3);
+});
diff --git a/packages/create-fes-app/templates/app/pc/package.json b/packages/create-fes-app/templates/app/pc/package.json
index 63a3f925..c0fededc 100644
--- a/packages/create-fes-app/templates/app/pc/package.json
+++ b/packages/create-fes-app/templates/app/pc/package.json
@@ -1,12 +1,13 @@
{
"name": "@webank/fes-template",
- "version": "2.0.0-alpha.1",
+ "version": "2.0.0-alpha",
"description": "fes项目模版",
"scripts": {
"build": "fes build",
"prod": "FES_ENV=prod fes build",
"analyze": "ANALYZE=1 fes build",
- "dev": "fes dev"
+ "dev": "fes dev",
+ "test:unit": "fes test:unit"
},
"keywords": [
"管理端",
@@ -15,6 +16,18 @@
"easy",
"strong"
],
+ "files": [
+ ".eslintrc.js",
+ ".gitignore",
+ ".fes.js",
+ ".fes.prod.js",
+ "mock.js",
+ "package.json",
+ "README.md",
+ "tsconfig.json",
+ "/src",
+ "/config"
+ ],
"repository": {
"type": "git",
"url": "git+https://github.com/WeBankFinTech/fes.js.git",
@@ -26,18 +39,25 @@
"url": "https://github.com/WeBankFinTech/fes.js/issues"
},
"homepage": "https://github.com/WeBankFinTech/fes.js#readme",
+ "publishConfig": {
+ "access": "public"
+ },
"devDependencies": {
- "@vue/compiler-sfc": "^3.0.0",
- "@webank/eslint-config-webank": "0.2.7"
+ "@webank/eslint-config-webank": "0.2.10"
},
"dependencies": {
- "@webank/fes": "^2.0.0-alpha.0",
- "@webank/fes-plugin-access": "^2.0.0-alpha.0",
- "@webank/fes-plugin-layout": "^2.0.0-alpha.0",
- "@webank/fes-plugin-locale": "^2.0.0-alpha.0",
- "@webank/fes-plugin-model": "^2.0.0-alpha.0",
+ "@webank/fes": "^2.0.0-alpha.6",
+ "@webank/fes-plugin-access": "^2.0.0-alpha.6",
+ "@webank/fes-plugin-layout": "^2.0.0-alpha.6",
+ "@webank/fes-plugin-locale": "^2.0.0-alpha.6",
+ "@webank/fes-plugin-model": "^2.0.0-alpha.6",
+ "@webank/fes-plugin-enums": "^2.0.0-alpha.6",
+ "@webank/fes-plugin-jest": "^2.0.0-alpha.6",
+ "@webank/fes-plugin-vuex": "^2.0.0-alpha.6",
"ant-design-vue": "2.0.0-rc.3",
- "vue": "3.0.4"
- }
+ "vue": "3.0.5",
+ "vuex": "^4.0.0-rc.2"
+ },
+ "private": true
}
\ No newline at end of file
diff --git a/packages/create-fes-app/templates/app/pc/public/index.html b/packages/create-fes-app/templates/app/pc/public/index.html
index 2a306e03..8520c0ec 100644
--- a/packages/create-fes-app/templates/app/pc/public/index.html
+++ b/packages/create-fes-app/templates/app/pc/public/index.html
@@ -3,7 +3,8 @@
- fes Vue3
+ fes.js
+
diff --git a/packages/create-fes-app/templates/app/pc/public/logo.png b/packages/create-fes-app/templates/app/pc/public/logo.png
new file mode 100644
index 00000000..4958e971
Binary files /dev/null and b/packages/create-fes-app/templates/app/pc/public/logo.png differ
diff --git a/packages/create-fes-app/templates/app/pc/src/pages/index.vue b/packages/create-fes-app/templates/app/pc/src/pages/index.vue
index 186149c9..3a304634 100644
--- a/packages/create-fes-app/templates/app/pc/src/pages/index.vue
+++ b/packages/create-fes-app/templates/app/pc/src/pages/index.vue
@@ -5,6 +5,11 @@
accessOnepicess1
accessOnepicess2
+ 数据字典
+ {{item.value}}:{{item.key}}
+ {{item.name}}:{{item.disabled}}
+ {{enumsGet('roles', '2', { dir: 'eName' })}}
+ Vuex
@@ -14,9 +19,10 @@
}
diff --git a/packages/create-fes-app/templates/app/pc/src/stores/counter.js b/packages/create-fes-app/templates/app/pc/src/stores/counter.js
new file mode 100644
index 00000000..78070ba3
--- /dev/null
+++ b/packages/create-fes-app/templates/app/pc/src/stores/counter.js
@@ -0,0 +1,23 @@
+export default {
+ namespaced: true,
+ state: () => ({
+ count: 0
+ }),
+ mutations: {
+ increment(state) {
+ state.count++;
+ }
+ },
+ getters: {
+ doubleCount(state) {
+ return state.count * 2;
+ }
+ },
+ actions: {
+ asyncIncrement({ commit }) {
+ setTimeout(() => {
+ commit('increment');
+ }, 2000);
+ }
+ }
+};
diff --git a/packages/create-fes-app/templates/app/pc/src/stores/plugin-loger.js b/packages/create-fes-app/templates/app/pc/src/stores/plugin-loger.js
new file mode 100644
index 00000000..6b8a9826
--- /dev/null
+++ b/packages/create-fes-app/templates/app/pc/src/stores/plugin-loger.js
@@ -0,0 +1,3 @@
+import { createLogger } from 'vuex';
+
+export default createLogger();
diff --git a/packages/create-fes-app/templates/app/pc/src/stores/user.js b/packages/create-fes-app/templates/app/pc/src/stores/user.js
new file mode 100644
index 00000000..e6ffcceb
--- /dev/null
+++ b/packages/create-fes-app/templates/app/pc/src/stores/user.js
@@ -0,0 +1,25 @@
+export default {
+ namespaced: true,
+ state: () => ({
+ name: 'aring',
+ age: 20,
+ count: 0
+ }),
+ mutations: {
+ increment(state) {
+ state.count++;
+ }
+ },
+ getters: {
+ doubleCount(state) {
+ return state.count * 2;
+ }
+ },
+ actions: {
+ asyncIncrement({ commit }) {
+ setTimeout(() => {
+ commit('increment');
+ }, 2000);
+ }
+ }
+};
diff --git a/packages/create-fes-app/templates/app/pc/src/utils/sum.js b/packages/create-fes-app/templates/app/pc/src/utils/sum.js
new file mode 100644
index 00000000..54b28e5a
--- /dev/null
+++ b/packages/create-fes-app/templates/app/pc/src/utils/sum.js
@@ -0,0 +1,3 @@
+export default function sum(a, b) {
+ return a + b;
+}
diff --git a/packages/fes-compiler/README copy.md b/packages/fes-compiler/README copy.md
new file mode 100644
index 00000000..14704fc6
--- /dev/null
+++ b/packages/fes-compiler/README copy.md
@@ -0,0 +1,3 @@
+# fes
+
+一个好用的前端应用解决方案
\ No newline at end of file
diff --git a/packages/fes-compiler/README.md b/packages/fes-compiler/README.md
new file mode 100644
index 00000000..14704fc6
--- /dev/null
+++ b/packages/fes-compiler/README.md
@@ -0,0 +1,3 @@
+# fes
+
+一个好用的前端应用解决方案
\ No newline at end of file
diff --git a/packages/fes-compiler/package.json b/packages/fes-compiler/package.json
index f2ec7324..ff104d38 100644
--- a/packages/fes-compiler/package.json
+++ b/packages/fes-compiler/package.json
@@ -1,6 +1,6 @@
{
"name": "@webank/fes-compiler",
- "version": "2.0.0-alpha.2",
+ "version": "2.0.0-alpha.8",
"description": "@webank/fes-compiler",
"main": "lib/index.js",
"files": [
@@ -27,6 +27,7 @@
"@babel/register": "^7.12.13",
"@babel/preset-env": "^7.12.13",
"@umijs/utils": "3.3.3",
+ "commander": "^7.0.0",
"dotenv": "8.2.0",
"joi": "17.3.0",
"readline": "^1.3.0",
diff --git a/packages/fes-compiler/src/config/index.js b/packages/fes-compiler/src/config/index.js
index 7ce22819..ea8d6a60 100644
--- a/packages/fes-compiler/src/config/index.js
+++ b/packages/fes-compiler/src/config/index.js
@@ -130,41 +130,9 @@ export default class Config {
getUserConfig() {
const configFile = this.getConfigFile();
this.configFile = configFile;
- // 潜在问题:
- // .local 和 .env 的配置必须有 configFile 才有效
- if (configFile) {
- let envConfigFile;
- if (process.env.FES_ENV) {
- const envConfigFileName = this.addAffix(
- configFile,
- process.env.FES_ENV
- );
- const fileNameWithoutExt = envConfigFileName.replace(
- extname(envConfigFileName),
- ''
- );
- envConfigFile = getFile({
- base: this.cwd,
- fileNameWithoutExt,
- type: 'javascript'
- }).filename;
- if (!envConfigFile) {
- throw new Error(
- `get user config failed, ${envConfigFile} does not exist, but process.env.FES_ENV is set to ${process.env.FES_ENV}.`
- );
- }
- }
- const files = [
- configFile,
- envConfigFile,
- this.localConfig && this.addAffix(configFile, 'local')
- ]
- .filter(f => !!f)
- .map(f => join(this.cwd, f))
- .filter(f => existsSync(f));
-
+ if (configFile.length > 0) {
// clear require cache and set babel register
- const requireDeps = files.reduce((memo, file) => {
+ const requireDeps = configFile.reduce((memo, file) => {
memo = memo.concat(parseRequireDeps(file));
return memo;
}, []);
@@ -175,7 +143,7 @@ export default class Config {
});
// require config and merge
- return this.mergeConfig(...this.requireConfigs(files));
+ return this.mergeConfig(...this.requireConfigs(configFile));
}
return {};
}
@@ -201,8 +169,41 @@ export default class Config {
getConfigFile() {
// TODO: support custom config file
- const configFile = CONFIG_FILES.find(f => existsSync(join(this.cwd, f)));
- return configFile ? winPath(configFile) : null;
+ let configFile = CONFIG_FILES.find(f => existsSync(join(this.cwd, f)));
+ if (!configFile) return [];
+ configFile = winPath(configFile);
+ let envConfigFile;
+ // 潜在问题:
+ // .local 和 .env 的配置必须有 configFile 才有效
+ if (process.env.FES_ENV) {
+ const envConfigFileName = this.addAffix(
+ configFile,
+ process.env.FES_ENV
+ );
+ const fileNameWithoutExt = envConfigFileName.replace(
+ extname(envConfigFileName),
+ ''
+ );
+ envConfigFile = getFile({
+ base: this.cwd,
+ fileNameWithoutExt,
+ type: 'javascript'
+ }).filename;
+ if (!envConfigFile) {
+ throw new Error(
+ `get user config failed, ${envConfigFile} does not exist, but process.env.FES_ENV is set to ${process.env.FES_ENV}.`
+ );
+ }
+ }
+ const files = [
+ configFile,
+ envConfigFile,
+ this.localConfig && this.addAffix(configFile, 'local')
+ ]
+ .filter(f => !!f)
+ .map(f => join(this.cwd, f))
+ .filter(f => existsSync(f));
+ return files;
}
getWatchFilesAndDirectories() {
diff --git a/packages/fes-compiler/src/service/index.js b/packages/fes-compiler/src/service/index.js
index 50c2a34f..bc7cc5ba 100644
--- a/packages/fes-compiler/src/service/index.js
+++ b/packages/fes-compiler/src/service/index.js
@@ -3,11 +3,12 @@ import { EventEmitter } from 'events';
import assert from 'assert';
import { AsyncSeriesWaterfallHook } from 'tapable';
import { existsSync } from 'fs';
-import { lodash } from '@umijs/utils';
-import BabelRegister from './babelRegister';
+import { lodash, chalk } from '@umijs/utils';
+import { Command, Option } from 'commander';
import { resolvePresets, pathToObj, resolvePlugins } from './utils/pluginUtils';
import loadDotEnv from './utils/loadDotEnv';
import isPromise from './utils/isPromise';
+import BabelRegister from './babelRegister';
import PluginAPI from './pluginAPI';
import {
ApplyPluginsType,
@@ -88,6 +89,8 @@ export default class Service extends EventEmitter {
// repoDir should be the root dir of repo
this.pkg = opts.pkg || this.resolvePackage();
this.env = opts.env || process.env.NODE_ENV;
+ this.fesPkg = opts.fesPkg || {};
+
assert(existsSync(this.cwd), `cwd ${this.cwd} does not exist.`);
@@ -112,6 +115,10 @@ export default class Service extends EventEmitter {
env: this.env
});
+ this.program = new Command();
+
+ this.initCommand();
+
// setup initial plugins
const baseOpts = {
pkg: this.pkg,
@@ -262,6 +269,7 @@ export default class Service extends EventEmitter {
'paths',
'cwd',
'pkg',
+ 'configInstance',
'userConfig',
'config',
'env',
@@ -483,12 +491,28 @@ export default class Service extends EventEmitter {
}
}
- async run({ name, args = {} }) {
- args._ = args._ || [];
- // shift the command itself
- if (args._[0] === name) args._.shift();
+ initCommand() {
+ this.program
+ .usage(' [options]')
+ .version(`@webank/fes ${this.fesPkg.version}`, '-v, --vers', 'output the current version')
+ .description(chalk.cyan('一个好用的前端应用解决方案'));
+ }
- this.args = args;
+ parseCommand() {
+ this.program.on('--help', () => {
+ console.log();
+ console.log(
+ ` Run ${chalk.cyan(
+ 'fes --help'
+ )} for detailed usage of given command.`
+ );
+ console.log();
+ });
+ this.program.commands.forEach(c => c.on('--help', () => console.log()));
+ this.program.parse(process.argv);
+ }
+
+ async run({ rawArgv = {}, args = {} }) {
await this.init();
this.setStage(ServiceStage.run);
@@ -500,27 +524,37 @@ export default class Service extends EventEmitter {
}
});
- return this.runCommand({
- name,
- args
- });
+ this.runCommand({ rawArgv, args });
}
- async runCommand({ name, args = {} }) {
+ async runCommand({ rawArgv = {}, args = {} }) {
assert(this.stage >= ServiceStage.init, 'service is not initialized.');
- args._ = args._ || [];
- // shift the command itself
- if (args._[0] === name) args._.shift();
-
- const command = typeof this.commands[name] === 'string'
- ? this.commands[this.commands[name]]
- : this.commands[name];
- assert(command, `run command failed, command ${name} does not exists.`);
-
- const { fn } = command;
- return fn({
- args
+ Object.keys(this.commands).forEach((command) => {
+ const commandOptionConfig = this.commands[command];
+ const program = this.program;
+ let c = program.command(command).description(commandOptionConfig.description);
+ if (Array.isArray(commandOptionConfig.options)) {
+ commandOptionConfig.options.forEach((config) => {
+ const option = new Option(config.name, config.description);
+ if (config.default) {
+ option.default(config.default);
+ }
+ if (config.choices) {
+ option.choices(config.choices);
+ }
+ c = c.addOption(option);
+ });
+ }
+ if (commandOptionConfig.fn) {
+ c.action(async () => {
+ await commandOptionConfig.fn({
+ rawArgv, args, options: c.opts(), program
+ });
+ });
+ }
});
+
+ this.parseCommand();
}
}
diff --git a/packages/fes-compiler/src/service/pluginAPI.js b/packages/fes-compiler/src/service/pluginAPI.js
index d4d67511..ffa83f3f 100644
--- a/packages/fes-compiler/src/service/pluginAPI.js
+++ b/packages/fes-compiler/src/service/pluginAPI.js
@@ -3,6 +3,7 @@ import * as utils from '@umijs/utils';
import { isValidPlugin, pathToObj } from './utils/pluginUtils';
import { EnableBy, PluginType, ServiceStage } from './enums';
import Logger from '../logger';
+
// TODO
// 标准化 logger
export default class PluginAPI {
@@ -62,16 +63,21 @@ export default class PluginAPI {
).concat(hook);
}
- registerCommand(command) {
- const { name, alias } = command;
+ registerCommand(commandOption) {
+ const { command, fn } = commandOption;
assert(
- !this.service.commands[name],
- `api.registerCommand() failed, the command ${name} is exists.`
+ !this.service.commands[command],
+ `api.registerCommand() failed, the command ${command} is exists.`
);
- this.service.commands[name] = command;
- if (alias) {
- this.service.commands[alias] = name;
- }
+ assert(
+ typeof command === 'string',
+ 'api.registerCommand() failed, the command must be String.'
+ );
+ assert(
+ typeof fn === 'function',
+ 'api.registerCommand() failed, the fn must be function.'
+ );
+ this.service.commands[command] = commandOption;
}
// 在 preset 初始化阶段放后面,在插件注册阶段放前面
diff --git a/packages/fes-plugin-access/README.md b/packages/fes-plugin-access/README.md
index 139597f9..14704fc6 100644
--- a/packages/fes-plugin-access/README.md
+++ b/packages/fes-plugin-access/README.md
@@ -1,2 +1,3 @@
+# fes
-
+一个好用的前端应用解决方案
\ No newline at end of file
diff --git a/packages/fes-plugin-access/package.json b/packages/fes-plugin-access/package.json
index 2b75b44c..1470ab11 100644
--- a/packages/fes-plugin-access/package.json
+++ b/packages/fes-plugin-access/package.json
@@ -1,6 +1,6 @@
{
"name": "@webank/fes-plugin-access",
- "version": "2.0.0-alpha.2",
+ "version": "2.0.0-alpha.8",
"description": "@webank/fes-plugin-access",
"main": "lib/index.js",
"files": [
@@ -28,6 +28,6 @@
},
"peerDependencies": {
"@webank/fes": "^2.0.0-alpha.0",
- "vue": "^3.0.4"
+ "vue": "3.0.5"
}
}
diff --git a/packages/fes-plugin-enums/README.md b/packages/fes-plugin-enums/README.md
index 139597f9..14704fc6 100644
--- a/packages/fes-plugin-enums/README.md
+++ b/packages/fes-plugin-enums/README.md
@@ -1,2 +1,3 @@
+# fes
-
+一个好用的前端应用解决方案
\ No newline at end of file
diff --git a/packages/fes-plugin-enums/package.json b/packages/fes-plugin-enums/package.json
index 8c96c2b5..01f55fda 100644
--- a/packages/fes-plugin-enums/package.json
+++ b/packages/fes-plugin-enums/package.json
@@ -1,6 +1,6 @@
{
"name": "@webank/fes-plugin-enums",
- "version": "2.0.0-alpha.2",
+ "version": "2.0.0-alpha.8",
"description": "@webank/fes-plugin-enums",
"main": "lib/index.js",
"files": [
@@ -28,6 +28,6 @@
},
"peerDependencies": {
"@webank/fes": "^2.0.0-alpha.0",
- "vue": "^3.0.4"
+ "vue": "3.0.5"
}
}
diff --git a/packages/fes-plugin-enums/src/runtime/core.tpl b/packages/fes-plugin-enums/src/runtime/core.tpl
index e5789ce1..325a59d7 100644
--- a/packages/fes-plugin-enums/src/runtime/core.tpl
+++ b/packages/fes-plugin-enums/src/runtime/core.tpl
@@ -25,13 +25,15 @@ function get(name, key, opt = { dir: 'value', extend: []}) {
key = ''
}
let list = ENUMS[name] || []
+ let value
if (key) {
let res = list.filter(item => item.key === key)[0]
if (!res) return key
- return readonly(parseValueDir(res.value, opt.dir) || key)
+ value = parseValueDir(res.value, opt.dir) || key
} else {
- return readonly(format(list, opt.extend))
+ value = format(list, opt.extend)
}
+ return typeof value === 'object' ? readonly(value) : value
}
/**
diff --git a/packages/fes-plugin-icon/README.md b/packages/fes-plugin-icon/README.md
new file mode 100644
index 00000000..14704fc6
--- /dev/null
+++ b/packages/fes-plugin-icon/README.md
@@ -0,0 +1,3 @@
+# fes
+
+一个好用的前端应用解决方案
\ No newline at end of file
diff --git a/packages/fes-plugin-icon/package.json b/packages/fes-plugin-icon/package.json
index fb9d5cd1..063d892f 100644
--- a/packages/fes-plugin-icon/package.json
+++ b/packages/fes-plugin-icon/package.json
@@ -1,6 +1,6 @@
{
"name": "@webank/fes-plugin-icon",
- "version": "2.0.0-alpha.2",
+ "version": "2.0.0-alpha.8",
"description": "@webank/fes-plugin-icon",
"main": "lib/index.js",
"files": [
@@ -27,7 +27,7 @@
"access": "public"
},
"peerDependencies": {
- "vue": "^3.0.4"
+ "vue": "3.0.5"
},
"dependencies": {
"svgo": "1.3.2"
diff --git a/packages/fes-plugin-jest/.fatherrc.js b/packages/fes-plugin-jest/.fatherrc.js
new file mode 100644
index 00000000..332f1bff
--- /dev/null
+++ b/packages/fes-plugin-jest/.fatherrc.js
@@ -0,0 +1,3 @@
+export default {
+ disableTypeCheck: false,
+};
diff --git a/packages/fes-plugin-jest/LICENSE b/packages/fes-plugin-jest/LICENSE
new file mode 100644
index 00000000..0978fbf7
--- /dev/null
+++ b/packages/fes-plugin-jest/LICENSE
@@ -0,0 +1,21 @@
+MIT License
+
+Copyright (c) 2020-present webank
+
+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.
\ No newline at end of file
diff --git a/packages/fes-plugin-jest/README.md b/packages/fes-plugin-jest/README.md
new file mode 100644
index 00000000..14704fc6
--- /dev/null
+++ b/packages/fes-plugin-jest/README.md
@@ -0,0 +1,3 @@
+# fes
+
+一个好用的前端应用解决方案
\ No newline at end of file
diff --git a/packages/fes-plugin-jest/helpers/setupFiles/jasmine.js b/packages/fes-plugin-jest/helpers/setupFiles/jasmine.js
new file mode 100644
index 00000000..addfbcc4
--- /dev/null
+++ b/packages/fes-plugin-jest/helpers/setupFiles/jasmine.js
@@ -0,0 +1 @@
+jasmine.DEFAULT_TIMEOUT_INTERVAL = 200000; // eslint-disable-line
diff --git a/packages/fes-plugin-jest/helpers/setupFiles/shim.js b/packages/fes-plugin-jest/helpers/setupFiles/shim.js
new file mode 100644
index 00000000..8c5e66ec
--- /dev/null
+++ b/packages/fes-plugin-jest/helpers/setupFiles/shim.js
@@ -0,0 +1,6 @@
+require('core-js/stable');
+require('regenerator-runtime/runtime');
+
+if (typeof window !== 'undefined') {
+ require('whatwg-fetch');
+}
diff --git a/packages/fes-plugin-jest/helpers/transformers/css.js b/packages/fes-plugin-jest/helpers/transformers/css.js
new file mode 100644
index 00000000..327fba7c
--- /dev/null
+++ b/packages/fes-plugin-jest/helpers/transformers/css.js
@@ -0,0 +1,9 @@
+module.exports = {
+ process() {
+ return 'module.exports = {};';
+ },
+ getCacheKey() {
+ // The output is always the same.
+ return 'css';
+ }
+};
diff --git a/packages/fes-plugin-jest/helpers/transformers/javascript.js b/packages/fes-plugin-jest/helpers/transformers/javascript.js
new file mode 100644
index 00000000..c232155e
--- /dev/null
+++ b/packages/fes-plugin-jest/helpers/transformers/javascript.js
@@ -0,0 +1,7 @@
+const babelJest = require('babel-jest');
+
+module.exports = babelJest.createTransformer({
+ presets: [require.resolve('@umijs/babel-preset-umi/node')],
+ babelrc: false,
+ configFile: false
+});
diff --git a/packages/fes-plugin-jest/package.json b/packages/fes-plugin-jest/package.json
new file mode 100644
index 00000000..54d75c18
--- /dev/null
+++ b/packages/fes-plugin-jest/package.json
@@ -0,0 +1,50 @@
+{
+ "name": "@webank/fes-plugin-jest",
+ "version": "2.0.0-alpha.8",
+ "description": "@webank/fes-plugin-jest",
+ "main": "lib/index.js",
+ "files": [
+ "lib",
+ "helpers"
+ ],
+ "scripts": {
+ "test": "echo \"Error: no test specified\" && exit 1"
+ },
+ "repository": {
+ "type": "git",
+ "url": "git+https://github.com/WeBankFinTech/fes.js.git",
+ "directory": "packages/fes-plugin-jest"
+ },
+ "keywords": [
+ "fes",
+ "unit",
+ "jest"
+ ],
+ "author": "harrywan",
+ "license": "MIT",
+ "bugs": {
+ "url": "https://github.com/WeBankFinTech/fes.js/issues"
+ },
+ "homepage": "https://github.com/WeBankFinTech/fes.js#readme",
+ "publishConfig": {
+ "access": "public"
+ },
+ "dependencies": {
+ "@babel/core": "7.11.6",
+ "@umijs/babel-preset-umi": "3.2.24",
+ "@webank/fes-compiler": "^2.0.0-alpha.8",
+ "babel-core": "^7.0.0-bridge.0",
+ "babel-jest": "^26.6.3",
+ "core-js": "3.6.5",
+ "jest": "^26.6.3",
+ "jest-cli": "^26.6.3",
+ "jest-serializer-vue": "^2.0.2",
+ "jest-transform-stub": "^2.0.0",
+ "jest-watch-typeahead": "^0.6.1",
+ "regenerator-runtime": "^0.13.7",
+ "ts-jest": "^26.5.0",
+ "typescript": "~4.1.2",
+ "vue-jest": "^5.0.0-0",
+ "whatwg-fetch": "^3.4.1"
+ }
+}
diff --git a/packages/fes-plugin-jest/src/createDefaultConfig.js b/packages/fes-plugin-jest/src/createDefaultConfig.js
new file mode 100644
index 00000000..809809d1
--- /dev/null
+++ b/packages/fes-plugin-jest/src/createDefaultConfig.js
@@ -0,0 +1,60 @@
+import { existsSync } from 'fs';
+import { join } from 'path';
+
+export default (cwd, args) => {
+ const testMatchTypes = ['spec', 'test'];
+ if (args.e2e) {
+ testMatchTypes.push('e2e');
+ }
+ const hasSrc = existsSync(join(cwd, 'src'));
+ return {
+ collectCoverageFrom: [
+ 'index.{js,jsx,ts,tsx,vue}',
+ hasSrc && 'src/**/*.{js,jsx,ts,tsx,vue}',
+ '!**/.fes/**',
+ '!**/typings/**',
+ '!**/types/**',
+ '!**/fixtures/**',
+ '!**/examples/**',
+ '!**/*.d.ts'
+ ].filter(Boolean),
+ moduleFileExtensions: [
+ 'js',
+ 'jsx',
+ 'json',
+ // tell Jest to handle *.vue files
+ 'vue'
+ ],
+ transform: {
+ // process *.vue files with vue-jest
+ '^.+\\.vue$': require.resolve('vue-jest'),
+ '.+\\.(css|styl|less|sass|scss|jpg|jpeg|png|svg|gif|eot|otf|webp|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$':
+ require.resolve('jest-transform-stub'),
+ '^.+\\.jsx?$': require.resolve(
+ '../helpers/transformers/javascript'
+ )
+ },
+ setupFiles: [require.resolve('../helpers/setupFiles/shim')],
+ setupFilesAfterEnv: [require.resolve('../helpers/setupFiles/jasmine')],
+ transformIgnorePatterns: ['/node_modules/'],
+ // support the same @ -> src alias mapping in source code
+ moduleNameMapper: {
+ '^@/(.*)$': '/src/$1'
+ },
+ // serializer for snapshots
+ snapshotSerializers: [
+ 'jest-serializer-vue'
+ ],
+ testMatch: [
+ `**/tests/**/*.(${testMatchTypes.join('|')}).[jt]s?(x)`,
+ '**/__tests__/**/*.[jt]s?(x)'
+ ],
+ // https://github.com/facebook/jest/issues/6766
+ testURL: 'http://localhost/',
+ watchPlugins: [
+ require.resolve('jest-watch-typeahead/filename'),
+ require.resolve('jest-watch-typeahead/testname')
+ ],
+ verbose: true
+ };
+};
diff --git a/packages/fes-plugin-jest/src/index.js b/packages/fes-plugin-jest/src/index.js
new file mode 100644
index 00000000..2081b5c5
--- /dev/null
+++ b/packages/fes-plugin-jest/src/index.js
@@ -0,0 +1,97 @@
+
+import assert from 'assert';
+import { join } from 'path';
+import { existsSync } from 'fs';
+import { Logger } from '@webank/fes-compiler';
+import { options as CliOptions } from 'jest-cli/build/cli/args';
+import createDefaultConfig from './createDefaultConfig';
+
+const logger = new Logger('fes:plugin-unit-jest');
+
+function getCommandOptiton() {
+ const opts = [];
+ Object.keys(CliOptions).forEach((key) => {
+ const option = CliOptions[key];
+ const opt = {};
+ if (key !== 'version') {
+ if (option.alias) {
+ opt.name = `-${option.alias} --${key}`;
+ } else {
+ opt.name = `--${key}`;
+ }
+ opt.description = option.description;
+ opts.push(opt);
+ }
+ });
+ return opts;
+}
+
+export default function (api) {
+ const { utils: { mergeConfig }, cwd } = api;
+
+ api.registerCommand({
+ command: 'test',
+ description: 'run unit tests with jest',
+ options: getCommandOptiton(),
+ async fn({ args }) {
+ process.env.NODE_ENV = 'test';
+
+ if (args._[0] === 'test') {
+ args._.shift();
+ }
+
+ args.debug && logger.log(`args: ${JSON.stringify(args)}`);
+
+ // Read config from cwd/jest.config.js
+ const userJestConfigFile = join(cwd, 'jest.config.js');
+ const userJestConfig = existsSync(userJestConfigFile) && require(userJestConfigFile);
+ args.debug && logger.log(`config from jest.config.js: ${JSON.stringify(userJestConfig)}`);
+
+ // Read jest config from package.json
+ const packageJSONPath = join(cwd, 'package.json');
+ const packageJestConfig = existsSync(packageJSONPath) && require(packageJSONPath).jest;
+ args.debug && logger.log(`jest config from package.json: ${JSON.stringify(packageJestConfig)}`);
+
+ // Merge configs
+ // user config and args config could have value function for modification
+ const config = mergeConfig(
+ createDefaultConfig(cwd, args),
+ packageJestConfig,
+ userJestConfig
+ );
+ args.debug && logger.log(`final config: ${JSON.stringify(config)}`);
+
+ // Generate jest options
+ const argsConfig = Object.keys(CliOptions).reduce((prev, name) => {
+ if (args[name]) prev[name] = args[name];
+
+ // Convert alias args into real one
+ const { alias } = CliOptions[name];
+ if (alias && args[alias]) prev[name] = args[alias];
+ return prev;
+ }, {});
+ args.debug && logger.log(`config from args: ${JSON.stringify(argsConfig)}`);
+
+ // 比较大的库建议使用require,使用时才加载,提升fes命令的效率
+ const { runCLI } = require('jest');
+ // Run jest
+ const result = await runCLI(
+ {
+ // @ts-ignore
+ _: args._ || [],
+ // @ts-ignore
+ $0: args.$0 || '',
+ // 必须是单独的 config 配置,值为 string,否则不生效
+ // @ts-ignore
+ config: JSON.stringify(config),
+ ...argsConfig
+ },
+ [cwd]
+ );
+ args.debug && logger.log(result);
+
+ // Throw error when run failed
+ assert(result.results.success, 'Test with jest failed');
+ }
+ });
+}
diff --git a/packages/fes-plugin-layout/package.json b/packages/fes-plugin-layout/package.json
index a04d77fe..75d3b32a 100644
--- a/packages/fes-plugin-layout/package.json
+++ b/packages/fes-plugin-layout/package.json
@@ -1,6 +1,6 @@
{
"name": "@webank/fes-plugin-layout",
- "version": "2.0.0-alpha.5",
+ "version": "2.0.0-alpha.8",
"description": "@webank/fes-plugin-layout",
"main": "lib/index.js",
"files": [
@@ -33,6 +33,6 @@
"@ant-design/icons-vue": "^5.1.6",
"@webank/fes": "^2.0.0-alpha.0",
"ant-design-vue": "2.0.0-rc.3",
- "vue": "^3.0.4"
+ "vue": "3.0.5"
}
}
diff --git a/packages/fes-plugin-layout/src/index.js b/packages/fes-plugin-layout/src/index.js
index b4e8efcf..b293597a 100644
--- a/packages/fes-plugin-layout/src/index.js
+++ b/packages/fes-plugin-layout/src/index.js
@@ -28,11 +28,10 @@ export default (api) => {
const HAS_LOCALE = api.hasPlugins(['@webank/fes-plugin-locale']);
- const HAS_ACCESS = api.hasPlugins(['@webank/fes-plugin-access']);
-
// .fes配置
const userConfig = {
title: name,
+ footer: 'Created by Fes.js',
...(api.config.layout || {})
};
diff --git a/packages/fes-plugin-layout/src/runtime/views/BaseLayout.vue b/packages/fes-plugin-layout/src/runtime/views/BaseLayout.vue
index 0773a683..df25b5a9 100644
--- a/packages/fes-plugin-layout/src/runtime/views/BaseLayout.vue
+++ b/packages/fes-plugin-layout/src/runtime/views/BaseLayout.vue
@@ -56,8 +56,8 @@
-