+Fes.js 支持插件和插件集,通过这张图应该很好理解到他们的关系,通过插件集我们把插件收敛依赖然后支持不同的业务类型。
-### GitBook
+### .fes 临时文件
+.fes 临时目录是整个 Fes.js 项目的发动机,你的入口文件、路由等等都在这里,这些是由 fes 内部插件及三方插件生成的。
-We’ve been using GitBook for most of our sub project docs. The primary problem with GitBook is that its development reload performance is intolerable with a large amount of files. The default theme also has a pretty limiting navigation structure, and the theming system is, again, not Vue based. The team behind GitBook is also more focused on turning it into a commercial product rather than an open-source tool.
+你通常会在 .fes 下看到以下目录
+```
++ .fes
+ + core # 内部插件生成
+ + pluginA # 外部插件生成
+ + presetB # 外部插件生成
+ + fes.js # 入口文件
+```
+
+临时文件是 Fes.js 中非常重要的一部分,框架或插件会根据你的代码生成临时文件,这些原来需要放在项目里的脏乱差的部分都被藏在了这里。
+
+你可以在这里调试代码,但不要在 .git 仓库里提交他,因为他的临时性,每次启动 fes 时都会被删除并重新生成。
+
+
+
+## 为什么不是 ...?
+
+### Vue CLI
+
+Vue CLI 是基于 Vue.js 进行快速开发的完整系统,提供交互式脚手架、丰富的官方插件,并且可通过插件进行扩展,他在打包层把体验做到了极致,但是不包含路由,不是框架。所以,如果大家想基于他修改部分配置,或者希望在打包层之外也做技术收敛时,就会遇到困难。
+
+### UMI
+
+UMI 是个很好的选择,Fes.js 很多功能是借鉴 UMI 做的。UMI 是基于 React 封装的应用级框架,贯彻着函数式编程的思维。而 Vue 有所不同,虽然 Vue 3.0 向函数式迈了一大步,但大家可能依然喜欢编写 `.vue`文件,而非 `.jsx` 文件。两种思维方式会导致部分API设计上有所差异,虽然 UMI 有 `plugin-vue` ,但不太 "vue"。推荐喜欢 React 的同学使用 UMI。
diff --git a/docs/guide/advanced/markdown.md b/docs/guide/advanced/markdown.md
deleted file mode 100644
index b781a528..00000000
--- a/docs/guide/advanced/markdown.md
+++ /dev/null
@@ -1,102 +0,0 @@
-# Markdown and Vue SFC
-
-Each Markdown file is first compiled into HTML, and then converted to a Vue SFC. In other words, you can take Markdown as Vue SFC:
-
-- Blocks `
-
-
-```
-
-**Output**
-
-_Hello, {{ msg }}_
-
-
-
-_Current count is: {{ count }}_
-
-
-
-
-
-
-
-
diff --git a/docs/guide/advanced/plugin.md b/docs/guide/advanced/plugin.md
deleted file mode 100644
index d0251ed7..00000000
--- a/docs/guide/advanced/plugin.md
+++ /dev/null
@@ -1,3 +0,0 @@
-# Writing a Plugin
-
-> TODO
diff --git a/docs/guide/advanced/theme.md b/docs/guide/advanced/theme.md
deleted file mode 100644
index 89476481..00000000
--- a/docs/guide/advanced/theme.md
+++ /dev/null
@@ -1,3 +0,0 @@
-# Writing a Theme
-
-> TODO
diff --git a/docs/guide/assets.md b/docs/guide/assets.md
deleted file mode 100644
index 5cd1b8b3..00000000
--- a/docs/guide/assets.md
+++ /dev/null
@@ -1,104 +0,0 @@
-# Assets
-
-## Relative URLs
-
-You can reference any assets using relative URLs in your Markdown content:
-
-```md
-
-```
-
-This is generally the suggested way to import images, as users usually place images near the Markdown file that references them.
-
-## Public Files
-
-You can put some static assets inside public directory, and they will be copied to the root of the generated directory.
-
-The default public directory is `.vuepress/public`, which can be changed in config.
-
-It would be useful in some cases:
-
-- You may need to provide static assets that are not directly referenced in any of your Markdown files, for example, favicon and PWA icons.
-- You may need to serve some shared static assets, which may even be referenced outside your site, for example, logo images.
-- You may want to reference images using absolute URLs in your Markdown content.
-
-Take our documentation source files as an example, we are putting the logo of VuePress inside the public directory:
-
-```bash
-└─ docs
- ├─ .vuepress
- | └─ public
- | └─ hero.png # <- Logo file
- └─ guide
- └─ assets.md # <- Here we are
-```
-
-We can reference our logo in current page like this:
-
-**Input**
-
-```md
-
-```
-
-**Output**
-
-
-
-::: tip
-Config reference: [public](../reference/config.md#public)
-:::
-
-### Base Helper
-
-If your site is deployed to a non-root URL, i.e. the [base](../reference/config.md#base) is not `"/"`, you will need to prepend the `base` to the absolute URLs of your public files.
-
-For example, if you plan to deploy your site to `https://foo.github.io/bar/`, then `base` should be set to `"/bar/"`, and you have to reference your public files in Markdown like this:
-
-```md
-
-```
-
-Obviously, it is brittle if you ever decide to change the `base`. This is the reason why we suggest to reference static assets using relative URLs.
-
-To help with that, VuePress provides a built-in helper `$withBase` that generates the correct path:
-
-```md
-
-```
-
-The helper is verbose in Markdown. So it might be more helpful for theme and plugin authors.
-
-::: tip
-Config reference: [base](../reference/config.md#base)
-:::
-
-## Packages and Path Aliases
-
-Although it is not a common usage, you can reference images from dependent packages:
-
-```bash
-npm install -D package-name
-```
-
-```md
-
-```
-
-The path aliases that set in config file are also supported:
-
-```js
-module.exports = {
- alias: {
- '@alias': '/path/to/some/dir',
- },
-}
-```
-
-```md
-
-```
-
-::: tip
-Config reference: [alias](../reference/config.md#alias)
-:::
diff --git a/docs/guide/bundler.md b/docs/guide/bundler.md
deleted file mode 100644
index 82ea1f1c..00000000
--- a/docs/guide/bundler.md
+++ /dev/null
@@ -1,3 +0,0 @@
-# Bundler
-
-> TODO
diff --git a/docs/guide/config.md b/docs/guide/config.md
new file mode 100644
index 00000000..206bc963
--- /dev/null
+++ b/docs/guide/config.md
@@ -0,0 +1,79 @@
+# 配置
+
+Fes.js 约定 `.fes.js` 文件为项目基础配置文件,一份常见的配置示例如下:
+```js
+export default {
+ base: '/foo/',
+ publicPath: '/',
+ devServer: {
+ port: 8080
+ }
+ mock: {
+ prefix: '/v2'
+ },
+ proxy: {
+ '/v2': {
+ 'target': 'https://api.douban.com/',
+ 'changeOrigin': true,
+ },
+ },
+ layout: {
+ title: "Fes.js",
+ footer: 'Created by MumbelFe',
+ multiTabs: false,
+ menus: [{
+ name: 'index'
+ }, {
+ name: 'onepiece'
+ }, {
+ name: 'store'
+ }, {
+ name: 'simpleList'
+ }]
+ }
+}
+```
+
+## 配置文件
+可以新建 `.fes.local.js` 作为本地临时配置文件。这份配置会和 `.fes.js` 做 deep merge 后形成最终配置。
+```js
+// .fes.js
+export default { mock: false };
+
+// .fes.local.js
+export default {
+ mock: true,
+ dvServer: { port: 8080 }
+};
+```
+最终的配置是:
+```js
+{
+ mock: true,
+ devServer: { port: 8080 }
+};
+```
+::: tip
+`.fes.local.js` 仅在 fes dev 时有效。
+
+`.fes.local.js` 是本地验证使用的临时配置,请将其添加到 `.gitignore`,务必不要提交到 git 仓库中。
+
+`.fes.local.js` 配置的优先级最高,比 `FES_ENV` 指定的配置更高。
+:::
+
+## 多环境多份配置
+可以通过环境变量 `FES_ENV` 区分不同环境,来指定配置文件。
+```js
+ // .fes.js
+ export default { mock: false };
+
+ // .fes.local.js
+ export default {
+ mock: true,
+ dvServer: { port: 8080 }
+ };
+```
+根据指定的 `FES_ENV` 拿对应的配置,这份配置会和 `.fes.js` 做 deep merge 后形成最终配。
+::: tip
+`FES_ENV` 指定的配置优先级高于 `.fes.js` 文件的配置
+:::
\ No newline at end of file
diff --git a/docs/guide/configuration.md b/docs/guide/configuration.md
deleted file mode 100644
index 0aaf3f66..00000000
--- a/docs/guide/configuration.md
+++ /dev/null
@@ -1,84 +0,0 @@
-# Configuration
-
-## Config File
-
-Without any configuration, the VuePress site is pretty minimal. To customize your site, let’s first create a `.vuepress` directory inside your docs directory. This is where all VuePress-specific files will be placed. Your project structure is probably like this:
-
-```
-├─ docs
-│ ├─ .vuepress
-│ │ └─ config.js
-│ └─ README.md
-├─ .gitignore
-└─ package.json
-```
-
-The essential file for configuring a VuePress site is `.vuepress/config.js`, which should export a JavaScript object. If you are using TypeScript, you can use `.vuepress/config.ts` instead to get better types hint for VuePress Config.
-
-
-
-
-```js
-module.exports = {
- lang: 'en-US',
- title: 'Hello, VuePress!',
- description: 'This is my first VuePress site',
-
- themeConfig: {
- logo: 'https://vuejs.org/images/logo.png',
- },
-}
-```
-
-
-
-
-
-```ts
-import type { UserConfig, DefaultThemeOptions } from 'vuepress'
-
-const config: UserConfig = {
- lang: 'en-US',
- title: 'Hello VuePress',
- description: 'Just playing around',
-
- themeConfig: {
- logo: 'https://vuejs.org/images/logo.png',
- },
-}
-
-export = config
-```
-
-
-
-
-::: tip
-We will refer the config object as **VuePress Config**.
-:::
-
-## Config Scopes
-
-You may have noticed that there is a `themeConfig` option in VuePress Config.
-
-Options outside `themeConfig` are **Site Config**, while options inside `themeConfig` are **Theme Config**.
-
-### Site Config
-
-Site config means that, no matter what theme you are using, these configurations are always valid.
-
-As we know, every site should have its own `lang`, `title`, `description`, etc. Thus, VuePress has built-in support for those options.
-
-::: tip
-Check out the [Config Reference](../reference/config.md) for a full list of site config.
-:::
-
-### Theme Config
-
-Theme config will be processed by VuePress theme, so it depends on the theme you are using.
-
-If you don't specify the `theme` option of VuePress Config, the default theme will be used.
-
-::: tip
-Check out the [Default Theme > Config Reference](../reference/default-theme/config.md) for theme config of default theme.
-:::
diff --git a/docs/guide/contributing.md b/docs/guide/contributing.md
new file mode 100644
index 00000000..b17a1746
--- /dev/null
+++ b/docs/guide/contributing.md
@@ -0,0 +1,70 @@
+# 贡献指南
+
+## 概览
+
+项目仓库借助于 [Yarn Classic 工作区](https://classic.yarnpkg.com/zh-Hans/docs/workspaces) 来实现 [Monorepo](https://en.wikipedia.org/wiki/Monorepo) ,在 `packages` 目录下存放了多个互相关联的独立 Package 。
+
+- `@fesjs/create-fes-app`: 创建项目模板模块。提供`create-fes-app`命令,提供创建多种类型项目模板的能力。
+
+- `@fesjs/fes`: 入口模块。提供`fes`命令和 API 入口。
+
+- `@fesjs/compiler`: 编译时插件管理模块。定义插件的生命周期、插件配置、插件通讯机制等。
+
+- `@fesjs/runtime`: 运行时插件模块。集成了vue-router,定义运行时插件生命周期、插件通讯机制。
+
+- `@fesjs/preset-build-in`: 内置插件集。包含`dev`、`build`等命令,集成webpack5+babel,提供方便编写插件的API,入口文件处理,路由处理等能力。
+
+- `@fesjs/fes-template`: 适用于PC类型的模板项目。
+
+- `@fesjs/fes-template-h5`: 适用于H5类型的模板项目。
+
+- `@fesjs/plugin-${name}`: 官方插件。
+
+- `@fesjs/fes`: 是 `@fesjs/compiler` + `@fesjs/runtime` + `@fesjs/preset-build-in` 的封装。用户只需要安装此依赖和额外的插件或者插件集。
+
+## 开发配置
+
+开发要求:
+
+- [Node.js](http://nodejs.org) **version 12+**
+- [Yarn v1 classic](https://classic.yarnpkg.com/zh-Hans/docs/install)
+
+克隆代码仓库,并安装依赖:
+
+```bash
+yarn
+```
+
+监听源文件修改:
+
+```bash
+yarn build
+```
+
+打开另一个终端,开始开发项目文档网站:
+
+```bash
+yarn docs:dev
+```
+
+本项目开发使用的一些主要工具:
+
+- [Jest](https://jestjs.io/) 用于单元测试
+- [ESLint](https://eslint.org/) + [Prettier](https://prettier.io/) 用于代码检查和格式化
+- [@umi/father](https://github.com/umijs/father) 用于将ES6语法编译成ES5或者CommonJS
+
+## 开发脚本
+
+### `yarn build`
+
+`build` 命令会使用 `father-build` 将 ES6 编译为 CommonJS。
+
+本项目在编写Node端的代码时也用ES6,所以你在克隆代码仓库后,可能需要先执行该命令来确保项目代码可以顺利运行,因为编译后的 JS 文件被 `.gitignore` 排除在仓库以外了。
+### `yarn docs:dev`
+`docs:` 前缀表明,这些命令是针对文档 (documentation) 进行操作的,即 `docs` 目录。
+使用 Vue Press在本地启动文档网站服务器,用于实时查看文档效果。
+
+### 调试功能
+在开发完插件代码后,需要在template项目中验证功能
+- 进入`packages/template`目录
+- 执行`yarn dev`
\ No newline at end of file
diff --git a/docs/guide/deployment.md b/docs/guide/deployment.md
deleted file mode 100644
index 5906a27c..00000000
--- a/docs/guide/deployment.md
+++ /dev/null
@@ -1,205 +0,0 @@
-# Deployment
-
-The following guides are based on some shared assumptions:
-
-- You are placing your Markdown source files inside the `docs` directory of your project;
-- You are using the default build output location (`.vuepress/dist`);
-- You are using [yarn classic](https://classic.yarnpkg.com/en/) as package manager, while npm is also supported;
-- VuePress is installed as a local dependency in your project, and you have setup the following script in `package.json`:
-
-```json
-{
- "scripts": {
- "docs:build": "vuepress build docs"
- }
-}
-```
-
-## GitHub Pages
-
-1. Set the correct [base](../reference/config.md#base) config.
-
- If you are deploying to `https://.github.io/`, you can omit this step as `base` defaults to `"/"`.
-
- If you are deploying to `https://.github.io//`, for example your repository is at `https://github.com//`, then set `base` to `"//"`.
-
-2. Choose your preferred CI tools. Here we take [GitHub Actions](https://github.com/features/actions) as an example.
-
- Create `.github/workflows/docs.yml` to set up the workflow.
-
-::: details Click to expand sample config
-```yaml
-name: docs
-
-on:
- # trigger deployment on every push to main branch
- push:
- branches: [main]
- # trigger deployment manually
- workflow_dispatch:
-
-jobs:
- docs:
- runs-on: ubuntu-latest
-
- steps:
- - uses: actions/checkout@v2
- with:
- # fetch all commits to get last updated time or other git log info
- fetch-depth: 0
-
- - name: Setup Node.js
- uses: actions/setup-node@v1
- with:
- # choose node.js version to use
- node-version: '14'
-
- # cache node_modules
- - name: Cache dependencies
- uses: actions/cache@v2
- id: yarn-cache
- with:
- path: |
- **/node_modules
- key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }}
- restore-keys: |
- ${{ runner.os }}-yarn-
-
- # install dependencies if the cache did not hit
- - name: Install dependencies
- if: steps.yarn-cache.outputs.cache-hit != 'true'
- run: yarn --frozen-lockfile
-
- # run build script
- - name: Build VuePress site
- run: yarn docs:build
-
- # please check out the docs of the workflow for more details
- # @see https://github.com/crazy-max/ghaction-github-pages
- - name: Deploy to GitHub Pages
- uses: crazy-max/ghaction-github-pages@v2
- with:
- # deploy to gh-pages branch
- target_branch: gh-pages
- # deploy the default output dir of VuePress
- build_dir: docs/.vuepress/dist
-```
-:::
-
-::: tip
-Please refer to [GitHub Pages official guide](https://pages.github.com/) for more details.
-:::
-
-## GitLab Pages
-
-1. Set the correct [base](../reference/config.md#base) config.
-
- If you are deploying to `https://.gitlab.io/`, you can omit `base` as it defaults to `"/"`.
-
- If you are deploying to `https://.gitlab.io//`, for example your repository is at `https://gitlab.com//`, then set `base` to `"//"`.
-
-2. Create `.gitlab-ci.yml` to set up [GitLab CI](https://about.gitlab.com/stages-devops-lifecycle/continuous-integration/) workflow.
-
-::: details Click to expand sample config
-```yaml
-# choose a docker image to use
-image: node:14-buster
-
-pages:
- # trigger deployment on every push to main branch
- only:
- - main
-
- # cache node_modules
- cache:
- paths:
- - node_modules/
-
- # install dependencies and run build script
- script:
- - yarn --frozen-lockfile
- - yarn docs:build --dest public
-
- artifacts:
- paths:
- - public
-```
-:::
-
-::: tip
-Please refer to [GitLab Pages official guide](https://docs.gitlab.com/ce/user/project/pages/#getting-started) for more details.
-:::
-
-## Google Firebase
-
-1. Make sure you have [firebase-tools](https://www.npmjs.com/package/firebase-tools) installed.
-
-2. Create `firebase.json` and `.firebaserc` at the root of your project with the following content:
-
-`firebase.json`:
-
-```json
-{
- "hosting": {
- "public": "./docs/.vuepress/dist",
- "ignore": []
- }
-}
-```
-
-`.firebaserc`:
-
-```json
-{
- "projects": {
- "default": ""
- }
-}
-```
-
-3. After running `yarn docs:build`, deploy using the command `firebase deploy`.
-
-::: tip
-Please refer to [Firebase CLI official guide](https://firebase.google.com/docs/cli) for more details.
-:::
-
-## Heroku
-
-1. Install [Heroku CLI](https://devcenter.heroku.com/articles/heroku-cli).
-
-2. Create a Heroku account by [signing up](https://signup.heroku.com).
-
-3. Run `heroku login` and fill in your Heroku credentials:
-
-```bash
-heroku login
-```
-
-4. Create a file called `static.json` in the root of your project with the below content:
-
-`static.json`:
-
-```json
-{
- "root": "./docs/.vuepress/dist"
-}
-```
-
-This is the configuration of your site; read more at [heroku-buildpack-static](https://github.com/heroku/heroku-buildpack-static).
-
-## Netlify
-
-1. On [Netlify](https://netlify.com), set up a new project from GitHub with the following settings:
-
- - **Build Command:** `yarn docs:build`
- - **Publish directory:** `docs/.vuepress/dist`
-
-2. Set [Environment variables](https://docs.netlify.com/configure-builds/environment-variables) to choose node version:
-
- - `NODE_VERSION`: 14
-
-3. Hit the deploy button.
-
-## Vercel
-
-See [Creating and Deploying a VuePress App with Vercel](https://vercel.com/guides/deploying-vuepress-to-vercel).
diff --git a/docs/guide/directory-structure.md b/docs/guide/directory-structure.md
new file mode 100644
index 00000000..321ef685
--- /dev/null
+++ b/docs/guide/directory-structure.md
@@ -0,0 +1,67 @@
+# 目录结构
+
+在[快速上手](./getting-started.html)中,大家对框架应该有初步的印象,接下来我们了解下目录结构。Fes.js 遵循 `约定优于配置` 的原则,一个基础的 Fes.js 项目大致是这样的:
+```
+fes-template
+├── package.json
+├── tsconfig.json
+├── mock.js
+├── .fes.js
+├── .env
+├── dist
+├── public
+│ └── index.html
+└── src
+ ├── .fes
+ └── pages
+ │ └── index.vue
+ └── app.js
+```
+
+### 根目录
+
+#### package.json
+包含插件和插件集,以 `@fesjs/preset-`、`@fesjs/plugin-`、`fes-preset-` 和 `fes-plugin-` 开头的依赖会被自动注册为插件或插件集。
+
+#### tsconfig.json
+解决 `@fesjs/fes` 和使用 `@` 的 API 提示
+
+#### .fes.js
+配置文件,包含 Fes.js 内置功能和插件的配置。
+
+#### .env
+定义环境变量。
+
+比如 `.env` 文件内容如下:
+```
+PORT=8888
+FES_ENV=prod
+```
+等同于 node 端运行时,设置如下:
+```
+process.env.PORT = '8888';
+process.env.FES_ENV = 'prod';
+```
+
+#### mock.js
+mock 数据的配置文件。
+
+### dist 目录
+执行 `fes build` 后,产物默认会存放在这里。
+
+### public 目录
+此目录下所有文件会被 `copy` 到输出路径。
+
+#### index.html
+默认的 `html` 模板文件,如果删除此 `html` 则会使用内置的 `html` 模板文件。
+
+### src 目录
+
+#### .fes 目录
+临时文件目录,比如入口文件、路由等,都会被临时生成到这里。不要提交 `.fes` 目录到 `git` 仓库,他们会在 `fes dev` 和 `fes build` 时被删除并重新生成。
+
+#### pages 目录
+所有路由组件存放在这里。
+
+#### app.js
+运行时配置文件,可以在这里扩展运行时的能力,比如修改路由等。
\ No newline at end of file
diff --git a/docs/guide/env.md b/docs/guide/env.md
new file mode 100644
index 00000000..8dfa1ec6
--- /dev/null
+++ b/docs/guide/env.md
@@ -0,0 +1,109 @@
+# 环境变量
+
+## 设置环境变量
+
+### 执行命令时添加
+比如:
+```bash
+# OS X, Linux
+PORT=3000 umi dev
+
+# Windows (cmd.exe)
+set PORT=3000 && umi dev
+```
+如果要同时考虑 OS X 和 Windows,可借助三方工具 [cross-env](https://github.com/kentcdodds/cross-env)
+
+
+
+```bash
+yarn add cross-env --dev
+cross-env PORT=3000 umi dev
+```
+
+
+
+
+```bash
+npm i cross-env --save-dev
+cross-env PORT=3000 umi dev
+```
+
+
+
+
+### 在 .env 文件中定义
+Fes.js 中约定根目录下以 `.env` 开头的文件为环境变量配置文件。
+
+比如:
+```bash
+PORT=3000
+```
+然后执行
+```bash
+fes dev
+```
+会以 3000 端口启动 dev server。
+
+#### 本地临时配置
+可以新建 `.env.local`,这份配置会和 `.env` 做 `merge` 后形成最终配置。
+
+#### 多环境多份配置
+可以通过环境变量 `FES_ENV` 区分不同环境来指定配置,这时候必须在执行命令前添加 `FES_ENV` 保证执行加载环境变量配置文件逻辑前 `FES_ENV` 已设置。
+
+举个 🌰 :
+```bash
+FES_ENV=sit umi dev
+```
+如果存在 `.env.sit` 文件,则会将 `.env.sit` 的配置和 `.env` 做 `merge` 后形成最终配置。
+
+#### 配置优先级
+
+临时配置 > 环境配置 > .env
+
+::: tip
+如果两份配置中存在相同的项,则优先级高的会覆盖优先级低的。
+:::
+
+
+## 环境变量列表
+
+### FES_ENV
+指定当前的环境,不同环境各自的配置文件。
+
+### FES_PRESETS
+添加额外的插件集入口
+
+### FES_PLUGINS
+添加额外的插件入口
+
+### PORT
+`fes dev` 时服务指定的端口号,默认是 `8080`
+
+### HOST
+默认是 `localhost`。
+
+### HTTPS
+默认是 `false`。
+
+### WATCH
+设为 none 时不监听文件变更。比如:
+```
+WATCH=none fes dev
+```
+
+### BABEL_CACHE
+默认开启 Babel 编译缓存,值为 none 时禁用缓存。
+
+### ANALYZE
+用于分析 bundle 构成,默认关闭。
+
+比如:
+```
+ANALYZE=1 fes build
+```
+
+### ANALYZE_MODE
+默认是`server`
+
+### ANALYZE_PORT
+默认是`8888`
diff --git a/docs/guide/faq.md b/docs/guide/faq.md
new file mode 100644
index 00000000..307d253c
--- /dev/null
+++ b/docs/guide/faq.md
@@ -0,0 +1 @@
+# 常见问题
\ No newline at end of file
diff --git a/docs/guide/getting-started.md b/docs/guide/getting-started.md
index 1561404b..cfa0cdc0 100644
--- a/docs/guide/getting-started.md
+++ b/docs/guide/getting-started.md
@@ -1,29 +1,43 @@
-# Getting Started
-
-## Prerequisites
-
-- [Node.js v12+](https://nodejs.org/)
-- [Yarn v1 classic](https://classic.yarnpkg.com/en/) (Optional)
-
-## Manual Installation
-
-This section will help you build a basic VuePress documentation site from ground up. If you already have an existing project and would like to keep documentation inside the project, start from Step 3.
-
-- **Step 1**: Create and change into a new directory
+# 快速上手
+## 依赖环境
+首先得有 [Node.js](https://nodejs.org/),并确保 node 版本是 10.13 或以上。
```bash
-mkdir vuepress-starter
-cd vuepress-starter
+# 打印 node 版本
+node -v
+v10.13.0
+```
+推荐使用 yarn 管理 npm 依赖
+```bash
+# 全局安装 yarn
+npm i yarn -g
```
-- **Step 2**: Initialize your project
+## 安装模板
+这一章节会帮助你从头搭建一个简单的 Fes.js 前端应用。
+
+##### 步骤1 创建工作空间
+如果不存在,则创建
+```bash
+# 创建目录 workspace
+mkdir workspace
+# 进入目录 workspace
+cd workspace
+```
+如果已存在工作空间,则直接进入
+```bash
+# 进入目录 workspace
+cd workspace
+```
+
+##### 步骤2 创建模板
```bash
-git init
-yarn init
+# 创建模板
+yarn create @fesjs/fes-app myapp
```
@@ -31,20 +45,30 @@ yarn init
```bash
-git init
-npm init
+# 创建模板
+npx @fesjs/create-fes-app myapp
```
-- **Step 3**: Install VuePress locally
+如果项目目录 `workspace/myapp` 已经存在,则会提示目录已存在,你可以选择 `Overwrite` 删除目录后重新创建项目,也可以选择 `Merge` 使用模板文件覆盖当前目录文件。
+
+
+如果项目目录 `workspace/myapp` 不存在,你会被提示选取一个 template。你可以选默认适用于中后台前端应用的 `PC` 类型,也可以选适用于移动端的 `H5` 类型。
+
+
+
+##### 步骤3 安装依赖
```bash
-yarn add -D vuepress@next
+# 进入项目目录
+cd myapp
+# 安装依赖
+yarn
```
@@ -52,43 +76,31 @@ yarn add -D vuepress@next
```bash
-npm install -D vuepress@next
+# 进入项目目录
+cd myapp
+# 安装依赖
+npm i
```
-- **Step 4**: Add some [scripts](https://classic.yarnpkg.com/en/docs/package-json#toc-scripts) to `package.json`
-
-```json
-{
- "scripts": {
- "docs:dev": "vuepress dev docs",
- "docs:build": "vuepress build docs"
- }
-}
-```
-
-- **Step 5**: Add the default temp and cache directory to `.gitignore` file
-
-```bash
-echo 'node_modules\n.temp\n.cache' >> .gitignore
-```
-
-- **Step 6**: Create your first document
-
-```bash
-mkdir docs
-echo '# Hello VuePress' > docs/README.md
-```
-
-- **Step 7**: Serve the documentation site in the local server
-
+## 启动项目
```bash
-yarn docs:dev
+# 开发调试
+yarn dev
+
+yarn run v1.22.4
+$ fes dev
+Starting the development server http://localhost:8080 ...
+
+✔ Webpack
+ Compiled successfully in 15.91s
+
+ DONE Compiled successfully in 15917ms 11:17:08 AM
```
@@ -96,12 +108,85 @@ yarn docs:dev
```bash
-npm run docs:dev
+# 开发调试
+npm run dev
+
+
+> @fesjs/fes-template@2.0.0-alpha.1 dev /Users/harrywan/company/git/fes.js/packages/fes-template
+> fes dev
+
+Starting the development server http://localhost:8080 ...
+
+✔ Webpack
+ Compiled successfully in 3.66s
+
+ DONE Compiled successfully in 3662ms 11:17:46 AM
```
- VuePress will start a hot-reloading development server at [http://localhost:8080](http://localhost:8080). When you modify your markdown files, the content in the browser will be auto updated.
-By now, you should have a basic but functional VuePress documentation site. Next, learn about the basics of [configuration](./configuration.md) in VuePress.
+Fes.js 会在 [http://localhost:8080](http://localhost:8080) 启动一个热重载的开发服务器。当你修改你的 .vue 文件时,浏览器中的内容也会自动更新。
+
+
+
+
+## 部署发布
+
+### 构建
+
+
+
+```bash
+# 构建
+yarn build
+
+yarn run v1.22.4
+$ fes build
+
+✔ Webpack
+ Compiled successfully in 45.37s
+
+✨ Done in 48.87s.
+```
+
+
+
+
+
+```bash
+# 构建
+npm run build
+
+> @fesjs/fes-template@2.0.0-alpha.1 build /Users/harrywan/company/git/fes.js/packages/fes-template
+> fes build
+
+✔ Webpack
+ Compiled successfully in 45.37s
+```
+
+
+
+
+构建产物默认生成到 ./dist 下,然后通过 tree 命令查看。
+```base
+tree ./dist
+
+dist
+├── chunk-vendors.27cd4686.js
+├── chunk-vendors.a5f5de67.css
+├── index.11411d43.css
+├── index.d72f1ba2.js
+├── index.html
+├── logo.png
+└── static
+ └── logo.0f85bba0.png
+```
+
+### 本地验证
+发布之前,可以通过 [serve](https://github.com/vercel/serve) 做本地验证,验证结果应该跟执行 `dev` 的结果一样。
+
+
+### 部署
+本地验证完,就可以部署了。你需要把 dist 目录部署到服务器上。
\ No newline at end of file
diff --git a/docs/guide/i18n.md b/docs/guide/i18n.md
deleted file mode 100644
index 18ed8f86..00000000
--- a/docs/guide/i18n.md
+++ /dev/null
@@ -1,70 +0,0 @@
-# I18n
-
-## Site I18n Config
-
-To take advantage of multi-language support in VuePress, you first need to use the following file and directory structure:
-
-```
-docs
-├─ README.md
-├─ foo.md
-├─ nested
-│ └─ README.md
-└─ zh
- ├─ README.md
- ├─ foo.md
- └─ nested
- └─ README.md
-```
-
-Then, specify the `locales` option in your [config file](./configuration.md#config-file):
-
-```js
-module.exports = {
- locales: {
- // The key is the path for the locale to be nested under.
- // As a special case, the default locale can use '/' as its path.
- '/': {
- lang: 'en-US',
- title: 'VuePress',
- description: 'Vue-powered Static Site Generator',
- },
- '/zh/': {
- lang: 'zh-CN',
- title: 'VuePress',
- description: 'Vue 驱动的静态网站生成器',
- },
- },
-}
-```
-
-If a locale does not have a `lang`, `title`, `description` or `head`, VuePress will fallback to the root-level values. You can omit the root level config as long as they are provided in each locale.
-
-::: tip
-Config reference: [locales](../reference/config.md#locales)
-:::
-
-## Theme I18n Config
-
-VuePress does not restrict how themes provide multi-language support, so each theme may have different way to handle i18n, and some themes may not provide multi-language support at all. You'd better refer to the theme documentation for detailed guide.
-
-If you are using default theme, the multi-language support is the same with above:
-
-```js
-module.exports = {
- themeConfig: {
- locales: {
- '/': {
- selectLanguageName: 'English',
- },
- '/zh/': {
- selectLanguageName: '简体中文',
- },
- },
- },
-}
-```
-
-::: tip
-Config reference: [Default Theme > locales](../reference/default-theme/config.md#locales)
-:::
diff --git a/docs/guide/markdown.md b/docs/guide/markdown.md
deleted file mode 100644
index b67f60be..00000000
--- a/docs/guide/markdown.md
+++ /dev/null
@@ -1,366 +0,0 @@
-# Markdown
-
-Make sure you have known Markdown well before reading this section. If not, please learn some [Markdown tutorials](https://commonmark.org/help/) first.
-
-## Syntax Extensions
-
-The Markdown content in VuePress will be parsed by [markdown-it](https://github.com/markdown-it/markdown-it), which supports [syntax extensions](https://github.com/markdown-it/markdown-it#syntax-extensions) via markdown-it plugins.
-
-This section will introduce built-in Markdown syntax extensions of VuePress.
-
-You can also configure those built-in extensions, load more markdown-it plugins and implement your own extensions via [markdown](../reference/config.md#markdown) option and [extendsMarkdown](../reference/plugin-api.md#extendsmarkdown) option.
-
-### Embedded
-
-Embedded by markdown-it:
-
-- [Tables](https://help.github.com/articles/organizing-information-with-tables/) (GFM)
-- [Strikethrough](https://help.github.com/articles/basic-writing-and-formatting-syntax/#styling-text) (GFM)
-
-### Header Anchors
-
-You might have noticed that, a `#` anchor is displayed when you hover the mouse on the headers of each section. By clicking the `#` anchor, you can jump to the section directly.
-
-::: tip
-This header anchors extension is supported by [markdown-it-anchor](https://github.com/valeriangalliat/markdown-it-anchor).
-
-Config reference: [markdown.anchor](../reference/config.md#markdown-anchor)
-:::
-
-### Links
-
-When using Markdown [link syntax](https://spec.commonmark.org/0.29/#link-reference-definitions), VuePress will implement some conversions for you.
-
-Take our documentation source files as an example:
-
-```bash
-└─ docs
- ├─ guide
- │ ├─ getting-started.md
- │ ├─ markdown.md # <- Here we are
- │ └─ README.md
- ├─ reference
- │ └─ config.md
- └─ README.md
-```
-
-**Raw Markdown**
-
-```md
-[Home](/README.md)
-[Guide](/guide/)
-[Getting Started](./getting-started.md)
-[markdown.links](../reference/config.md#links)
-[GitHub](https://github.com)
-```
-
-**Converted to**
-
-```vue
-Home
-Guide
-Getting Started
-markdown.links
-GitHub
-```
-
-**Rendered as**
-
-[Home](/README.md)
-[Guide](/guide/)
-[Getting Started](./getting-started.md)
-[markdown.links](../reference/config.md#links)
-[GitHub](https://github.com)
-
-**Explanation**
-
-- Internal links will be converted to `` for SPA navigation.
-- Internal links to `.md` files will be converted to the [page route path](./page.md#routing), and both absolute path and relative path are supported.
-- External links will get `target="_blank" rel="noopener noreferrer"` attrs and a indicator.
-
-::: tip
-This links extension is supported by our built-in plugin.
-
-Config reference: [markdown.links](../reference/config.md#markdown-links)
-
-Also see: [Built-in Components > OutboundLink](../reference/components.md#outboundlink)
-:::
-
-### Emoji :tada:
-
-You can add emoji to your Markdown content by typing `:EMOJICODE:`.
-
-For a full list of available emoji and codes, check out [emoji-cheat-sheet.com](https://emoji-cheat-sheet.com/).
-
-**Input**
-
-```md
-VuePress 2 is out :tada: !
-```
-
-**Output**
-
-VuePress 2 is out :tada: !
-
-::: tip
-This emoji extension is supported by [markdown-it-emoji](https://github.com/markdown-it/markdown-it-emoji).
-
-Config reference: [markdown.emoji](../reference/config.md#markdown-emoji)
-:::
-
-### Table of Contents
-
-If you want to put the table of contents (TOC) of your current page inside your Markdown content, you can use the `[[toc]]` syntax.
-
-**Input**
-
-```md
-[[toc]]
-```
-
-**Output**
-
-[[toc]]
-
-The headers in TOC will link to the corresponding [header anchors](#header-anchors), so TOC won't work well if you disable header anchors.
-
-::: tip
-This toc extension is supported by our built-in plugin, which is forked and modified from [markdown-it-toc-done-right](https://github.com/nagaozen/markdown-it-toc-done-right).
-
-Config reference: [markdown.toc](../reference/config.md#markdown-toc)
-:::
-
-### Code Blocks
-
-Following code blocks extensions are implemented during markdown parsing in Node side. That means, the code blocks won't be processed in client side.
-
-If you want to implement client-side syntax highlighting via [prism.js](https://prismjs.com/#basic-usage) or [highlight.js](https://highlightjs.org/), you could disable our code blocks extensions, and introduce your library manually in client side.
-
-#### Syntax Highlighting
-
-VuePress uses [Prism](https://prismjs.com/) to highlight language syntax in Markdown code blocks, using coloured text.
-
-Prism supports a wide variety of programming languages. For a full list of available languages, check out [Prism supported languages](https://prismjs.com/#supported-languages).
-
-You can add an optional language identifier to enable syntax highlighting in your fenced code blocks:
-
-**Input**
-
-````md
-```ts
-import type { UserConfig } from '@vuepress/cli'
-
-export const config: UserConfig = {
- title: 'Hello, VuePress',
-}
-```
-````
-
-**Output**
-
-```ts
-import type { UserConfig } from '@vuepress/cli'
-
-export const config: UserConfig = {
- title: 'Hello, VuePress',
-}
-```
-
-::: tip
-This syntax highlighting extension is supported by our built-in plugin.
-
-Config reference: [markdown.code.highlight](../reference/config.md#markdown-code-highlight)
-:::
-
-#### Line Highlighting
-
-You can highlight specified lines of your code blocks by adding line ranges mark in your fenced code blocks:
-
-**Input**
-
-````md
-```ts{1,6-8}
-import type { UserConfig } from '@vuepress/cli'
-
-export const config: UserConfig = {
- title: 'Hello, VuePress',
-
- themeConfig: {
- logo: 'https://vuejs.org/images/logo.png',
- },
-}
-```
-````
-
-**Output**
-
-```ts{1,6-8}
-import type { UserConfig } from '@vuepress/cli'
-
-export const config: UserConfig = {
- title: 'Hello, VuePress',
-
- themeConfig: {
- logo: 'https://vuejs.org/images/logo.png',
- },
-}
-```
-
-Examples for line ranges mark:
-
-- Line ranges: `{5-8}`
-- Multiple single lines: `{4,7,9}`
-- Combined: `{4,7-13,16,23-27,40}`
-
-::: tip
-This line highlighting extension is supported by our built-in plugin, which is forked and modified from [markdown-it-highlight-lines](https://github.com/egoist/markdown-it-highlight-lines).
-
-Config reference: [markdown.code.highlightLines](../reference/config.md#markdown-code-highlightlines)
-:::
-
-#### Line Numbers
-
-You must have noticed that the number of lines is displayed on the left side of code blocks. This is enabled by default and you can disable it in config.
-
-You can add `:line-numbers` / `:no-line-numbers` mark in your fenced code blocks to override the value set in config.
-
-**Input**
-
-````md
-```ts
-// line-numbers is enabled by default
-const line2 = 'This is line 2'
-const line3 = 'This is line 3'
-```
-
-```ts:no-line-numbers
-// line-numbers is disabled
-const line2 = 'This is line 2'
-const line3 = 'This is line 3'
-```
-````
-
-**Output**
-
-```ts
-// line-numbers is enabled by default
-const line2 = 'This is line 2'
-const line3 = 'This is line 3'
-```
-
-```ts:no-line-numbers
-// line-numbers is disabled
-const line2 = 'This is line 2'
-const line3 = 'This is line 3'
-```
-
-::: tip
-This line numbers extension is supported by our built-in plugin.
-
-Config reference: [markdown.code.lineNumbers](../reference/config.md#markdown-code-linenumbers)
-:::
-
-#### Wrap with v-pre
-
-As [template syntax is allowed in Markdown](#template-syntax), it would also work in code blocks, too.
-
-To avoid your code blocks being compiled by Vue, VuePress will add [v-pre](https://v3.vuejs.org/api/directives.html#v-pre) directive to your code blocks by default, which can be disabled in config.
-
-You can add `:v-pre` / `:no-v-pre` mark in your fenced code blocks to override the value set in config.
-
-::: warning
-The template syntax characters, for example, the "Mustache" syntax (double curly braces) might be parsed by the syntax highlighter. Thus, as the following example, `:no-v-pre` might not work well in some languages.
-
-If you want to make Vue syntax work in those languages anyway, try to disable the default syntax highlighting and implement your own syntax highlighting in client side.
-:::
-
-**Input**
-
-````md
-```md
-
-1 + 2 + 3 = {{ 1 + 2 + 3 }}
-```
-
-```md:no-v-pre
-
-1 + 2 + 3 = {{ 1 + 2 + 3 }}
-```
-
-```js:no-v-pre
-// This won't be compiled correctly because of js syntax highlighting
-const onePlusTwoPlusThree = {{ 1 + 2 + 3 }}
-```
-````
-
-**Output**
-
-```md
-
-1 + 2 + 3 = {{ 1 + 2 + 3 }}
-```
-
-```md:no-v-pre
-
-1 + 2 + 3 = {{ 1 + 2 + 3 }}
-```
-
-```js:no-v-pre
-// This won't be compiled correctly because of js syntax highlighting
-const onePlusTwoPlusThree = {{ 1 + 2 + 3 }}
-```
-
-::: tip
-This v-pre extension is supported by our built-in plugin.
-
-Config reference: [markdown.code.vPre](../reference/config.md#markdown-vpre)
-:::
-
-## Using Vue in Markdown
-
-This section will introduce some basic usage of Vue in Markdown.
-
-Check out [Advanced > Markdown and Vue SFC](./advanced/markdown.md) for more details.
-
-### Template Syntax
-
-As we know:
-
-- HTML is allowed in Markdown.
-- Vue template syntax is compatible with HTML.
-
-That means, [Vue template syntax](https://v3.vuejs.org/guide/template-syntax.html) is allowed in Markdown.
-
-**Input**
-
-```md
-One plus one equals: {{ 1 + 1 }}
-
- span: {{ i }}
-```
-
-**Output**
-
-One plus one equals: {{ 1 + 1 }}
-
- span: {{ i }}
-
-### Components
-
-You can use Vue components directly in Markdown.
-
-**Input**
-
-```md
-This is default theme built-in `` component
-```
-
-**Output**
-
-This is default theme built-in `` component
-
-::: tip
-Check out the [Built-in Components](../reference/components.md) for a full list of built-in components.
-
-Check out the [Default Theme > Built-in Components](../reference/default-theme/components.md) for a full list of default theme built-in components.
-:::
diff --git a/docs/guide/mock.md b/docs/guide/mock.md
new file mode 100644
index 00000000..a2147e6c
--- /dev/null
+++ b/docs/guide/mock.md
@@ -0,0 +1,183 @@
+# Mock 数据
+
+Mock 数据是前端开发过程中必不可少的一环,是分离前后端开发的关键链路。通过预先跟服务器端约定好的接口,模拟请求数据甚至逻辑,能够让前端开发独立自主,不会被服务端的开发所阻塞。
+
+## 约定式 Mock 文件
+
+Fes.js 约定 `src/mock.js` 为 mock 文件。
+
+比如:
+```
+.
+├── mock.js
+└── src
+ └── pages
+ └── index.vue
+```
+
+## 编写 Mock 文件
+
+可以参考如下例子:
+``` js
+module.exports = function ({ cgiMock, mockjs, utils }) {
+ const { Random } = mockjs;
+
+ // 测试 proxy 与 mock 用例集合
+ cgiMock('/movie/in_theaters_mock', (req, res) => {
+ res.send(JSON.stringify({
+ code: '0',
+ msg: '',
+ result: {
+ text: 'movie: movie/in_theaters_mock ~~~~~'
+ }
+ }));
+ });
+ cgiMock('/movie/test_mock', (req, res) => {
+ res.send(JSON.stringify({
+ code: '0',
+ msg: '',
+ result: {
+ text: 'mock: movie/test_mock'
+ }
+ }));
+ });
+
+ // 测试用例: mock.js change,重现请求,需要能拉最新的数据
+ cgiMock('/watchtest', (req, res) => {
+ res.send(JSON.stringify({
+ code: '0',
+ msg: '',
+ result: {
+ text: '通过 register 测试 mock watch: 初始状态'
+ }
+ }));
+ });
+
+ // 返回一个数字
+ // cgiMock('/number', 666);
+ cgiMock('/number', 999);
+
+ // 返回一个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('/random', mockjs.mock({
+ 'string|1-10': '★'
+ }));
+
+ // 正则匹配url, 返回一个字符串
+ cgiMock(/\/abc|\/xyz/, 'regexp test!');
+
+ // option.result 参数如果是一个函数, 可以实现自定义返回内容, 接收的参数是是经过 express 封装的 req 和 res 对象.
+ cgiMock(/\/function$/, (req, res) => {
+ res.send('function test');
+ });
+
+ // 返回文本 readFileSync
+ cgiMock('/file', utils.file('./package.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
+ }
+ ]
+ });
+
+ // 携带参数的请求
+ cgiMock('/v2/audit/list', (req, res) => {
+ const {
+ currentPage, pageSize, isAudited
+ } = req.body;
+ res.send({
+ code: '0',
+ msg: '',
+ data: {
+ currentPage,
+ pageSize,
+ totalPage: 2,
+ totalCount: 12,
+ pageData: Array.from({ length: pageSize }, () => ({
+ title: Random.title(),
+ authorName: Random.cname(),
+ authorId: Random.name(),
+ createTime: Date.now(),
+ updateTime: Date.now(),
+ readCount: Random.integer(60, 1000),
+ favoriteCount: Random.integer(1, 50),
+ postId: '12323',
+ serviceTag: '业务类型',
+ productTag: '产品类型',
+ requestTag: '需求类型',
+ handleTag: '已采纳',
+ postType: 'voice',
+ postStatus: isAudited ? 'pass' : 'auditing',
+ auditStatus: 'audit1'
+ }))
+ }
+ });
+ });
+
+ // multipart/form-data 类型
+ cgiMock('/v2/upload', (req, res) => {
+ res.send({
+ code: '0',
+ msg: '文件上传成功'
+ });
+ });
+};
+```
+
+### cgiMock 参数
+创建一个 mock 接口,参数非常灵活,参考上面的 demo 即可。
+
+
+### mockjs 参数
+[Mock.js](http://mockjs.com/) 是常用的辅助生成模拟数据的三方库,借助他可以提升我们的 mock 数据能力。
+
+比如:
+```js
+module.exports = function ({ cgiMock, mockjs, utils }) {
+ cgiMock('/random', mockjs.mock({
+ 'string|1-10': '★'
+ }));
+}
+```
+
+### utils 参数
+工具函数:
+- utils.file(path),从项目根目录根据path寻找文件,返回文件流。
+
+## 配置 Mock
+详见配置 #mock。
+
+## 关闭 Mock
+可以通过配置关闭。
+```js
+export default {
+ mock: false,
+};
+```
diff --git a/docs/guide/page.md b/docs/guide/page.md
deleted file mode 100644
index b67d9918..00000000
--- a/docs/guide/page.md
+++ /dev/null
@@ -1,55 +0,0 @@
-# Page
-
-VuePress is markdown-centered. Each markdown file inside your project is a standalone page.
-
-## Routing
-
-By default, the route path of a page is determined by the relative path of your markdown file.
-
-Assuming this is the directory structure of your markdown files:
-
-```
-└─ docs
- ├─ guide
- │ ├─ getting-started.md
- │ └─ README.md
- ├─ contributing.md
- └─ README.md
-```
-
-Take the `docs` directory as your [sourceDir](../reference/cli.md), e.g. your are running `vuepress dev docs` command. Then the route paths of your markdown files would be:
-
-| Relative Path | Route Path |
-|--------------------|----------------------|
-| `/README.md` | `/` |
-| `/contributing.md` | `/contributing.html` |
-| `/guide/README.md` | `/guide/` |
-| `/guide/page.md` | `/guide/page.html` |
-
-## Frontmatter
-
-A markdown file could contain a [YAML](https://yaml.org/) frontmatter. The frontmatter must be at the top of the Markdown file and must be wrapped with a couple of triple-dashed lines. Here is a basic example:
-
-```md
----
-lang: en-US
-title: Title of this page
-description: Description of this page
----
-```
-
-You must have noticed that those fields are similar with the [Site Config](./configuration.md#site-config) of in the [Config File](./configuration.md#config-file). You can override `lang`, `title`, `description`, etc., of current page via frontmatter. So you can take frontmatter as page scope config.
-
-Also, VuePress has built-in support for some frontmatter fields, and your theme may have its own special frontmatter, too.
-
-::: tip
-Check out the [Frontmatter Reference](../reference/frontmatter.md) for a full list of VuePress built-in frontmatter.
-
-Check out the [Default Theme > Frontmatter Reference](../reference/default-theme/frontmatter.md) for the frontmatter of default theme.
-:::
-
-## Content
-
-The main content of your page is written in Markdown. VuePress will firstly transform your Markdown to HTML code, then treat the HTML code as `` of Vue SFC.
-
-With the power of [markdown-it](https://github.com/markdown-it/markdown-it) and Vue template syntax, the basic Markdown can be extended a lot. Next, check out the [Markdown](./markdown.md) guide for all the extensions of Markdown in VuePress.
diff --git a/docs/guide/plugin.md b/docs/guide/plugin.md
index c52db6e4..b81a3570 100644
--- a/docs/guide/plugin.md
+++ b/docs/guide/plugin.md
@@ -1,45 +1,71 @@
-# Plugin
+# 插件
-With the help of [Plugin API](../references/plugin-api.md), VuePress plugin can provide different features for you.
+## 插件的 id 和 key
+每个插件都会对应一个 `id` 和一个 `key`,**`id` 是路径的简写,`key` 是进一步简化后用于配置的唯一值**。
-## Community Plugin
+比如插件 `/node_modules/@fesjs/plugin-foo/index.js`,通常来说,其 `id` 为 `@fesjs/plugin-foo`,`key` 为 `foo`。
-Community users have created lots of plugins and published them to [NPM](https://www.npmjs.com/search?q=keywords:vuepress-plugin). VuePress team also maintains some official plugins under the [@vuepress](https://www.npmjs.com/search?q=%40vuepress%20keywords%3Aplugin) scope. You should check the plugin's own documentation for detailed guide.
+::: tip
+id 一般用不上,对于普通开发者 key 用来配置插件,而插件开发者可以使用 key 判断是否安装某个插件。
+:::
-In general, you need to specify the name of the plugin to use in [plugins](../reference/plugin-api.md#plugins) option:
+## 启动插件
+有多种方式引入插件
-```js
-module.exports = {
- plugins: [
- 'foo',
- ['bar', { /* options */ }]
- ],
+### package.json 依赖
+Fes.js 会自动检测 `dependencies` 和 `devDependencies` 里的 fes 插件,比如:
+```json
+{
+ "dependencies": {
+ "@fesjs/plugin-request": "^2.0.0"
+ }
}
```
+那么 `@fesjs/plugin-request` 会自动被注册,无需在配置里重复声明。
-You can use either plugin name or its shorthand:
-
-| Plugin Name | Shorthand |
-|---------------------------|---------------------|
-| `vuepress-plugin-foo` | `foo` |
-| `@org/vuepress-plugin-bar`| `@org/bar` |
-| `@vuepress/plugin-foobar` | `@vuepress/foobar` |
-
-## Local Plugin
-
-If you want to use your own plugin but don't want to publish it, you can create a local plugin.
-
-It is recommended to use the [Config File](./configuration.md#config-file) directly as a plugin, because [almost all of the Plugin APIs are available](../reference/config.md#plugin-api), which would be more convenient in most cases.
-
-But if you have too many things to do in your config file, it's better to extract them into separate plugins, and use them by setting the absolute path to them or requiring them:
-
+### 配置
+在配置里可通过 `presets` 和 `plugins` 配置插件,比如:
```js
-module.exports = {
- plugins: [
- '/path/to/your-plugin.js',
- require('./another-plugin'),
- ],
+export default {
+ presets: ['./preset', 'foo/presets'],
+ plugins: ['./plugin'],
}
```
+通常用于几种情况:
-You can refer to [Advanced > Writing a Plugin](./advanced/plugin.md) for how to write your own plugin.
+1. 项目相对路径的插件
+2. 非 npm 包入口文件的插件
+
+::: warning
+请不要配置 npm 包的插件,否则会报重复注册的错误
+:::
+
+### 环境变量
+还可通过环境变量 `FES_PRESETS` 和 `FES_PLUGINS` 注册额外插件。
+
+比如:
+```bash
+FES_PRESETS=/a/b/preset.js fes dev
+```
+
+## 禁用插件
+
+通过配置插件的 `key` 为 `false`,比如:
+```js
+export default {
+ mock: false,
+}
+```
+Mock 插件的 `key` 是 `mock`,我们在配置文件中配置 `mock` 为 `false`,则会禁用 Mock 插件及其功能。
+
+## 配置插件
+
+通过插件的 `key` 来配置插件,比如:
+```js
+export default {
+ mock: {
+ prefix: '/v2'
+ }
+}
+```
+这里的 `mock` 是 Mock插件 的 key。
\ No newline at end of file
diff --git a/docs/guide/route.md b/docs/guide/route.md
new file mode 100644
index 00000000..d684ca77
--- /dev/null
+++ b/docs/guide/route.md
@@ -0,0 +1,259 @@
+# 路由
+
+像 Vue 、React 这类框架是用组件化搭建页面,路由解决的是路径到组件的匹配问题。Fes.js 基于 `Vue Router` 实现的路由,想了解更多的同学可以看看[官方文档](https://next.router.vuejs.org/)。
+
+## 路由配置
+
+在配置文件 `.fes.js`中通过 `router` 进行配置。
+```js
+export default {
+ router: {
+ routes: [],
+ mode: 'hash'
+ }
+}
+```
+
+### routes
+`routes` 是配置添加到路由的初始路由列表,格式为路由信息的数组。具体使用参考 [Vue Router 文档](https://next.router.vuejs.org/zh/guide/) 中关于路由配置、路由匹配相关内容。
+
+
+### mode
+创建历史记录的类型:
+- **h5**,对应 [createWebHistory](https://next.router.vuejs.org/zh/api/#createwebhistory)
+- **hash**,对应 [createWebHashHistory](https://next.router.vuejs.org/zh/api/#createWebHashHistory)
+- **memory**,对应 [createMemoryHistory](https://next.router.vuejs.org/zh/api/#createWebHashHistory)
+
+## 约定式路由
+约定式路由也叫文件路由,就是不需要手写配置,文件系统即路由,通过目录和文件及其命名分析出路由配置。
+
+### 约定规范
+比如以下文件结构:
+```
+pages
+├── index.vue # 根路由页面 路径为 /
+├── *.vue # 模糊匹配 路径为 *
+├── a.vue # 路径 /a
+├── b # 文件夹b
+│ ├── index.vue # 路径 /b
+│ ├── @id.vue # 动态路由 /b/:id
+│ ├── c.vue # 路径 /b/c
+│ └── layout.vue # /b 路径下所有页面公共的布局组件
+└── layout.vue # 根路由下所有页面共用的布局组件
+```
+编译后会得到以下路由配置:
+```js
+[
+ {
+ "path": "/",
+ "component": require('@/pages/layout').default,
+ "count": 5,
+ "children": [
+ {
+ "path": "/a",
+ "component": require('@/pages/a').default,
+ "name": "a",
+ "meta": {},
+ "count": 7
+ },
+ {
+ "path": "/b",
+ "component": require('@/pages/b/layout').default,
+ "count": 7,
+ "children": [
+ {
+ "path": "/b/c",
+ "component": require('@/pages/b/c').default,
+ "name": "b_c",
+ "meta": {},
+ "count": 14
+ },
+ {
+ "path": "/b/:id",
+ "component": require('@/pages/b/@id').default,
+ "name": "b__id",
+ "meta": {},
+ "count": 13
+ },
+ {
+ "path": "/b",
+ "component": require('@/pages/b/index').default,
+ "name": "b_index",
+ "meta": {},
+ "count": 7
+ }
+ ]
+ },
+ {
+ "path": "/",
+ "component": require('@/pages/index').default,
+ "name": "index",
+ "meta": {},
+ "count": 5
+ },
+ {
+ "path": "/:pathMatch(.*)",
+ "component": require('@/pages/*').default,
+ "name": "FUZZYMATCH",
+ "meta": {},
+ "count": 3
+ }
+ ]
+ }
+]
+```
+
+**需要注意的是,满足以下任意规则的文件不会被注册为路由**:
+- 不是 `.vue` 文件
+- `components` 目录中的文件
+
+
+### 动态路由
+Fes.js 里约定以 `@` 开头的文件或文件夹映射为动态路由。
+比如:
+
+- `src/pages/users/@id.vue` 会成为 `/users/:id`
+- `src/pages/users/@id/settings.vue` 会成为 `/users/:id/settings`
+
+### 嵌套路由
+Fes.js 里约定目录下有 `layout.vue` 时会生成嵌套路由,以 `layout.vue` 为该目录的公共父组件,`layout.vue` 中必须实现 `RouterView`
+
+比如以下目录结构:
+```
+pages
+└── users
+ ├── layout.vue
+ ├── index.vue
+ └── list.vue
+```
+会生成路由:
+```js
+[
+ {
+ path: '/users', component: require('@/pages/users/layout').default,
+ children: [
+ { path: '/users', component: require('@/pages/users/index').default },
+ { path: '/users/list', component: require('@/pages/users/list').default },
+ ]
+ }
+]
+```
+
+### 模糊匹配
+Fes.js 下约定文件名为 `*` 的路由是模糊匹配路由,可以用此特性实现 [404 路由](https://next.router.vuejs.org/zh/guide/essentials/dynamic-matching.html#%E6%8D%95%E8%8E%B7%E6%89%80%E6%9C%89%E8%B7%AF%E7%94%B1%E6%88%96-404-not-found-%E8%B7%AF%E7%94%B1)。
+
+比如以下目录结构:
+
+```
+pages
+├── index.vue # 根路由页面 路径为 /
+└── *.vue # 模糊匹配 路径为 *
+```
+会生成路由:
+```js
+[
+ {
+ path: '/', component: require('@/pages/index').default, count: 5
+ },
+ {
+ path: '/:pathMatch(.*)', component: require('@/pages/**').default, count: 3
+ }
+]
+```
+这样,如果访问 `/foo`,`/` 不能匹配,会 fallback 到 `*` 路由,通过 `src/pages/*.vue` 进行渲染。
+
+### 扩展路由元信息
+我们在定义路由时可以配置`meta`字段,用来记录一些跟路由相关的信息:
+```js
+const router = new VueRouter({
+ routes: [
+ {
+ path: '/foo',
+ component: Foo,
+ children: [
+ {
+ path: 'bar',
+ component: Bar,
+ // a meta field
+ meta: { requiresAuth: true }
+ }
+ ]
+ }
+ ]
+})
+```
+在 Fes.js 里约定在 `.vue` 文件中的 `config` 为 `meta` 配置。如果 `pages/a.vue` 中有如下配置:
+```vue
+
+{
+ "name": "store",
+ "title": "vuex测试"
+}
+
+```
+则编译后的路由配置为:
+```js{5-8}
+[
+ {
+ path: '/a',
+ component: require('@/pages/a').default,
+ meta: {
+ "name": "store",
+ "title": "vuex测试"
+ }
+ },
+]
+```
+
+### 智能路由
+可以看到,编译后路由都会有 `count` 属性,这是我们根据精准匹配优先算法原则设计出路由排名算法,对匹配到的路由打分:
+- 路由的路径每个子项得到4分
+- 子项为静态细分(`/list`)再加3分
+- 子项为动态细分(`/:orderId`)再加2分
+- 根段(`/`)再1分
+- 通配符(`*`)匹配到的减去1分
+
+当我们跳转路由时,如果 URL 匹配到多个路由,则选择分数最高的路由。
+
+## 路由跳转
+想学习更多,可以查看 [Vue Router 官方文档](https://next.router.vuejs.org/zh/guide/essentials/navigation.html#%E6%9B%BF%E6%8D%A2%E5%BD%93%E5%89%8D%E4%BD%8D%E7%BD%AE)。
+
+### 声明式
+```vue
+
+ Home
+
+```
+
+### 命令式
+页面跳转 API 由 `router` 实例提供,查看 [Vue Rouer 文档](https://next.router.vuejs.org/zh/api/#router-%E6%96%B9%E6%B3%95)了解更多。
+
+```js
+import { useRouter } from '@fesjs/fes';
+
+export default {
+ setup(){
+ const router = useRouter();
+ // 这三种形式是等价的
+ router.push('/users/posva#bio')
+ router.push({ path: '/users/posva', hash: '#bio' })
+ router.push({ name: 'users', params: { username: 'posva' }, hash: '#bio' })
+ // 只改变 hash
+ router.push({ hash: '#bio' })
+ // 只改变 query
+ router.push({ query: { page: '2' } })
+ // 只改变 param
+ router.push({ params: { username: 'jolyne' } })
+
+ // 跳转到上一个路由
+ router.goBack();
+
+ // 跳转到前一个历史记录
+ router.go(1);
+
+ // 替换历史堆栈中的记录
+ router.replace('/new');
+ }
+}
+
+```
\ No newline at end of file
diff --git a/docs/guide/template.md b/docs/guide/template.md
new file mode 100644
index 00000000..b3f77067
--- /dev/null
+++ b/docs/guide/template.md
@@ -0,0 +1,53 @@
+# HTML模板
+
+Fes.js 基于 [html-webpack-plugin](https://github.com/jantimon/html-webpack-plugin) 实现的模板功能,默认 HTML模板 是:
+```html
+
+
+
+
+
+
+ <%= htmlWebpackPlugin.options.title %>
+
+
+
+
+
+```
+
+
+## 替换模板
+在 `src/public` 文件夹中创建`index.html`,Fes.js 约定如果这个文件存在,则会替换默认模板。
+
+## 配置模板
+
+### 配置
+在配置文件(`.fes.js`)中配置 `html`,把配置的对象作为参数传入 `html-webpack-plugin` 实例。
+
+举个 :chestnut: :
+```js
+export default {
+ html: {
+ title: '海贼王'
+ }
+}
+```
+页面的 title 会设置成'海贼王'。
+
+### 手动
+当然我们也可以手动编写模板,在模板中添加`link`、`link`、`meta`等标签。在我们手动配置模板时,有时候需要用到一些环境变量,模板里可以获取到的变量如下:
+
+- **htmlWebpackPlugin**,特定于此插件的数据
+- **webpackConfig**,用于此编译的webpack配置。例如,它可用于获取publicPath(webpackConfig.output.publicPath)。
+- **compilation**,webpack编译对象。例如,可以使用它来获取已处理资产的内容,并将其直接内联到页面中compilation.assets[...].source()
+
+举个 🌰 :
+```html
+
+```
+
+除上述 `html-webpack-plugin` 三点之外,Fes.js 还把 `process.env` 中所有环境变量都添加到了模板作用域内:
+- `NODE_ENV`
+- `FES_ENV`
+- `.env` 文件中定义的以 `FES_APP_` 开头的变量
\ No newline at end of file
diff --git a/docs/guide/theme.md b/docs/guide/theme.md
deleted file mode 100644
index 9110fd4f..00000000
--- a/docs/guide/theme.md
+++ /dev/null
@@ -1,52 +0,0 @@
-# Theme
-
-VuePress theme can provide layouts, styles and many other features for you, helping you to focus on writing Markdown content.
-
-VuePress has a default theme out of the box, which is applied to our documentation site you are currently browsing. The default theme provides basic but useful features for documentation site, you can check out [Default Theme Config Reference](../reference/default-theme/config.md) for a full list of config.
-
-However, you might think it is not good enough. Or, you want to build a different type of site, for example, a blog, instead of a documentation. Then, you can try to [use a community theme](#community-theme) or [create a local theme](#local-theme).
-
-## Community Theme
-
-Community users have created lots of theme and published them to [NPM](https://www.npmjs.com/search?q=keywords:vuepress-theme). You should check the theme's own documentation for detailed guide.
-
-In general, you need to specify the name of the theme to use in [theme](../reference/config.md#theme) option:
-
-```js
-module.exports = {
- theme: 'foo',
-}
-```
-
-You can use either theme name or its shorthand:
-
-| Theme Name | Shorthand |
-|---------------------------|---------------------|
-| `vuepress-theme-foo` | `foo` |
-| `@org/vuepress-theme-bar` | `@org/bar` |
-| `@vuepress/theme-default` | `@vuepress/default` |
-
-## Local Theme
-
-If you want to use your own custom theme but don't want to publish it, you can create a local theme.
-
-First, create the local theme directory, typically `.vuepress/theme` :
-
-```
-└─ docs
- ├─ .vuepress
- │ ├─ theme
- │ │ └─ index.js
- │ └─ config.js
- └─ README.md
-```
-
-Then, set the absolute path of the theme directory to use it:
-
-```js
-module.exports = {
- theme: '/path/to/docs/.vuepress/theme',
-}
-```
-
-Next, refer to [Advanced > Writing a Theme](./advanced/theme.md) for how to write your own theme.
diff --git a/docs/reference/api.md b/docs/reference/api.md
new file mode 100644
index 00000000..59327929
--- /dev/null
+++ b/docs/reference/api.md
@@ -0,0 +1 @@
+# API
diff --git a/docs/reference/api/README.md b/docs/reference/api/README.md
new file mode 100644
index 00000000..10c76697
--- /dev/null
+++ b/docs/reference/api/README.md
@@ -0,0 +1,182 @@
+---
+sidebar: auto
+---
+
+# API
+Fes.js 统一了API的出口,所有运行时API(包含Fes.js内置API和插件提供的API)全部通过`@fesjs/fes`导出。
+```js
+import { someApi } from "@fesjs/fes"
+```
+
+## 基础API
+
+### plugin
+::: tip
+主要在插件里面使用,项目代码中一般用不到。
+:::
+
+运行时插件接口,是 Umi 内置的跑在浏览器里的一套插件体系。
+```js
+import { plugin, ApplyPluginsType } from 'umi';
+
+// 注册插件
+plugin.register({
+ apply: { dva: { foo: 1 } },
+ path: 'foo',
+});
+plugin.register({
+ apply: { dva: { bar: 1 } },
+ path: 'bar',
+});
+
+// 执行插件
+// 得到 { foo: 1, bar: 1 }
+plugin.applyPlugins({
+ key: 'dva',
+ type: ApplyPluginsType.modify,
+ initialValue: {},
+ args: {},
+ async: false,
+});
+
+```
+
+#### **plugin.register** 参数包含:
+- apply 插件文件导出的内容
+- path 插件文件路径
+
+
+
+
+#### **plugin.applyPlugins** 参数包含:
+- key,坑位的 key
+- type,执行方式类型,详见 [ApplyPluginsType](#applypluginstype)
+- initialValue,初始值
+- args,参数
+- async,是否异步执行且返回 Promise
+
+### ApplyPluginsType
+::: tip
+主要在插件里面使用,项目代码中一般用不到。
+:::
+
+运行时插件执行类型,enum 类型,包含三个属性:
+- compose,用于合并执行多个函数,函数可决定前序函数的执行时机
+- modify,用于修改值
+- event,用于执行事件,前面没有依赖关系
+
+
+## 路由API
+
+Fes.js 路由基于 [Vue Router 4.0](https://next.router.vuejs.org/introduction.html),想了解更多的同学可以看看官方文档。
+
+### useRoute
+返回当前 `route` 实例,相当于在模板内使用 `$route`。必须在 `setup` 函数内调用。
+```js
+import { useRoute } from "@fesjs/fes";
+export default {
+ setup(){
+ const route = useRoute()
+ }
+}
+```
+
+### useRouter
+返回 `router` 实例,相当于在模板语法中使用 `$router`。必须在 `setup` 函数内调用。
+```js
+import { useRouter } from "@fesjs/fes";
+export default {
+ setup(){
+ const router = useRouter()
+ }
+}
+```
+
+### onBeforeRouteUpdate
+添加导航守卫,在当前路由即将更新时触发。类似于之前的`beforeRouteUpdate`,但是可用于任何组件。卸载组件时,将移除守卫。
+```js
+import { onBeforeRouteUpdate } from "@fesjs/fes";
+export default {
+ setup(){
+ onBeforeRouteUpdate((to, from, next)=>{
+ })
+ }
+}
+```
+### onBeforeRouteLeave
+添加导航守卫,在当前路由即将离开时触发。类似于之前的`beforeRouteLeave`,但可用于任何组件。卸载组件时,将移除守卫。
+```js
+import { onBeforeRouteLeave } from "@fesjs/fes";
+export default {
+ setup(){
+ onBeforeRouteLeave((to, from, next)=>{
+ })
+ }
+}
+```
+
+### createWebHashHistory
+::: tip
+在开发插件时可能用上,平时一般用不上
+:::
+创建一个 hash 历史记录。对于没有主机的 web 应用程序 (例如 file://),或当配置服务器不能处理任意URL时这非常有用。注意:如果 SEO 对你很重要,你应该使用 `createWebHistory`。
+
+### createWebHistory
+::: tip
+在开发插件时可能用上,平时一般用不上
+:::
+创建HTML5历史记录。单页应用程序最常见的历史记录。必须通过 http 服务打开页面地址 。
+
+### createMemoryHistory
+::: tip
+在开发插件时可能用上,平时一般用不上
+:::
+创建一个基于内存的历史记录。这个历史记录的主要目的是处理 SSR。它在一个特殊的位置开始,这个位置无处不在。如果用户不在浏览器上下文中,它们可以通过调用 router.push() 或 router.replace() 将该位置替换为启动位置。
+
+### createRouter
+创建一个路由器实例,该实例可用于 Vue 应用程序。查看[路由器选项](https://next.router.vuejs.org/api/#routeroptions),了解可以传递的所有属性的列表。
+
+### RouterLink
+使用自定义组件路由器链接来创建链接,而不是使用常规标签。这使得 Vue 路由器无需重新加载页面即可更改 URL、处理 URL 生成及其编码。
+```html
+Go to About
+```
+可以查看[官方文档](https://next.router.vuejs.org/api/#router-link-props)了解更多 RouterLink 的 Porps。查看[官方文档](https://next.router.vuejs.org/api/#router-link-s-v-slot)了解 RouterLink 的作用域插槽。
+
+### useLink
+返回的结果跟 RouterLink 的作用域插槽的属性一致,查看[官方API](https://next.router.vuejs.org/api/#router-link-s-v-slot)了解更多。
+```js
+import { RouterLink, useLink } from '@fesjs/fes'
+
+export default {
+ name: 'AppLink',
+
+ props: {
+ // add @ts-ignore if using TypeScript
+ ...RouterLink.props,
+ inactiveClass: String,
+ },
+
+ setup(props) {
+ // `props` contains `to` and any other prop that can be passed to
+ const { navigate, href, route, isActive, isExactActive } = useLink(props)
+
+ // profit!
+
+ return { isExternalLink }
+ },
+}
+```
+
+### RouterView
+router-view 将显示当前 URL 的对应的路由组件。你可以把它放在任何地方,以适应你的布局。
+```html
+
+
+
+
+```
+可以查看[官方文档](https://next.router.vuejs.org/api/#router-view-props)了解更多 RouterView 的 Porps。查看[官方文档](https://next.router.vuejs.org/api/#router-view-s-v-slot)了解 RouterView 的作用域插槽。
+
+### Router Methods
+查看[官方文档](https://next.router.vuejs.org/api/#router-methods)了解更多
\ No newline at end of file
diff --git a/docs/reference/bundler/vite.md b/docs/reference/bundler/vite.md
deleted file mode 100644
index d5d34b36..00000000
--- a/docs/reference/bundler/vite.md
+++ /dev/null
@@ -1,3 +0,0 @@
-# Vite
-
-> TODO
diff --git a/docs/reference/bundler/webpack.md b/docs/reference/bundler/webpack.md
deleted file mode 100644
index 9486fe94..00000000
--- a/docs/reference/bundler/webpack.md
+++ /dev/null
@@ -1,3 +0,0 @@
-# Webpack
-
-> TODO
diff --git a/docs/reference/cli.md b/docs/reference/cli.md
index a2cff8a7..1c7eda5e 100644
--- a/docs/reference/cli.md
+++ b/docs/reference/cli.md
@@ -1,8 +1,8 @@
-# Command Line Interface
+# 命令行接口
-VuePress CLI is provided by [@vuepress/cli](https://www.npmjs.com/package/@vuepress/cli) package. It is a dependency of the [vuepress](https://www.npmjs.com/package/vuepress) package, and you can also install it separately.
+VuePress 命令行接口是由 [@vuepress/cli](https://www.npmjs.com/package/@vuepress/cli) 包提供的。它是 [vuepress](https://www.npmjs.com/package/vuepress) 包的依赖之一,当然你也可以单独安装它。
-Run `vuepress --help` to get following help messages:
+执行 `vuepress --help` 来获取下列帮助信息:
```bash
Usage:
@@ -25,7 +25,7 @@ Options:
## dev
-Start a development server to develop your VuePress site locally.
+启动一个开发服务器,在本地开发你的 VuePress 站点。
```bash
Usage:
@@ -47,12 +47,12 @@ Options:
```
::: tip
-Options set by CLI will override those options with the same name in your config file.
+通过命令行设置的配置项,会覆盖你配置文件中的同名配置项。
:::
## build
-Build your VuePress site to static files, which are ready for [deployment](../guide/deployment.md).
+将你的 VuePress 站点构建成静态文件,以便你进行后续[部署](../guide/deployment.md)。
```bash
Usage:
@@ -71,11 +71,11 @@ Options:
```
::: tip
-Options set by CLI will override those options with the same name in your config file.
+通过命令行设置的配置项,会覆盖你配置文件中的同名配置项。
:::
## info
-Outputs information about your system and dependencies.
+输出当前系统和依赖相关的信息。
-This command would be helpful when you want to check your environment or report an issue.
+在你想要检查你的环境,或者提交 Issue 时候,可以使用该命令。
diff --git a/docs/reference/cli/README.md b/docs/reference/cli/README.md
new file mode 100644
index 00000000..261a435d
--- /dev/null
+++ b/docs/reference/cli/README.md
@@ -0,0 +1,175 @@
+---
+sidebar: auto
+---
+
+# 命令行工具
+
+## create-fes-app
+通过 `create-fes-app` 命令创建项目模板,输入`create-fes-app -h`则可以看到如下信息:
+```
+Usage: create-fes-app
+
+Options:
+ -v, --version Output the current version
+ -h, --help Display help for command
+ -f, --force Overwrite target directory if it exists
+ -m, --merge Merge target directory if it exists
+```
+
+可以在本机安装后使用:
+
+
+
+```bash
+# 全局安装
+yarn global add @fesjs/create-fes-app
+
+# 创建模板
+create-fes-app fes-app
+```
+
+
+
+
+
+```bash
+# 全局安装
+npm i -g @fesjs/create-fes-app
+
+# 创建模板
+create-fes-app fes-app
+```
+
+
+
+
+推荐使用 `yarn create` 和 `npx` 方式创建模板,一直使用最新的模板:
+
+
+
+
+```bash
+# 创建模板
+yarn create @fesjs/fes-app myapp
+
+# 安装依赖
+yarn
+
+# 运行
+yarn dev
+```
+
+
+
+
+
+```bash
+# 创建模板
+npx @fesjs/create-fes-app myapp
+
+# 安装依赖
+npm install
+
+# 运行
+npm run dev
+```
+
+
+
+
+
+## fes
+需要在项目根目录执行 `fes` 命令,输入`fes -h`则可以看到如下信息:
+
+```
+Usage: fes [options]
+
+一个好用的前端应用解决方案
+
+Options:
+ -v, --vers output the current version
+ -h, --help display help for command
+
+Commands:
+ build build application for production
+ dev [options] start a local http service for development
+ help show command helps
+ info print debugging information about your environment
+ webpack [options] inspect webpack configurations
+
+ Run fes --help for detailed usage of given command.
+```
+
+### fes dev
+启动本地开发服务器进行项目的开发调试。
+```
+Usage: fes dev [options]
+
+start a local http service for development
+
+Options:
+ --port http service port, like 8080
+ --https whether to turn on the https service
+ -h, --help display help for command
+```
+比如:
+```bash
+fes dev --port=8080
+```
+
+### fes build
+编译构建 web 产物。
+```
+Usage: fes build [options]
+
+build application for production
+
+Options:
+ -h, --help display help for command
+```
+比如:
+```
+fes build
+```
+### fes help
+打印帮助文档。
+比如:
+```bash
+fes help
+```
+
+### fes info
+打印当前项目的有用的环境信息,用来帮助定位问题。
+```
+Usage: fes info [options]
+
+print debugging information about your environment
+
+Options:
+ -h, --help display help for command
+```
+比如:
+```bash
+fes info
+```
+
+### fes webpack
+查看项目使用的 webpack 配置。
+```
+Usage: fes webpack [options]
+
+inspect webpack configurations
+
+Options:
+ --rule inspect a specific module rule
+ --plugin inspect a specific plugin
+ --rules list all module rule names
+ --plugins list all plugin names
+ --verbose show full function definitions in output
+ -h, --help display help for command
+```
+
+比如:
+```bash
+fes webpack
+```
\ No newline at end of file
diff --git a/docs/reference/components.md b/docs/reference/components.md
deleted file mode 100644
index 40f13123..00000000
--- a/docs/reference/components.md
+++ /dev/null
@@ -1,56 +0,0 @@
-# Built-in Components
-
-## ClientOnly
-
-- Usage:
-
-```md
-
-
-
-```
-
-- Details:
-
- This component and its children will only be rendered in client-side. That means, it will not be rendered to HTML during build (SSR).
-
- If a component is trying to access Browser / DOM APIs directly in `setup()`, an error will occur during build because those APIs are unavailable in Node.js environment. In such case, you could do either:
-
- - Modify the component to only access Browser / DOM APIs in `onBeforeMount()` or `onMounted()` hook.
- - Wrap the component with ``.
-
-## Content
-
-- Props:
- - pagePath
- - Type: `string`
- - Required: `false`
-
-- Usage:
-
-```md
-
-
-```
-
-- Details:
-
- This component will render the Markdown content of a page.
-
- If the `pagePath` prop is not provided, it will render the page of current route path.
-
- This component is mainly for developing themes. You won't need it in most cases.
-
-## OutboundLink
-
-- Usage:
-
-```md
-
-```
-
-- Details:
-
- This component will render an indicator for links to external URLs.
-
- This component is mainly for developing themes. You won't need it in most cases.
diff --git a/docs/reference/config.md b/docs/reference/config.md
deleted file mode 100644
index 7cfbb39a..00000000
--- a/docs/reference/config.md
+++ /dev/null
@@ -1,574 +0,0 @@
-# Config
-
-Reference of VuePress config, which can be set via config file. The conventional config files are (in order of precedence):
-
-- In current working directory `cwd`:
- - `vuepress.config.ts`
- - `vuepress.config.js`
-- In source directory `sourceDir`:
- - `.vuepress/config.ts`
- - `.vuepress/config.js`
-
-You can also specify the config file via `--config` option of [CLI](./cli.md).
-
-## Site Config
-
-### base
-
-- Type: `string`
-
-- Default: `/`
-
-- Details:
-
- The base URL the site will be deployed at.
-
- You will need to set this if you plan to deploy your site under a sub path. It should always start and end with a slash. For example, if you plan to deploy your site to GitHub pages at `https://foo.github.io/bar/`, then you should set `base` to `"/bar/"`.
-
- The `base` is automatically prepended to all the URLs that start with `/` in other options, so you only need to specify it once.
-
-- Also see:
- - [Guide > Assets > Base Helper](../guide/assets.md#base-helper)
- - [Guide > Deployment](../guide/deployment.md)
-
-### lang
-
-- Type: `string`
-
-- Default: `en-US`
-
-- Details:
-
- Language for the site.
-
- This will be the `lang` attribute of the `` tag in the rendered HTML.
-
- This can be specified in different locales.
-
-- Also see:
- - [Config > locales](#locales)
- - [Frontmatter > lang](./frontmatter.md#lang)
-
-### title
-
-- Type: `string`
-
-- Default: `''`
-
-- Details:
-
- Title for the site.
-
- This will be the suffix for all page titles, and displayed in the navbar in the default theme.
-
- This can be specified in different locales.
-
-- Also see:
- - [Config > locales](#locales)
-
-### description
-
-- Type: `string`
-
-- Default: `''`
-
-- Details:
-
- Description for the site.
-
- This will be the `content` attribute of `` tag in the rendered HTML, which will be overrode by the `description` field of page frontmatter.
-
- This can be specified in different locales.
-
-- Also see:
- - [Config > locales](#locales)
- - [Frontmatter > description](./frontmatter.md#description)
-
-### head
-
-- Type: `HeadConfig[]`
-
-- Default: `[]`
-
-- Details:
-
- Extra tags to inject into the `` tag in the rendered HTML.
-
- You can specify each tag in the form of `[tagName, { attrName: attrValue }, innerHTML?]`.
-
- This can be specified in different locales.
-
-- Example:
-
- To add a custom favicon:
-
-```js
-module.exports = {
- head: [
- ['link', { rel: 'icon', href: '/logo.png' }]
- ]
-}
-```
-
- Rendered as:
-
-```html
-
-
-
-```
-
-- Also see:
- - [Config > locales](#locales)
- - [Frontmatter > head](./frontmatter.md#head)
-
-### locales
-
-- Type: `{ [path: string]: Partial }`
-
-- Default: `{}`
-
-- Details:
-
- Specify locales for i18n support.
-
- Acceptable fields:
-
- - [lang](#lang)
- - [title](#title)
- - [description](#description)
- - [head](#head)
-
-- Also see:
- - [Guide > I18n](../guide/i18n.md)
-
-## Theme Config
-
-### theme
-
-- Type: `string`
-
-- Default: `'@vuepress/default'`
-
-- Details:
-
- Name or absolute path of theme your want to use.
-
- This option accepts theme name, theme name shorthand, or absolute path to theme.
-
-- Example:
-
-```js
-module.exports = {
- theme: 'vuepress-theme-foo',
- theme: 'bar',
- theme: '/path/to/local/theme',
-}
-```
-
-- Also see:
- - [Guide > Theme](../guide/theme.md)
-
-### themeConfig
-
-- Type: `ThemeConfig`
-
-- Default: `{}`
-
-- Details:
-
- Provide config options to the used theme. The options will vary depending on the theme you are using.
-
-- Also see:
- - [Default Theme > Config](./default-theme/config.md)
-
-## Bundler Config
-
-### bundler
-
-- Type: `string`
-
-- Default: `'@vuepress/webpack'`
-
-- Details:
-
- Name of bundler your want to use.
-
- Bundler name shorthand is acceptable.
-
-- Also see:
- - [Guide > Bundler](../guide/bundler.md)
-
-### bundlerConfig
-
-- Type: `BundlerConfig`
-
-- Default: `{}`
-
-- Details:
-
- Provide config options to the used bundler. The options will vary depending on the bundler you are using.
-
-## Directory Config
-
-### dest
-
-- Type: `string`
-
-- Default: `` `${sourceDir}/.vuepress/dist` ``
-
-- Details:
-
- Specify the output directory for `vuepress build` command.
-
-### temp
-
-- Type: `string`
-
-- Default: `` `${sourceDir}/.vuepress/.temp` ``
-
-- Details:
-
- Specify the directory for temporary files.
-
-### cache
-
-- Type: `string`
-
-- Default: `` `${sourceDir}/.vuepress/.cache` ``
-
-- Details:
-
- Specify the directory for cache .
-
-### public
-
-- Type: `string`
-
-- Default: `` `${sourceDir}/.vuepress/public` ``
-
-- Details:
-
- Specify the directory for public files.
-
-- Also see:
- - [Guide > Assets > Public Files](../guide/assets.md#public-files)
-
-## Markdown Config
-
-### markdown
-
-- Type: `MarkdownOptions`
-
-- Default: `{}`
-
-- Details:
-
- Configure VuePress built-in Markdown syntax extensions.
-
-- Also see:
- - [Guide > Markdown > Syntax Extensions](../guide/markdown.md#syntax-extensions)
-
-#### markdown.anchor
-
-- Type: `AnchorPluginOptions | false`
-
-- Details:
-
- Options for [markdown-it-anchor](https://github.com/valeriangalliat/markdown-it-anchor).
-
- Set to `false` to disable this plugin.
-
-- Also see:
- - [Guide > Markdown > Syntax Extensions > Header Anchors](../guide/markdown.md#header-anchors)
-
-#### markdown.assets
-
-- Type: `AssetsPluginOptions | false`
-
-- Details:
-
- Options for VuePress built-in markdown-it assets plugin.
-
- Set to `false` to disable this plugin.
-
-::: danger
-You should not configure it unless you understand what it is for.
-:::
-
-#### markdown.code
-
-- Type: `CodePluginOptions | false`
-
-- Details:
-
- Options for VuePress built-in markdown-it code plugin.
-
- Set to `false` to disable this plugin.
-
-- Also see:
- - [Guide > Markdown > Syntax Extensions > Code Blocks](../guide/markdown.md#code-blocks)
-
-##### markdown.code.highlight
-
-- Type: `boolean`
-
-- Default: `true`
-
-- Details:
-
- Enable code syntax highlighting or not.
-
- You can disable it if you want to implement client side highlighting by yourself. For example, [Prismjs](https://prismjs.com/) or [highlight.js](https://highlightjs.org/).
-
-- Also see:
- - [Guide > Markdown > Syntax Extensions > Code Blocks > Syntax Highlighting](../guide/markdown.md#syntax-highlighting)
-
-##### markdown.code.highlightLines
-
-- Type: `boolean`
-
-- Default: `true`
-
-- Details:
-
- Enable code line highlighting or not.
-
-- Also see:
- - [Guide > Markdown > Syntax Extensions > Code Blocks > Line Highlighting](../guide/markdown.md#line-highlighting)
-
-##### markdown.code.lineNumbers
-
-- Type: `boolean`
-
-- Default: `true`
-
-- Details:
-
- Enable code line numbers or not.
-
-- Also see:
- - [Guide > Markdown > Syntax Extensions > Code Blocks > Line Numbers](../guide/markdown.md#line-numbers)
-
-##### markdown.code.preWrapper
-
-- Type: `boolean`
-
-- Default: `true`
-
-- Details:
-
- Enable the extra wrapper of the `
` tag or not.
-
- The wrapper is required by the `highlightLines` and `lineNumbers`. That means, if you disable `preWrapper`, the line highlighting and line numbers will also be disabled.
-
- You can disable it if you want to implement them in client side. For example, [Prismjs Line Highlight](https://prismjs.com/plugins/line-highlight/) or [Prismjs Line Numbers](https://prismjs.com/plugins/line-numbers/).
-
-##### markdown.code.vPre
-
-- Type: `boolean`
-
-- Default: `true`
-
-- Details:
-
- Enable the `v-pre` directive on `
` tag or not.
-
-- Also see:
- - [Guide > Markdown > Syntax Extensions > Code Blocks > Wrap with v-pre](../guide/markdown.md#wrap-with-v-pre)
-
-#### markdown.customComponent
-
-- Type: `undefined | false`
-
-- Details:
-
- Options for VuePress built-in markdown-it custom-component plugin.
-
- Set to `false` to disable this plugin.
-
-::: danger
-You should not configure it unless you understand what it is for.
-:::
-
-#### markdown.emoji
-
-- Type: `EmojiPluginOptions | false`
-
-- Details:
-
- Options for [markdown-it-emoji](https://github.com/markdown-it/markdown-it-emoji).
-
- Set to `false` to disable this plugin.
-
-- Also see:
- - [Guide > Markdown > Syntax Extensions > Emoji](../guide/markdown.md#emoji)
-
-#### markdown.extractHeaders
-
-- Type: `ExtractHeadersPluginOptions | false`
-
-- Details:
-
- Options for VuePress built-in markdown-it extract-headers plugin.
-
- It will extract page headers to page data, which will be used for generating sidebar, table of contents, etc. For example, the sidebar of current page is auto generated from the headers that extracted by this plugin.
-
- Set to `false` to disable this plugin.
-
-#### markdown.hoistTags
-
-- Type: `HoistTagsPluginOptions | false`
-
-- Details:
-
- Options for VuePress built-in markdown-it hoist-tags plugin.
-
- It will hoist specific HTML tags in your Markdown to the top-level of SFC. By default, only `
+```
+
+### 角色
+Fes.js 用角色定义一组资源。当访问 Fes.js 应用时,使用插件提供的 API 设置用户的角色,角色对应的资源才可见,非角色对应的资源不可见。
+
+
+当然有时候业务比较复杂,角色不能在前端应用中预先定义,而是用户自定义的,储存在数据库中。不要怕!插件也提供粒度更细的 API 来控制当前用户能访问的资源。
+
+## 启用方式
+在 `package.json` 中引入依赖:
+```json
+{
+ "dependencies": {
+ "@fesjs/fes": "^2.0.0",
+ "@fesjs/plugin-access": "^2.0.0"
+ },
+}
+```
+
+## 配置
+
+### 编译时配置
+在执行 `fes dev` 或者 `fes build` 时,通过此配置生成运行时的代码,在配置文件`.fes.js` 中配置:
+```js
+export default {
+ access: {
+ roles: {
+ admin: ["/", "/onepiece", '/store']
+ }
+ }
+}
+```
+
+#### roles
+- **类型**:对象
+
+- **默认值**:`{}`
+
+- **详情**:
+
+ 角色预定义列表。`key` 是角色 Id ,`value`是角色 Id 对应的资源列表。
+
+
+### 运行时配置
+在 `app.js` 中配置
+
+#### unAccessHandler
+- **类型**:`Function`
+
+- **默认值**:`null`
+
+- **详情**:
+
+ 当进入某个路由时,如果路由对应的页面不属于可见资源列表,则会暂停进入,调用 `unAccessHandler` 函数。
+- **参数**
+ - router:createRouter 创建的路由实例
+ - to: 准备进入的路由
+ - from:离开的路由
+ - next: [next函数](https://next.router.vuejs.org/zh/guide/advanced/navigation-guards.html#%E5%8F%AF%E9%80%89%E7%9A%84%E7%AC%AC%E4%B8%89%E4%B8%AA%E5%8F%82%E6%95%B0-next)
+
+比如:
+```js
+export const access = {
+ unAccessHandler({ to, next }) {
+ const accesssIds = accessApi.getAccess();
+ if (to.path === '/404') {
+ accessApi.setAccess(accesssIds.concat(['/404']));
+ return next('/404');
+ }
+ if (!accesssIds.includes('/403')) {
+ accessApi.setAccess(accesssIds.concat(['/403']));
+ }
+ next('/403');
+ }
+};
+
+```
+
+#### noFoundHandler
+- **类型**:`Function`
+
+- **默认值**:`null`
+
+- **详情**:
+
+ 当进入某个路由时,如果路由对应的页面不存在,则会调用 `noFoundHandler` 函数。
+- **参数**
+ - router:createRouter 创建的路由实例
+ - to: 准备进入的路由
+ - from:离开的路由
+ - next: [next函数](https://next.router.vuejs.org/zh/guide/advanced/navigation-guards.html#%E5%8F%AF%E9%80%89%E7%9A%84%E7%AC%AC%E4%B8%89%E4%B8%AA%E5%8F%82%E6%95%B0-next)
+
+比如:
+```js
+export const access = {
+ noFoundHandler({ next }) {
+ const accesssIds = accessApi.getAccess();
+ if (!accesssIds.includes('/404')) {
+ accessApi.setAccess(accesssIds.concat(['/404']));
+ }
+ next('/404');
+ }
+};
+
+```
+
+## API
+
+### access
+插件 API 通过 `@fesjs/fes` 导出:
+```js
+import { access } from '@fesjs/fes'
+```
+
+#### access.hasAccess
+- **类型**:函数
+
+- **详情**: 判断某个资源是否可见。
+- **参数**:
+ - accessId,资源Id
+- **返回值**:Boolean
+
+#### access.isDataReady
+- **类型**:函数
+
+- **详情**:可以用异步数据来设置权限,`isDataReady` 用来判断异步数据是否已经加载完毕。
+- **参数**:null
+- **返回值**:Boolean
+```js
+import { access } from '@fesjs/fes';
+console.log(access.isDataReady())
+```
+
+
+#### access.setRole
+- **类型**:函数
+
+- **详情**:设置当前的角色。
+- **参数**:
+ - roleId,角色Id,有两种类型:
+ - String,对应着 `roles` 配置对象中的 `key`。
+ - Promise,Promise resolve 的结果应对应着 `roles` 配置对象中的 `key`。
+```js
+import { access } from '@fesjs/fes';
+access.setRole(['admin'])
+```
+
+#### access.setAccess
+- **类型**:函数
+
+- **详情**:设置当前的角色。
+- **参数**:
+ - accessIds,资源Id数组,有两种类型:
+ - Array,数组项对应着 `roles` 配置对象中的 `key`。
+ - Promise,Promise resolve 的结果应该是`Array`。
+```js
+import { access } from '@fesjs/fes';
+access.setAccess(['/a', '/b', '/c'])
+```
+
+#### access.getAccess
+- **类型**:函数
+
+- **详情**:返回当前可见的资源列表。
+- **参数**:null
+
+```js
+import { access } from '@fesjs/fes';
+access.getAccess();
+```
+
+### useAccess
+- **类型**:[composition]((https://v3.cn.vuejs.org/guide/composition-api-introduction.html)) 函数
+
+- **详情**:判断某个资源是否可见。
+- **参数**:
+ - accessId,资源Id
+- **返回值**:`ref`
+
+```vue
+
+