diff --git a/src/api/infra/apiAccessLog.js b/src/api/infra/apiAccessLog.js
new file mode 100644
index 00000000..6456a508
--- /dev/null
+++ b/src/api/infra/apiAccessLog.js
@@ -0,0 +1,20 @@
+import request from '@/utils/request'
+
+// 获得API 访问日志分页
+export function getApiAccessLogPage(query) {
+ return request({
+ url: '/infra/api-access-log/page',
+ method: 'get',
+ params: query
+ })
+}
+
+// 导出API 访问日志 Excel
+export function exportApiAccessLogExcel(query) {
+ return request({
+ url: '/infra/api-access-log/export-excel',
+ method: 'get',
+ params: query,
+ responseType: 'blob'
+ })
+}
diff --git a/src/api/infra/apiErrorLog.js b/src/api/infra/apiErrorLog.js
new file mode 100644
index 00000000..65942d13
--- /dev/null
+++ b/src/api/infra/apiErrorLog.js
@@ -0,0 +1,28 @@
+import request from '@/utils/request'
+
+// 更新 API 错误日志的处理状态
+export function updateApiErrorLogProcess(id, processStatus) {
+ return request({
+ url: '/infra/api-error-log/update-status?id=' + id + '&processStatus=' + processStatus,
+ method: 'put',
+ })
+}
+
+// 获得API 错误日志分页
+export function getApiErrorLogPage(query) {
+ return request({
+ url: '/infra/api-error-log/page',
+ method: 'get',
+ params: query
+ })
+}
+
+// 导出API 错误日志 Excel
+export function exportApiErrorLogExcel(query) {
+ return request({
+ url: '/infra/api-error-log/export-excel',
+ method: 'get',
+ params: query,
+ responseType: 'blob'
+ })
+}
diff --git a/src/api/infra/codegen.js b/src/api/infra/codegen.js
new file mode 100644
index 00000000..ffb3e043
--- /dev/null
+++ b/src/api/infra/codegen.js
@@ -0,0 +1,90 @@
+import request from '@/utils/request'
+
+// 获得表定义分页
+export function getCodegenTablePage(query) {
+ return request({
+ url: '/infra/codegen/table/page',
+ method: 'get',
+ params: query
+ })
+}
+
+// 获得表和字段的明细
+export function getCodegenDetail(tableId) {
+ return request({
+ url: '/infra/codegen/detail?tableId=' + tableId,
+ method: 'get',
+ })
+}
+
+// 修改代码生成信息
+export function updateCodegen(data) {
+ return request({
+ url: '/infra/codegen/update',
+ method: 'put',
+ data: data
+ })
+}
+
+// 基于数据库的表结构,同步数据库的表和字段定义
+export function syncCodegenFromDB(tableId) {
+ return request({
+ url: '/infra/codegen/sync-from-db?tableId=' + tableId,
+ method: 'put'
+ })
+}
+
+// 基于 SQL 建表语句,同步数据库的表和字段定义
+export function syncCodegenFromSQL(tableId, sql) {
+ return request({
+ url: '/infra/codegen/sync-from-sql?tableId=' + tableId,
+ method: 'put',
+ headers:{
+ 'Content-type': 'application/x-www-form-urlencoded'
+ },
+ data: 'tableId=' + tableId + "&sql=" + sql,
+ })
+}
+
+// 预览生成代码
+export function previewCodegen(tableId) {
+ return request({
+ url: '/infra/codegen/preview?tableId=' + tableId,
+ method: 'get',
+ })
+}
+
+// 下载生成代码
+export function downloadCodegen(tableId) {
+ return request({
+ url: '/infra/codegen/download?tableId=' + tableId,
+ method: 'get',
+ responseType: 'blob'
+ })
+}
+
+// 获得表定义分页
+export function getSchemaTableList(query) {
+ return request({
+ url: '/infra/codegen/db/table/list',
+ method: 'get',
+ params: query
+ })
+}
+
+// 基于数据库的表结构,创建代码生成器的表定义
+export function createCodegenList(data) {
+ return request({
+ url: '/infra/codegen/create-list',
+ method: 'post',
+ data: data
+ })
+}
+
+// 删除数据库的表和字段定义
+export function deleteCodegen(tableId) {
+ return request({
+ url: '/infra/codegen/delete?tableId=' + tableId,
+ method: 'delete'
+ })
+}
diff --git a/src/api/infra/config.js b/src/api/infra/config.js
new file mode 100644
index 00000000..eec15177
--- /dev/null
+++ b/src/api/infra/config.js
@@ -0,0 +1,62 @@
+import request from '@/utils/request'
+
+// 查询参数列表
+export function listConfig(query) {
+ return request({
+ url: '/infra/config/page',
+ method: 'get',
+ params: query
+ })
+}
+
+// 查询参数详细
+export function getConfig(configId) {
+ return request({
+ url: '/infra/config/get?id=' + configId,
+ method: 'get'
+ })
+}
+
+// 根据参数键名查询参数值
+export function getConfigKey(configKey) {
+ return request({
+ url: '/infra/config/get-value-by-key?key=' + configKey,
+ method: 'get'
+ })
+}
+
+// 新增参数配置
+export function addConfig(data) {
+ return request({
+ url: '/infra/config/create',
+ method: 'post',
+ data: data
+ })
+}
+
+// 修改参数配置
+export function updateConfig(data) {
+ return request({
+ url: '/infra/config/update',
+ method: 'put',
+ data: data
+ })
+}
+
+// 删除参数配置
+export function delConfig(configId) {
+ return request({
+ url: '/infra/config/delete?id=' + configId,
+ method: 'delete'
+ })
+}
+
+// 导出参数
+export function exportConfig(query) {
+ return request({
+ url: '/infra/config/export',
+ method: 'get',
+ params: query,
+ responseType: 'blob'
+ })
+}
diff --git a/src/api/infra/dataSourceConfig.js b/src/api/infra/dataSourceConfig.js
new file mode 100644
index 00000000..2d96b49c
--- /dev/null
+++ b/src/api/infra/dataSourceConfig.js
@@ -0,0 +1,43 @@
+import request from '@/utils/request'
+
+// 创建数据源配置
+export function createDataSourceConfig(data) {
+ return request({
+ url: '/infra/data-source-config/create',
+ method: 'post',
+ data: data
+ })
+}
+
+// 更新数据源配置
+export function updateDataSourceConfig(data) {
+ return request({
+ url: '/infra/data-source-config/update',
+ method: 'put',
+ data: data
+ })
+}
+
+// 删除数据源配置
+export function deleteDataSourceConfig(id) {
+ return request({
+ url: '/infra/data-source-config/delete?id=' + id,
+ method: 'delete'
+ })
+}
+
+// 获得数据源配置
+export function getDataSourceConfig(id) {
+ return request({
+ url: '/infra/data-source-config/get?id=' + id,
+ method: 'get'
+ })
+}
+
+// 获得数据源配置列表
+export function getDataSourceConfigList() {
+ return request({
+ url: '/infra/data-source-config/list',
+ method: 'get',
+ })
+}
diff --git a/src/api/infra/dbDoc.js b/src/api/infra/dbDoc.js
new file mode 100644
index 00000000..015c6d71
--- /dev/null
+++ b/src/api/infra/dbDoc.js
@@ -0,0 +1,26 @@
+// 导出参数
+import request from "@/utils/request";
+
+export function exportHtml() {
+ return request({
+ url: '/infra/db-doc/export-html',
+ method: 'get',
+ responseType: 'blob'
+ })
+}
+
+export function exportWord() {
+ return request({
+ url: '/infra/db-doc/export-word',
+ method: 'get',
+ responseType: 'blob'
+ })
+}
+
+export function exportMarkdown() {
+ return request({
+ url: '/infra/db-doc/export-markdown',
+ method: 'get',
+ responseType: 'blob'
+ })
+}
diff --git a/src/api/infra/file.js b/src/api/infra/file.js
new file mode 100644
index 00000000..2aeda2e9
--- /dev/null
+++ b/src/api/infra/file.js
@@ -0,0 +1,18 @@
+import request from '@/utils/request'
+
+// 删除文件
+export function deleteFile(id) {
+ return request({
+ url: '/infra/file/delete?id=' + id,
+ method: 'delete'
+ })
+}
+
+// 获得文件分页
+export function getFilePage(query) {
+ return request({
+ url: '/infra/file/page',
+ method: 'get',
+ params: query
+ })
+}
diff --git a/src/api/infra/fileConfig.js b/src/api/infra/fileConfig.js
new file mode 100644
index 00000000..4b75773b
--- /dev/null
+++ b/src/api/infra/fileConfig.js
@@ -0,0 +1,59 @@
+import request from '@/utils/request'
+
+// 创建文件配置
+export function createFileConfig(data) {
+ return request({
+ url: '/infra/file-config/create',
+ method: 'post',
+ data: data
+ })
+}
+
+// 更新文件配置
+export function updateFileConfig(data) {
+ return request({
+ url: '/infra/file-config/update',
+ method: 'put',
+ data: data
+ })
+}
+
+// 更新文件配置为主配置
+export function updateFileConfigMaster(id) {
+ return request({
+ url: '/infra/file-config/update-master?id=' + id,
+ method: 'put'
+ })
+}
+
+// 删除文件配置
+export function deleteFileConfig(id) {
+ return request({
+ url: '/infra/file-config/delete?id=' + id,
+ method: 'delete'
+ })
+}
+
+// 获得文件配置
+export function getFileConfig(id) {
+ return request({
+ url: '/infra/file-config/get?id=' + id,
+ method: 'get'
+ })
+}
+
+// 获得文件配置分页
+export function getFileConfigPage(query) {
+ return request({
+ url: '/infra/file-config/page',
+ method: 'get',
+ params: query
+ })
+}
+
+export function testFileConfig(id) {
+ return request({
+ url: '/infra/file-config/test?id=' + id,
+ method: 'get'
+ })
+}
diff --git a/src/api/infra/job.js b/src/api/infra/job.js
new file mode 100644
index 00000000..c2237c51
--- /dev/null
+++ b/src/api/infra/job.js
@@ -0,0 +1,82 @@
+import request from '@/utils/request'
+
+// 查询定时任务调度列表
+export function listJob(query) {
+ return request({
+ url: '/infra/job/page',
+ method: 'get',
+ params: query
+ })
+}
+
+// 查询定时任务调度详细
+export function getJob(jobId) {
+ return request({
+ url: '/infra/job/get?id=' + jobId,
+ method: 'get'
+ })
+}
+
+// 新增定时任务调度
+export function addJob(data) {
+ return request({
+ url: '/infra/job/create',
+ method: 'post',
+ data: data
+ })
+}
+
+// 修改定时任务调度
+export function updateJob(data) {
+ return request({
+ url: '/infra/job/update',
+ method: 'put',
+ data: data
+ })
+}
+
+// 删除定时任务调度
+export function delJob(jobId) {
+ return request({
+ url: '/infra/job/delete?id=' + jobId,
+ method: 'delete'
+ })
+}
+
+// 导出定时任务调度
+export function exportJob(query) {
+ return request({
+ url: '/infra/job/export-excel',
+ method: 'get',
+ params: query,
+ responseType: 'blob'
+ })
+}
+
+// 任务状态修改
+export function updateJobStatus(jobId, status) {
+ return request({
+ url: '/infra/job/update-status',
+ method: 'put',
+ headers:{
+ 'Content-type': 'application/x-www-form-urlencoded'
+ },
+ data: 'id=' + jobId + "&status=" + status,
+ })
+}
+
+// 定时任务立即执行一次
+export function runJob(jobId) {
+ return request({
+ url: '/infra/job/trigger?id=' + jobId,
+ method: 'put'
+ })
+}
+
+// 获得定时任务的下 n 次执行时间
+export function getJobNextTimes(jobId) {
+ return request({
+ url: '/infra/job/get_next_times?id=' + jobId,
+ method: 'get'
+ })
+}
diff --git a/src/api/infra/jobLog.js b/src/api/infra/jobLog.js
new file mode 100644
index 00000000..348f2ed9
--- /dev/null
+++ b/src/api/infra/jobLog.js
@@ -0,0 +1,28 @@
+import request from '@/utils/request'
+
+// 获得定时任务
+export function getJobLog(id) {
+ return request({
+ url: '/infra/job-log/get?id=' + id,
+ method: 'get'
+ })
+}
+
+// 获得定时任务分页
+export function getJobLogPage(query) {
+ return request({
+ url: '/infra/job-log/page',
+ method: 'get',
+ params: query
+ })
+}
+
+// 导出定时任务 Excel
+export function exportJobLogExcel(query) {
+ return request({
+ url: '/infra/job-log/export-excel',
+ method: 'get',
+ params: query,
+ responseType: 'blob'
+ })
+}
diff --git a/src/api/infra/redis.js b/src/api/infra/redis.js
new file mode 100644
index 00000000..fce18747
--- /dev/null
+++ b/src/api/infra/redis.js
@@ -0,0 +1,9 @@
+import request from '@/utils/request'
+
+// 查询缓存详细
+export function getCache() {
+ return request({
+ url: '/infra/redis/get-monitor-info',
+ method: 'get'
+ })
+}
diff --git a/src/api/infra/testDemo.js b/src/api/infra/testDemo.js
new file mode 100644
index 00000000..cec5742b
--- /dev/null
+++ b/src/api/infra/testDemo.js
@@ -0,0 +1,54 @@
+import request from '@/utils/request'
+
+// 创建字典类型
+export function createTestDemo(data) {
+ return request({
+ url: '/infra/test-demo/create',
+ method: 'post',
+ data: data
+ })
+}
+
+// 更新字典类型
+export function updateTestDemo(data) {
+ return request({
+ url: '/infra/test-demo/update',
+ method: 'put',
+ data: data
+ })
+}
+
+// 删除字典类型
+export function deleteTestDemo(id) {
+ return request({
+ url: '/infra/test-demo/delete?id=' + id,
+ method: 'delete'
+ })
+}
+
+// 获得字典类型
+export function getTestDemo(id) {
+ return request({
+ url: '/infra/test-demo/get?id=' + id,
+ method: 'get'
+ })
+}
+
+// 获得字典类型分页
+export function getTestDemoPage(query) {
+ return request({
+ url: '/infra/test-demo/page',
+ method: 'get',
+ params: query
+ })
+}
+
+// 导出字典类型 Excel
+export function exportTestDemoExcel(query) {
+ return request({
+ url: '/infra/test-demo/export-excel',
+ method: 'get',
+ params: query,
+ responseType: 'blob'
+ })
+}
diff --git a/src/components/Crontab/day.vue b/src/components/Crontab/day.vue
new file mode 100644
index 00000000..d6c74fad
--- /dev/null
+++ b/src/components/Crontab/day.vue
@@ -0,0 +1,160 @@
+
+
+
+
+ 日,允许的通配符[, - * ? / L W]
+
+
+
+
+
+ 不指定
+
+
+
+
+
+ 周期从
+ -
+ 日
+
+
+
+
+
+ 从
+ 号开始,每
+ 日执行一次
+
+
+
+
+
+ 每月
+ 号最近的那个工作日
+
+
+
+
+
+ 本月最后一天
+
+
+
+
+
+ 指定
+
+ {{item}}
+
+
+
+
+
+
+
diff --git a/src/components/Crontab/hour.vue b/src/components/Crontab/hour.vue
new file mode 100644
index 00000000..3b890273
--- /dev/null
+++ b/src/components/Crontab/hour.vue
@@ -0,0 +1,114 @@
+
+
+
+
+ 小时,允许的通配符[, - * /]
+
+
+
+
+
+ 周期从
+ -
+ 小时
+
+
+
+
+
+ 从
+ 小时开始,每
+ 小时执行一次
+
+
+
+
+
+ 指定
+
+ {{item-1}}
+
+
+
+
+
+
+
diff --git a/src/components/Crontab/index.vue b/src/components/Crontab/index.vue
new file mode 100644
index 00000000..f0bc8569
--- /dev/null
+++ b/src/components/Crontab/index.vue
@@ -0,0 +1,430 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 确定
+ 重置
+ 取消
+
+
+
+
+
+
+
diff --git a/src/components/Crontab/min.vue b/src/components/Crontab/min.vue
new file mode 100644
index 00000000..71a1fc17
--- /dev/null
+++ b/src/components/Crontab/min.vue
@@ -0,0 +1,116 @@
+
+
+
+
+ 分钟,允许的通配符[, - * /]
+
+
+
+
+
+ 周期从
+ -
+ 分钟
+
+
+
+
+
+ 从
+ 分钟开始,每
+ 分钟执行一次
+
+
+
+
+
+ 指定
+
+ {{item-1}}
+
+
+
+
+
+
+
+
diff --git a/src/components/Crontab/month.vue b/src/components/Crontab/month.vue
new file mode 100644
index 00000000..2539935e
--- /dev/null
+++ b/src/components/Crontab/month.vue
@@ -0,0 +1,114 @@
+
+
+
+
+ 月,允许的通配符[, - * /]
+
+
+
+
+
+ 周期从
+ -
+ 月
+
+
+
+
+
+ 从
+ 月开始,每
+ 月月执行一次
+
+
+
+
+
+ 指定
+
+ {{item}}
+
+
+
+
+
+
+
diff --git a/src/components/Crontab/result.vue b/src/components/Crontab/result.vue
new file mode 100644
index 00000000..605c8c9f
--- /dev/null
+++ b/src/components/Crontab/result.vue
@@ -0,0 +1,559 @@
+
+
+
+
+
diff --git a/src/components/Crontab/second.vue b/src/components/Crontab/second.vue
new file mode 100644
index 00000000..0b5df29c
--- /dev/null
+++ b/src/components/Crontab/second.vue
@@ -0,0 +1,117 @@
+
+
+
+
+ 秒,允许的通配符[, - * /]
+
+
+
+
+
+ 周期从
+ -
+ 秒
+
+
+
+
+
+ 从
+ 秒开始,每
+ 秒执行一次
+
+
+
+
+
+ 指定
+
+ {{item-1}}
+
+
+
+
+
+
+
diff --git a/src/components/Crontab/week.vue b/src/components/Crontab/week.vue
new file mode 100644
index 00000000..c84f3b3b
--- /dev/null
+++ b/src/components/Crontab/week.vue
@@ -0,0 +1,202 @@
+
+
+
+
+ 周,允许的通配符[, - * ? / L #]
+
+
+
+
+
+ 不指定
+
+
+
+
+
+ 周期从星期
+
+ {{item.value}}
+
+ -
+
+ {{item.value}}
+
+
+
+
+
+
+ 第
+ 周的星期
+
+ {{item.value}}
+
+
+
+
+
+
+ 本月最后一个星期
+
+ {{item.value}}
+
+
+
+
+
+
+ 指定
+
+ {{item.value}}
+
+
+
+
+
+
+
+
diff --git a/src/components/Crontab/year.vue b/src/components/Crontab/year.vue
new file mode 100644
index 00000000..2c8eec32
--- /dev/null
+++ b/src/components/Crontab/year.vue
@@ -0,0 +1,131 @@
+
+
+
+
+ 不填,允许的通配符[, - * /]
+
+
+
+
+
+ 每年
+
+
+
+
+
+ 周期从
+ -
+
+
+
+
+
+
+ 从
+ 年开始,每
+ 年执行一次
+
+
+
+
+
+
+ 指定
+
+
+
+
+
+
+
+
+
diff --git a/src/components/generator/config.js b/src/components/generator/config.js
new file mode 100644
index 00000000..780a1bb6
--- /dev/null
+++ b/src/components/generator/config.js
@@ -0,0 +1,630 @@
+// 表单属性【右面板】
+export const formConf = {
+ formRef: 'elForm',
+ formModel: 'formData',
+ size: 'medium',
+ labelPosition: 'right',
+ labelWidth: 100,
+ formRules: 'rules',
+ gutter: 15,
+ disabled: false,
+ span: 24,
+ formBtns: true
+}
+
+// 输入型组件 【左面板】
+export const inputComponents = [
+ {
+ // 组件的自定义配置
+ __config__: {
+ label: '单行文本',
+ labelWidth: null,
+ showLabel: true,
+ changeTag: true,
+ tag: 'el-input',
+ tagIcon: 'input',
+ defaultValue: undefined,
+ required: true,
+ layout: 'colFormItem',
+ span: 24,
+ document: 'https://element.eleme.cn/#/zh-CN/component/input',
+ // 正则校验规则
+ regList: []
+ },
+ // 组件的插槽属性
+ __slot__: {
+ prepend: '',
+ append: ''
+ },
+ // 其余的为可直接写在组件标签上的属性
+ placeholder: '请输入',
+ style: { width: '100%' },
+ clearable: true,
+ 'prefix-icon': '',
+ 'suffix-icon': '',
+ maxlength: null,
+ 'show-word-limit': false,
+ readonly: false,
+ disabled: false
+ },
+ {
+ __config__: {
+ label: '多行文本',
+ labelWidth: null,
+ showLabel: true,
+ tag: 'el-input',
+ tagIcon: 'textarea',
+ defaultValue: undefined,
+ required: true,
+ layout: 'colFormItem',
+ span: 24,
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/input'
+ },
+ type: 'textarea',
+ placeholder: '请输入',
+ autosize: {
+ minRows: 4,
+ maxRows: 4
+ },
+ style: { width: '100%' },
+ maxlength: null,
+ 'show-word-limit': false,
+ readonly: false,
+ disabled: false
+ },
+ {
+ __config__: {
+ label: '密码',
+ showLabel: true,
+ labelWidth: null,
+ changeTag: true,
+ tag: 'el-input',
+ tagIcon: 'password',
+ defaultValue: undefined,
+ layout: 'colFormItem',
+ span: 24,
+ required: true,
+ regList: [],
+ document: 'https://element.eleme.cn/#/zh-CN/component/input'
+ },
+ __slot__: {
+ prepend: '',
+ append: ''
+ },
+ placeholder: '请输入',
+ 'show-password': true,
+ style: { width: '100%' },
+ clearable: true,
+ 'prefix-icon': '',
+ 'suffix-icon': '',
+ maxlength: null,
+ 'show-word-limit': false,
+ readonly: false,
+ disabled: false
+ },
+ {
+ __config__: {
+ label: '计数器',
+ showLabel: true,
+ changeTag: true,
+ labelWidth: null,
+ tag: 'el-input-number',
+ tagIcon: 'number',
+ defaultValue: undefined,
+ span: 24,
+ layout: 'colFormItem',
+ required: true,
+ regList: [],
+ document: 'https://element.eleme.cn/#/zh-CN/component/input-number'
+ },
+ placeholder: '',
+ min: undefined,
+ max: undefined,
+ step: 1,
+ 'step-strictly': false,
+ precision: undefined,
+ 'controls-position': '',
+ disabled: false
+ },
+ {
+ __config__: {
+ label: '编辑器',
+ showLabel: true,
+ changeTag: true,
+ labelWidth: null,
+ tag: 'tinymce',
+ tagIcon: 'rich-text',
+ defaultValue: null,
+ span: 24,
+ layout: 'colFormItem',
+ required: true,
+ regList: [],
+ document: 'http://tinymce.ax-z.cn'
+ },
+ placeholder: '请输入',
+ height: 300, // 编辑器高度
+ branding: false // 隐藏右下角品牌烙印
+ }
+]
+
+// 选择型组件 【左面板】
+export const selectComponents = [
+ {
+ __config__: {
+ label: '下拉选择',
+ showLabel: true,
+ labelWidth: null,
+ tag: 'el-select',
+ tagIcon: 'select',
+ layout: 'colFormItem',
+ span: 24,
+ required: true,
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/select'
+ },
+ __slot__: {
+ options: [{
+ label: '选项一',
+ value: 1
+ }, {
+ label: '选项二',
+ value: 2
+ }]
+ },
+ placeholder: '请选择',
+ style: { width: '100%' },
+ clearable: true,
+ disabled: false,
+ filterable: false,
+ multiple: false
+ },
+ {
+ __config__: {
+ label: '级联选择',
+ url: 'https://www.fastmock.site/mock/f8d7a54fb1e60561e2f720d5a810009d/fg/cascaderList',
+ method: 'get',
+ dataPath: 'list',
+ dataConsumer: 'options',
+ showLabel: true,
+ labelWidth: null,
+ tag: 'el-cascader',
+ tagIcon: 'cascader',
+ layout: 'colFormItem',
+ defaultValue: [],
+ dataType: 'dynamic',
+ span: 24,
+ required: true,
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/cascader'
+ },
+ options: [{
+ id: 1,
+ value: 1,
+ label: '选项1',
+ children: [{
+ id: 2,
+ value: 2,
+ label: '选项1-1'
+ }]
+ }],
+ placeholder: '请选择',
+ style: { width: '100%' },
+ props: {
+ props: {
+ multiple: false,
+ label: 'label',
+ value: 'value',
+ children: 'children'
+ }
+ },
+ 'show-all-levels': true,
+ disabled: false,
+ clearable: true,
+ filterable: false,
+ separator: '/'
+ },
+ {
+ __config__: {
+ label: '单选框组',
+ labelWidth: null,
+ showLabel: true,
+ tag: 'el-radio-group',
+ tagIcon: 'radio',
+ changeTag: true,
+ defaultValue: undefined,
+ layout: 'colFormItem',
+ span: 24,
+ optionType: 'default',
+ regList: [],
+ required: true,
+ border: false,
+ document: 'https://element.eleme.cn/#/zh-CN/component/radio'
+ },
+ __slot__: {
+ options: [{
+ label: '选项一',
+ value: 1
+ }, {
+ label: '选项二',
+ value: 2
+ }]
+ },
+ style: {},
+ size: 'medium',
+ disabled: false
+ },
+ {
+ __config__: {
+ label: '多选框组',
+ tag: 'el-checkbox-group',
+ tagIcon: 'checkbox',
+ defaultValue: [],
+ span: 24,
+ showLabel: true,
+ labelWidth: null,
+ layout: 'colFormItem',
+ optionType: 'default',
+ required: true,
+ regList: [],
+ changeTag: true,
+ border: false,
+ document: 'https://element.eleme.cn/#/zh-CN/component/checkbox'
+ },
+ __slot__: {
+ options: [{
+ label: '选项一',
+ value: 1
+ }, {
+ label: '选项二',
+ value: 2
+ }]
+ },
+ style: {},
+ size: 'medium',
+ min: null,
+ max: null,
+ disabled: false
+ },
+ {
+ __config__: {
+ label: '开关',
+ tag: 'el-switch',
+ tagIcon: 'switch',
+ defaultValue: false,
+ span: 24,
+ showLabel: true,
+ labelWidth: null,
+ layout: 'colFormItem',
+ required: true,
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/switch'
+ },
+ style: {},
+ disabled: false,
+ 'active-text': '',
+ 'inactive-text': '',
+ 'active-color': null,
+ 'inactive-color': null,
+ 'active-value': true,
+ 'inactive-value': false
+ },
+ {
+ __config__: {
+ label: '滑块',
+ tag: 'el-slider',
+ tagIcon: 'slider',
+ defaultValue: null,
+ span: 24,
+ showLabel: true,
+ layout: 'colFormItem',
+ labelWidth: null,
+ required: true,
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/slider'
+ },
+ disabled: false,
+ min: 0,
+ max: 100,
+ step: 1,
+ 'show-stops': false,
+ range: false
+ },
+ {
+ __config__: {
+ label: '时间选择',
+ tag: 'el-time-picker',
+ tagIcon: 'time',
+ defaultValue: null,
+ span: 24,
+ showLabel: true,
+ layout: 'colFormItem',
+ labelWidth: null,
+ required: true,
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/time-picker'
+ },
+ placeholder: '请选择',
+ style: { width: '100%' },
+ disabled: false,
+ clearable: true,
+ 'picker-options': {
+ selectableRange: '00:00:00-23:59:59'
+ },
+ format: 'HH:mm:ss',
+ 'value-format': 'HH:mm:ss'
+ },
+ {
+ __config__: {
+ label: '时间范围',
+ tag: 'el-time-picker',
+ tagIcon: 'time-range',
+ span: 24,
+ showLabel: true,
+ labelWidth: null,
+ layout: 'colFormItem',
+ defaultValue: null,
+ required: true,
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/time-picker'
+ },
+ style: { width: '100%' },
+ disabled: false,
+ clearable: true,
+ 'is-range': true,
+ 'range-separator': '至',
+ 'start-placeholder': '开始时间',
+ 'end-placeholder': '结束时间',
+ format: 'HH:mm:ss',
+ 'value-format': 'HH:mm:ss'
+ },
+ {
+ __config__: {
+ label: '日期选择',
+ tag: 'el-date-picker',
+ tagIcon: 'date',
+ defaultValue: null,
+ showLabel: true,
+ labelWidth: null,
+ span: 24,
+ layout: 'colFormItem',
+ required: true,
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/date-picker'
+ },
+ placeholder: '请选择',
+ type: 'date',
+ style: { width: '100%' },
+ disabled: false,
+ clearable: true,
+ format: 'yyyy-MM-dd',
+ 'value-format': 'yyyy-MM-dd',
+ readonly: false
+ },
+ {
+ __config__: {
+ label: '日期范围',
+ tag: 'el-date-picker',
+ tagIcon: 'date-range',
+ defaultValue: null,
+ span: 24,
+ showLabel: true,
+ labelWidth: null,
+ required: true,
+ layout: 'colFormItem',
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/date-picker'
+ },
+ style: { width: '100%' },
+ type: 'daterange',
+ 'range-separator': '至',
+ 'start-placeholder': '开始日期',
+ 'end-placeholder': '结束日期',
+ disabled: false,
+ clearable: true,
+ format: 'yyyy-MM-dd',
+ 'value-format': 'yyyy-MM-dd',
+ readonly: false
+ },
+ {
+ __config__: {
+ label: '评分',
+ tag: 'el-rate',
+ tagIcon: 'rate',
+ defaultValue: 0,
+ span: 24,
+ showLabel: true,
+ labelWidth: null,
+ layout: 'colFormItem',
+ required: true,
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/rate'
+ },
+ style: {},
+ max: 5,
+ 'allow-half': false,
+ 'show-text': false,
+ 'show-score': false,
+ disabled: false
+ },
+ {
+ __config__: {
+ label: '颜色选择',
+ tag: 'el-color-picker',
+ tagIcon: 'color',
+ span: 24,
+ defaultValue: null,
+ showLabel: true,
+ labelWidth: null,
+ layout: 'colFormItem',
+ required: true,
+ regList: [],
+ changeTag: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/color-picker'
+ },
+ 'show-alpha': false,
+ 'color-format': '',
+ disabled: false,
+ size: 'medium'
+ },
+ {
+ __config__: {
+ label: '上传',
+ tag: 'el-upload',
+ tagIcon: 'upload',
+ layout: 'colFormItem',
+ defaultValue: null,
+ showLabel: true,
+ labelWidth: null,
+ required: true,
+ span: 24,
+ showTip: false,
+ buttonText: '点击上传',
+ regList: [],
+ changeTag: true,
+ fileSize: 2,
+ sizeUnit: 'MB',
+ document: 'https://element.eleme.cn/#/zh-CN/component/upload'
+ },
+ __slot__: {
+ 'list-type': true
+ },
+ // action: process.env.VUE_APP_BASE_API + "/admin-api/infra/file/upload", // 请求地址
+ action: '/infra/file/upload', // 请求地址
+ disabled: false,
+ accept: '',
+ name: 'file',
+ 'auto-upload': true,
+ 'list-type': 'text',
+ multiple: false
+ }
+]
+
+// 布局型组件 【左面板】
+export const layoutComponents = [
+ {
+ __config__: {
+ layout: 'rowFormItem',
+ tagIcon: 'row',
+ label: '行容器',
+ layoutTree: true,
+ document: 'https://element.eleme.cn/#/zh-CN/component/layout#row-attributes'
+ },
+ type: 'default',
+ justify: 'start',
+ align: 'top'
+ },
+ {
+ __config__: {
+ label: '按钮',
+ showLabel: true,
+ changeTag: true,
+ labelWidth: null,
+ tag: 'el-button',
+ tagIcon: 'button',
+ span: 24,
+ layout: 'colFormItem',
+ document: 'https://element.eleme.cn/#/zh-CN/component/button'
+ },
+ __slot__: {
+ default: '主要按钮'
+ },
+ type: 'primary',
+ icon: 'el-icon-search',
+ round: false,
+ size: 'medium',
+ plain: false,
+ circle: false,
+ disabled: false
+ },
+ {
+ __config__: {
+ layout: 'colFormItem',
+ tagIcon: 'table',
+ tag: 'el-table',
+ document: 'https://element.eleme.cn/#/zh-CN/component/table',
+ span: 24,
+ formId: 101,
+ renderKey: 1595761764203,
+ componentName: 'row101',
+ showLabel: true,
+ changeTag: true,
+ labelWidth: null,
+ label: '表格[开发中]',
+ dataType: 'dynamic',
+ method: 'get',
+ dataPath: 'list',
+ dataConsumer: 'data',
+ url: 'https://www.fastmock.site/mock/f8d7a54fb1e60561e2f720d5a810009d/fg/tableData',
+ children: [{
+ __config__: {
+ layout: 'raw',
+ tag: 'el-table-column',
+ renderKey: 15957617660153
+ },
+ prop: 'date',
+ label: '日期'
+ }, {
+ __config__: {
+ layout: 'raw',
+ tag: 'el-table-column',
+ renderKey: 15957617660152
+ },
+ prop: 'address',
+ label: '地址'
+ }, {
+ __config__: {
+ layout: 'raw',
+ tag: 'el-table-column',
+ renderKey: 15957617660151
+ },
+ prop: 'name',
+ label: '名称'
+ }, {
+ __config__: {
+ layout: 'raw',
+ tag: 'el-table-column',
+ renderKey: 1595774496335,
+ children: [
+ {
+ __config__: {
+ label: '按钮',
+ tag: 'el-button',
+ tagIcon: 'button',
+ layout: 'raw',
+ renderKey: 1595779809901
+ },
+ __slot__: {
+ default: '主要按钮'
+ },
+ type: 'primary',
+ icon: 'el-icon-search',
+ round: false,
+ size: 'medium'
+ }
+ ]
+ },
+ label: '操作'
+ }]
+ },
+ data: [],
+ directives: [{
+ name: 'loading',
+ value: true
+ }],
+ border: true,
+ type: 'default',
+ justify: 'start',
+ align: 'top'
+ }
+]
diff --git a/src/components/generator/css.js b/src/components/generator/css.js
new file mode 100644
index 00000000..7cb86e64
--- /dev/null
+++ b/src/components/generator/css.js
@@ -0,0 +1,18 @@
+const styles = {
+ 'el-rate': '.el-rate{display: inline-block; vertical-align: text-top;}',
+ 'el-upload': '.el-upload__tip{line-height: 1.2;}'
+}
+
+function addCss(cssList, el) {
+ const css = styles[el.__config__.tag]
+ css && cssList.indexOf(css) === -1 && cssList.push(css)
+ if (el.__config__.children) {
+ el.__config__.children.forEach(el2 => addCss(cssList, el2))
+ }
+}
+
+export function makeUpCss(conf) {
+ const cssList = []
+ conf.fields.forEach(el => addCss(cssList, el))
+ return cssList.join('\n')
+}
diff --git a/src/components/generator/drawingDefalut.js b/src/components/generator/drawingDefalut.js
new file mode 100644
index 00000000..dbc1daf2
--- /dev/null
+++ b/src/components/generator/drawingDefalut.js
@@ -0,0 +1,37 @@
+export default [
+ {
+ __config__: {
+ label: '单行文本',
+ labelWidth: null,
+ showLabel: true,
+ changeTag: true,
+ tag: 'el-input',
+ tagIcon: 'input',
+ defaultValue: undefined,
+ required: true,
+ layout: 'colFormItem',
+ span: 24,
+ document: 'https://element.eleme.cn/#/zh-CN/component/input',
+ // 正则校验规则
+ regList: [{
+ pattern: '/^1(3|4|5|7|8|9)\\d{9}$/',
+ message: '手机号格式错误'
+ }]
+ },
+ // 组件的插槽属性
+ __slot__: {
+ prepend: '',
+ append: ''
+ },
+ __vModel__: 'mobile',
+ placeholder: '请输入手机号',
+ style: { width: '100%' },
+ clearable: true,
+ 'prefix-icon': 'el-icon-mobile',
+ 'suffix-icon': '',
+ maxlength: 11,
+ 'show-word-limit': true,
+ readonly: false,
+ disabled: false
+ }
+]
diff --git a/src/components/generator/html.js b/src/components/generator/html.js
new file mode 100644
index 00000000..6e9a32e6
--- /dev/null
+++ b/src/components/generator/html.js
@@ -0,0 +1,399 @@
+/* eslint-disable max-len */
+import ruleTrigger from './ruleTrigger'
+
+let confGlobal
+let someSpanIsNot24
+
+export function dialogWrapper(str) {
+ return `
+ ${str}
+
+ 取消
+ 确定
+
+ `
+}
+
+export function vueTemplate(str) {
+ return `
+
+ ${str}
+
+ `
+}
+
+export function vueScript(str) {
+ return ``
+}
+
+export function cssStyle(cssStr) {
+ return ``
+}
+
+function buildFormTemplate(scheme, child, type) {
+ let labelPosition = ''
+ if (scheme.labelPosition !== 'right') {
+ labelPosition = `label-position="${scheme.labelPosition}"`
+ }
+ const disabled = scheme.disabled ? `:disabled="${scheme.disabled}"` : ''
+ let str = `
+ ${child}
+ ${buildFromBtns(scheme, type)}
+ `
+ if (someSpanIsNot24) {
+ str = `
+ ${str}
+ `
+ }
+ return str
+}
+
+function buildFromBtns(scheme, type) {
+ let str = ''
+ if (scheme.formBtns && type === 'file') {
+ str = `
+ 提交
+ 重置
+ `
+ if (someSpanIsNot24) {
+ str = `
+ ${str}
+ `
+ }
+ }
+ return str
+}
+
+// span不为24的用el-col包裹
+function colWrapper(scheme, str) {
+ if (someSpanIsNot24 || scheme.__config__.span !== 24) {
+ return `
+ ${str}
+ `
+ }
+ return str
+}
+
+const layouts = {
+ colFormItem(scheme) {
+ const config = scheme.__config__
+ let labelWidth = ''
+ let label = `label="${config.label}"`
+ if (config.labelWidth && config.labelWidth !== confGlobal.labelWidth) {
+ labelWidth = `label-width="${config.labelWidth}px"`
+ }
+ if (config.showLabel === false) {
+ labelWidth = 'label-width="0"'
+ label = ''
+ }
+ const required = !ruleTrigger[config.tag] && config.required ? 'required' : ''
+ const tagDom = tags[config.tag] ? tags[config.tag](scheme) : null
+ let str = `
+ ${tagDom}
+ `
+ str = colWrapper(scheme, str)
+ return str
+ },
+ rowFormItem(scheme) {
+ const config = scheme.__config__
+ const type = scheme.type === 'default' ? '' : `type="${scheme.type}"`
+ const justify = scheme.type === 'default' ? '' : `justify="${scheme.justify}"`
+ const align = scheme.type === 'default' ? '' : `align="${scheme.align}"`
+ const gutter = scheme.gutter ? `:gutter="${scheme.gutter}"` : ''
+ const children = config.children.map(el => layouts[el.__config__.layout](el))
+ let str = `
+ ${children.join('\n')}
+ `
+ str = colWrapper(scheme, str)
+ return str
+ }
+}
+
+const tags = {
+ 'el-button': el => {
+ const {
+ tag, disabled
+ } = attrBuilder(el)
+ const type = el.type ? `type="${el.type}"` : ''
+ const icon = el.icon ? `icon="${el.icon}"` : ''
+ const round = el.round ? 'round' : ''
+ const size = el.size ? `size="${el.size}"` : ''
+ const plain = el.plain ? 'plain' : ''
+ const circle = el.circle ? 'circle' : ''
+ let child = buildElButtonChild(el)
+
+ if (child) child = `\n${child}\n` // 换行
+ return `<${tag} ${type} ${icon} ${round} ${size} ${plain} ${disabled} ${circle}>${child}${tag}>`
+ },
+ 'el-input': el => {
+ const {
+ tag, disabled, vModel, clearable, placeholder, width
+ } = attrBuilder(el)
+ const maxlength = el.maxlength ? `:maxlength="${el.maxlength}"` : ''
+ const showWordLimit = el['show-word-limit'] ? 'show-word-limit' : ''
+ const readonly = el.readonly ? 'readonly' : ''
+ const prefixIcon = el['prefix-icon'] ? `prefix-icon='${el['prefix-icon']}'` : ''
+ const suffixIcon = el['suffix-icon'] ? `suffix-icon='${el['suffix-icon']}'` : ''
+ const showPassword = el['show-password'] ? 'show-password' : ''
+ const type = el.type ? `type="${el.type}"` : ''
+ const autosize = el.autosize && el.autosize.minRows
+ ? `:autosize="{minRows: ${el.autosize.minRows}, maxRows: ${el.autosize.maxRows}}"`
+ : ''
+ let child = buildElInputChild(el)
+
+ if (child) child = `\n${child}\n` // 换行
+ return `<${tag} ${vModel} ${type} ${placeholder} ${maxlength} ${showWordLimit} ${readonly} ${disabled} ${clearable} ${prefixIcon} ${suffixIcon} ${showPassword} ${autosize} ${width}>${child}${tag}>`
+ },
+ 'el-input-number': el => {
+ const {
+ tag, disabled, vModel, placeholder
+ } = attrBuilder(el)
+ const controlsPosition = el['controls-position'] ? `controls-position=${el['controls-position']}` : ''
+ const min = el.min ? `:min='${el.min}'` : ''
+ const max = el.max ? `:max='${el.max}'` : ''
+ const step = el.step ? `:step='${el.step}'` : ''
+ const stepStrictly = el['step-strictly'] ? 'step-strictly' : ''
+ const precision = el.precision ? `:precision='${el.precision}'` : ''
+
+ return `<${tag} ${vModel} ${placeholder} ${step} ${stepStrictly} ${precision} ${controlsPosition} ${min} ${max} ${disabled}>${tag}>`
+ },
+ 'el-select': el => {
+ const {
+ tag, disabled, vModel, clearable, placeholder, width
+ } = attrBuilder(el)
+ const filterable = el.filterable ? 'filterable' : ''
+ const multiple = el.multiple ? 'multiple' : ''
+ let child = buildElSelectChild(el)
+
+ if (child) child = `\n${child}\n` // 换行
+ return `<${tag} ${vModel} ${placeholder} ${disabled} ${multiple} ${filterable} ${clearable} ${width}>${child}${tag}>`
+ },
+ 'el-radio-group': el => {
+ const { tag, disabled, vModel } = attrBuilder(el)
+ const size = `size="${el.size}"`
+ let child = buildElRadioGroupChild(el)
+
+ if (child) child = `\n${child}\n` // 换行
+ return `<${tag} ${vModel} ${size} ${disabled}>${child}${tag}>`
+ },
+ 'el-checkbox-group': el => {
+ const { tag, disabled, vModel } = attrBuilder(el)
+ const size = `size="${el.size}"`
+ const min = el.min ? `:min="${el.min}"` : ''
+ const max = el.max ? `:max="${el.max}"` : ''
+ let child = buildElCheckboxGroupChild(el)
+
+ if (child) child = `\n${child}\n` // 换行
+ return `<${tag} ${vModel} ${min} ${max} ${size} ${disabled}>${child}${tag}>`
+ },
+ 'el-switch': el => {
+ const { tag, disabled, vModel } = attrBuilder(el)
+ const activeText = el['active-text'] ? `active-text="${el['active-text']}"` : ''
+ const inactiveText = el['inactive-text'] ? `inactive-text="${el['inactive-text']}"` : ''
+ const activeColor = el['active-color'] ? `active-color="${el['active-color']}"` : ''
+ const inactiveColor = el['inactive-color'] ? `inactive-color="${el['inactive-color']}"` : ''
+ const activeValue = el['active-value'] !== true ? `:active-value='${JSON.stringify(el['active-value'])}'` : ''
+ const inactiveValue = el['inactive-value'] !== false ? `:inactive-value='${JSON.stringify(el['inactive-value'])}'` : ''
+
+ return `<${tag} ${vModel} ${activeText} ${inactiveText} ${activeColor} ${inactiveColor} ${activeValue} ${inactiveValue} ${disabled}>${tag}>`
+ },
+ 'el-cascader': el => {
+ const {
+ tag, disabled, vModel, clearable, placeholder, width
+ } = attrBuilder(el)
+ const options = el.options ? `:options="${el.__vModel__}Options"` : ''
+ const props = el.props ? `:props="${el.__vModel__}Props"` : ''
+ const showAllLevels = el['show-all-levels'] ? '' : ':show-all-levels="false"'
+ const filterable = el.filterable ? 'filterable' : ''
+ const separator = el.separator === '/' ? '' : `separator="${el.separator}"`
+
+ return `<${tag} ${vModel} ${options} ${props} ${width} ${showAllLevels} ${placeholder} ${separator} ${filterable} ${clearable} ${disabled}>${tag}>`
+ },
+ 'el-slider': el => {
+ const { tag, disabled, vModel } = attrBuilder(el)
+ const min = el.min ? `:min='${el.min}'` : ''
+ const max = el.max ? `:max='${el.max}'` : ''
+ const step = el.step ? `:step='${el.step}'` : ''
+ const range = el.range ? 'range' : ''
+ const showStops = el['show-stops'] ? `:show-stops="${el['show-stops']}"` : ''
+
+ return `<${tag} ${min} ${max} ${step} ${vModel} ${range} ${showStops} ${disabled}>${tag}>`
+ },
+ 'el-time-picker': el => {
+ const {
+ tag, disabled, vModel, clearable, placeholder, width
+ } = attrBuilder(el)
+ const startPlaceholder = el['start-placeholder'] ? `start-placeholder="${el['start-placeholder']}"` : ''
+ const endPlaceholder = el['end-placeholder'] ? `end-placeholder="${el['end-placeholder']}"` : ''
+ const rangeSeparator = el['range-separator'] ? `range-separator="${el['range-separator']}"` : ''
+ const isRange = el['is-range'] ? 'is-range' : ''
+ const format = el.format ? `format="${el.format}"` : ''
+ const valueFormat = el['value-format'] ? `value-format="${el['value-format']}"` : ''
+ const pickerOptions = el['picker-options'] ? `:picker-options='${JSON.stringify(el['picker-options'])}'` : ''
+
+ return `<${tag} ${vModel} ${isRange} ${format} ${valueFormat} ${pickerOptions} ${width} ${placeholder} ${startPlaceholder} ${endPlaceholder} ${rangeSeparator} ${clearable} ${disabled}>${tag}>`
+ },
+ 'el-date-picker': el => {
+ const {
+ tag, disabled, vModel, clearable, placeholder, width
+ } = attrBuilder(el)
+ const startPlaceholder = el['start-placeholder'] ? `start-placeholder="${el['start-placeholder']}"` : ''
+ const endPlaceholder = el['end-placeholder'] ? `end-placeholder="${el['end-placeholder']}"` : ''
+ const rangeSeparator = el['range-separator'] ? `range-separator="${el['range-separator']}"` : ''
+ const format = el.format ? `format="${el.format}"` : ''
+ const valueFormat = el['value-format'] ? `value-format="${el['value-format']}"` : ''
+ const type = el.type === 'date' ? '' : `type="${el.type}"`
+ const readonly = el.readonly ? 'readonly' : ''
+
+ return `<${tag} ${type} ${vModel} ${format} ${valueFormat} ${width} ${placeholder} ${startPlaceholder} ${endPlaceholder} ${rangeSeparator} ${clearable} ${readonly} ${disabled}>${tag}>`
+ },
+ 'el-rate': el => {
+ const { tag, disabled, vModel } = attrBuilder(el)
+ const max = el.max ? `:max='${el.max}'` : ''
+ const allowHalf = el['allow-half'] ? 'allow-half' : ''
+ const showText = el['show-text'] ? 'show-text' : ''
+ const showScore = el['show-score'] ? 'show-score' : ''
+
+ return `<${tag} ${vModel} ${max} ${allowHalf} ${showText} ${showScore} ${disabled}>${tag}>`
+ },
+ 'el-color-picker': el => {
+ const { tag, disabled, vModel } = attrBuilder(el)
+ const size = `size="${el.size}"`
+ const showAlpha = el['show-alpha'] ? 'show-alpha' : ''
+ const colorFormat = el['color-format'] ? `color-format="${el['color-format']}"` : ''
+
+ return `<${tag} ${vModel} ${size} ${showAlpha} ${colorFormat} ${disabled}>${tag}>`
+ },
+ 'el-upload': el => {
+ const { tag } = el.__config__
+ const disabled = el.disabled ? ':disabled=\'true\'' : ''
+ const action = el.action ? `:action="${el.__vModel__}Action"` : ''
+ const multiple = el.multiple ? 'multiple' : ''
+ const listType = el['list-type'] !== 'text' ? `list-type="${el['list-type']}"` : ''
+ const accept = el.accept ? `accept="${el.accept}"` : ''
+ const name = el.name !== 'file' ? `name="${el.name}"` : ''
+ const autoUpload = el['auto-upload'] === false ? ':auto-upload="false"' : ''
+ const beforeUpload = `:before-upload="${el.__vModel__}BeforeUpload"`
+ const fileList = `:file-list="${el.__vModel__}fileList"`
+ const ref = `ref="${el.__vModel__}"`
+ let child = buildElUploadChild(el)
+
+ if (child) child = `\n${child}\n` // 换行
+ return `<${tag} ${ref} ${fileList} ${action} ${autoUpload} ${multiple} ${beforeUpload} ${listType} ${accept} ${name} ${disabled}>${child}${tag}>`
+ },
+ tinymce: el => {
+ const { tag, vModel, placeholder } = attrBuilder(el)
+ const height = el.height ? `:height="${el.height}"` : ''
+ const branding = el.branding ? `:branding="${el.branding}"` : ''
+ return `<${tag} ${vModel} ${placeholder} ${height} ${branding}>${tag}>`
+ }
+}
+
+function attrBuilder(el) {
+ return {
+ tag: el.__config__.tag,
+ vModel: `v-model="${confGlobal.formModel}.${el.__vModel__}"`,
+ clearable: el.clearable ? 'clearable' : '',
+ placeholder: el.placeholder ? `placeholder="${el.placeholder}"` : '',
+ width: el.style && el.style.width ? ':style="{width: \'100%\'}"' : '',
+ disabled: el.disabled ? ':disabled=\'true\'' : ''
+ }
+}
+
+// el-buttin 子级
+function buildElButtonChild(scheme) {
+ const children = []
+ const slot = scheme.__slot__ || {}
+ if (slot.default) {
+ children.push(slot.default)
+ }
+ return children.join('\n')
+}
+
+// el-input 子级
+function buildElInputChild(scheme) {
+ const children = []
+ const slot = scheme.__slot__
+ if (slot && slot.prepend) {
+ children.push(`${slot.prepend}`)
+ }
+ if (slot && slot.append) {
+ children.push(`${slot.append}`)
+ }
+ return children.join('\n')
+}
+
+// el-select 子级
+function buildElSelectChild(scheme) {
+ const children = []
+ const slot = scheme.__slot__
+ if (slot && slot.options && slot.options.length) {
+ children.push(``)
+ }
+ return children.join('\n')
+}
+
+// el-radio-group 子级
+function buildElRadioGroupChild(scheme) {
+ const children = []
+ const slot = scheme.__slot__
+ const config = scheme.__config__
+ if (slot && slot.options && slot.options.length) {
+ const tag = config.optionType === 'button' ? 'el-radio-button' : 'el-radio'
+ const border = config.border ? 'border' : ''
+ children.push(`<${tag} v-for="(item, index) in ${scheme.__vModel__}Options" :key="index" :label="item.value" :disabled="item.disabled" ${border}>{{item.label}}${tag}>`)
+ }
+ return children.join('\n')
+}
+
+// el-checkbox-group 子级
+function buildElCheckboxGroupChild(scheme) {
+ const children = []
+ const slot = scheme.__slot__
+ const config = scheme.__config__
+ if (slot && slot.options && slot.options.length) {
+ const tag = config.optionType === 'button' ? 'el-checkbox-button' : 'el-checkbox'
+ const border = config.border ? 'border' : ''
+ children.push(`<${tag} v-for="(item, index) in ${scheme.__vModel__}Options" :key="index" :label="item.value" :disabled="item.disabled" ${border}>{{item.label}}${tag}>`)
+ }
+ return children.join('\n')
+}
+
+// el-upload 子级
+function buildElUploadChild(scheme) {
+ const list = []
+ const config = scheme.__config__
+ if (scheme['list-type'] === 'picture-card') list.push('')
+ else list.push(`${config.buttonText}`)
+ if (config.showTip) list.push(`
只能上传不超过 ${config.fileSize}${config.sizeUnit} 的${scheme.accept}文件
`)
+ return list.join('\n')
+}
+
+/**
+ * 组装html代码。【入口函数】
+ * @param {Object} formConfig 整个表单配置
+ * @param {String} type 生成类型,文件或弹窗等
+ */
+export function makeUpHtml(formConfig, type) {
+ const htmlList = []
+ confGlobal = formConfig
+ // 判断布局是否都沾满了24个栅格,以备后续简化代码结构
+ someSpanIsNot24 = formConfig.fields.some(item => item.__config__.span !== 24)
+ // 遍历渲染每个组件成html
+ formConfig.fields.forEach(el => {
+ htmlList.push(layouts[el.__config__.layout](el))
+ })
+ const htmlStr = htmlList.join('\n')
+ // 将组件代码放进form标签
+ let temp = buildFormTemplate(formConfig, htmlStr, type)
+ // dialog标签包裹代码
+ if (type === 'dialog') {
+ temp = dialogWrapper(temp)
+ }
+ confGlobal = null
+ return temp
+}
diff --git a/src/components/generator/js.js b/src/components/generator/js.js
new file mode 100644
index 00000000..f1605937
--- /dev/null
+++ b/src/components/generator/js.js
@@ -0,0 +1,271 @@
+import { isArray } from 'util'
+import { exportDefault, titleCase, deepClone } from '@/utils'
+import ruleTrigger from './ruleTrigger'
+
+const units = {
+ KB: '1024',
+ MB: '1024 / 1024',
+ GB: '1024 / 1024 / 1024'
+}
+let confGlobal
+const inheritAttrs = {
+ file: '',
+ dialog: 'inheritAttrs: false,'
+}
+
+/**
+ * 组装js 【入口函数】
+ * @param {Object} formConfig 整个表单配置
+ * @param {String} type 生成类型,文件或弹窗等
+ */
+export function makeUpJs(formConfig, type) {
+ confGlobal = formConfig = deepClone(formConfig)
+ const dataList = []
+ const ruleList = []
+ const optionsList = []
+ const propsList = []
+ const methodList = mixinMethod(type)
+ const uploadVarList = []
+ const created = []
+
+ formConfig.fields.forEach(el => {
+ buildAttributes(el, dataList, ruleList, optionsList, methodList, propsList, uploadVarList, created)
+ })
+
+ const script = buildexport(
+ formConfig,
+ type,
+ dataList.join('\n'),
+ ruleList.join('\n'),
+ optionsList.join('\n'),
+ uploadVarList.join('\n'),
+ propsList.join('\n'),
+ methodList.join('\n'),
+ created.join('\n')
+ )
+ confGlobal = null
+ return script
+}
+
+// 构建组件属性
+function buildAttributes(scheme, dataList, ruleList, optionsList, methodList, propsList, uploadVarList, created) {
+ const config = scheme.__config__
+ const slot = scheme.__slot__
+ buildData(scheme, dataList)
+ buildRules(scheme, ruleList)
+
+ // 特殊处理options属性
+ if (scheme.options || (slot && slot.options && slot.options.length)) {
+ buildOptions(scheme, optionsList)
+ if (config.dataType === 'dynamic') {
+ const model = `${scheme.__vModel__}Options`
+ const options = titleCase(model)
+ const methodName = `get${options}`
+ buildOptionMethod(methodName, model, methodList, scheme)
+ callInCreated(methodName, created)
+ }
+ }
+
+ // 处理props
+ if (scheme.props && scheme.props.props) {
+ buildProps(scheme, propsList)
+ }
+
+ // 处理el-upload的action
+ if (scheme.action && config.tag === 'el-upload') {
+ uploadVarList.push(
+ `${scheme.__vModel__}Action: '${scheme.action}',
+ ${scheme.__vModel__}fileList: [],`
+ )
+ methodList.push(buildBeforeUpload(scheme))
+ // 非自动上传时,生成手动上传的函数
+ if (!scheme['auto-upload']) {
+ methodList.push(buildSubmitUpload(scheme))
+ }
+ }
+
+ // 构建子级组件属性
+ if (config.children) {
+ config.children.forEach(item => {
+ buildAttributes(item, dataList, ruleList, optionsList, methodList, propsList, uploadVarList, created)
+ })
+ }
+}
+
+// 在Created调用函数
+function callInCreated(methodName, created) {
+ created.push(`this.${methodName}()`)
+}
+
+// 混入处理函数
+function mixinMethod(type) {
+ const list = []; const
+ minxins = {
+ file: confGlobal.formBtns ? {
+ submitForm: `submitForm() {
+ this.$refs['${confGlobal.formRef}'].validate(valid => {
+ if(!valid) return
+ // TODO 提交表单
+ })
+ },`,
+ resetForm: `resetForm() {
+ this.$refs['${confGlobal.formRef}'].resetFields()
+ },`
+ } : null,
+ dialog: {
+ onOpen: 'onOpen() {},',
+ onClose: `onClose() {
+ this.$refs['${confGlobal.formRef}'].resetFields()
+ },`,
+ close: `close() {
+ this.$emit('update:visible', false)
+ },`,
+ handelConfirm: `handelConfirm() {
+ this.$refs['${confGlobal.formRef}'].validate(valid => {
+ if(!valid) return
+ this.close()
+ })
+ },`
+ }
+ }
+
+ const methods = minxins[type]
+ if (methods) {
+ Object.keys(methods).forEach(key => {
+ list.push(methods[key])
+ })
+ }
+
+ return list
+}
+
+// 构建data
+function buildData(scheme, dataList) {
+ const config = scheme.__config__
+ if (scheme.__vModel__ === undefined) return
+ const defaultValue = JSON.stringify(config.defaultValue)
+ dataList.push(`${scheme.__vModel__}: ${defaultValue},`)
+}
+
+// 构建校验规则
+function buildRules(scheme, ruleList) {
+ const config = scheme.__config__
+ if (scheme.__vModel__ === undefined) return
+ const rules = []
+ if (ruleTrigger[config.tag]) {
+ if (config.required) {
+ const type = isArray(config.defaultValue) ? 'type: \'array\',' : ''
+ let message = isArray(config.defaultValue) ? `请至少选择一个${config.label}` : scheme.placeholder
+ if (message === undefined) message = `${config.label}不能为空`
+ rules.push(`{ required: true, ${type} message: '${message}', trigger: '${ruleTrigger[config.tag]}' }`)
+ }
+ if (config.regList && isArray(config.regList)) {
+ config.regList.forEach(item => {
+ if (item.pattern) {
+ rules.push(
+ `{ pattern: ${eval(item.pattern)}, message: '${item.message}', trigger: '${ruleTrigger[config.tag]}' }`
+ )
+ }
+ })
+ }
+ ruleList.push(`${scheme.__vModel__}: [${rules.join(',')}],`)
+ }
+}
+
+// 构建options
+function buildOptions(scheme, optionsList) {
+ if (scheme.__vModel__ === undefined) return
+ // el-cascader直接有options属性,其他组件都是定义在slot中,所以有两处判断
+ let { options } = scheme
+ if (!options) options = scheme.__slot__.options
+ if (scheme.__config__.dataType === 'dynamic') { options = [] }
+ const str = `${scheme.__vModel__}Options: ${JSON.stringify(options)},`
+ optionsList.push(str)
+}
+
+function buildProps(scheme, propsList) {
+ const str = `${scheme.__vModel__}Props: ${JSON.stringify(scheme.props.props)},`
+ propsList.push(str)
+}
+
+// el-upload的BeforeUpload
+function buildBeforeUpload(scheme) {
+ const config = scheme.__config__
+ const unitNum = units[config.sizeUnit]; let rightSizeCode = ''; let acceptCode = ''; const
+ returnList = []
+ if (config.fileSize) {
+ rightSizeCode = `let isRightSize = file.size / ${unitNum} < ${config.fileSize}
+ if(!isRightSize){
+ this.$message.error('文件大小超过 ${config.fileSize}${config.sizeUnit}')
+ }`
+ returnList.push('isRightSize')
+ }
+ if (scheme.accept) {
+ acceptCode = `let isAccept = new RegExp('${scheme.accept}').test(file.type)
+ if(!isAccept){
+ this.$message.error('应该选择${scheme.accept}类型的文件')
+ }`
+ returnList.push('isAccept')
+ }
+ const str = `${scheme.__vModel__}BeforeUpload(file) {
+ ${rightSizeCode}
+ ${acceptCode}
+ return ${returnList.join('&&')}
+ },`
+ return returnList.length ? str : ''
+}
+
+// el-upload的submit
+function buildSubmitUpload(scheme) {
+ const str = `submitUpload() {
+ this.$refs['${scheme.__vModel__}'].submit()
+ },`
+ return str
+}
+
+function buildOptionMethod(methodName, model, methodList, scheme) {
+ const config = scheme.__config__
+ const str = `${methodName}() {
+ // 注意:this.$axios是通过Vue.prototype.$axios = axios挂载产生的
+ this.$axios({
+ method: '${config.method}',
+ url: '${config.url}'
+ }).then(resp => {
+ var { data } = resp
+ this.${model} = data.${config.dataPath}
+ })
+ },`
+ methodList.push(str)
+}
+
+// js整体拼接
+function buildexport(conf, type, data, rules, selectOptions, uploadVar, props, methods, created) {
+ const str = `${exportDefault}{
+ ${inheritAttrs[type]}
+ components: {},
+ props: [],
+ data () {
+ return {
+ ${conf.formModel}: {
+ ${data}
+ },
+ ${conf.formRules}: {
+ ${rules}
+ },
+ ${uploadVar}
+ ${selectOptions}
+ ${props}
+ }
+ },
+ computed: {},
+ watch: {},
+ created () {
+ ${created}
+ },
+ mounted () {},
+ methods: {
+ ${methods}
+ }
+}`
+ return str
+}
diff --git a/src/components/generator/ruleTrigger.js b/src/components/generator/ruleTrigger.js
new file mode 100644
index 00000000..3c161b5d
--- /dev/null
+++ b/src/components/generator/ruleTrigger.js
@@ -0,0 +1,16 @@
+/**
+ * 用于生成表单校验,指定正则规则的触发方式。
+ * 未在此处声明无触发方式的组件将不生成rule!!
+ */
+export default {
+ 'el-input': 'blur',
+ 'el-input-number': 'blur',
+ 'el-select': 'change',
+ 'el-radio-group': 'change',
+ 'el-checkbox-group': 'change',
+ 'el-cascader': 'change',
+ 'el-time-picker': 'change',
+ 'el-date-picker': 'change',
+ 'el-rate': 'change',
+ tinymce: 'blur'
+}
diff --git a/src/components/tinymce/README.md b/src/components/tinymce/README.md
new file mode 100644
index 00000000..65c01e21
--- /dev/null
+++ b/src/components/tinymce/README.md
@@ -0,0 +1,3 @@
+## 简介
+富文本编辑器tinymce的一个vue版本封装。使用cdn动态脚本引入的方式加载。
+
diff --git a/src/components/tinymce/config.js b/src/components/tinymce/config.js
new file mode 100644
index 00000000..fc615544
--- /dev/null
+++ b/src/components/tinymce/config.js
@@ -0,0 +1,8 @@
+/* eslint-disable max-len */
+
+export const plugins = [
+ 'advlist anchor autolink autosave code codesample directionality emoticons fullscreen hr image imagetools insertdatetime link lists media nonbreaking noneditable pagebreak paste preview print save searchreplace spellchecker tabfocus table template textpattern visualblocks visualchars wordcount'
+]
+export const toolbar = [
+ 'code searchreplace bold italic underline strikethrough alignleft aligncenter alignright outdent indent blockquote removeformat subscript superscript codesample hr bullist numlist link image charmap preview anchor pagebreak insertdatetime media table emoticons forecolor backcolor fullscreen'
+]
diff --git a/src/components/tinymce/example/Index.vue b/src/components/tinymce/example/Index.vue
new file mode 100644
index 00000000..e5a9f65d
--- /dev/null
+++ b/src/components/tinymce/example/Index.vue
@@ -0,0 +1,38 @@
+
+
+
+
+
+
+
diff --git a/src/components/tinymce/index.js b/src/components/tinymce/index.js
new file mode 100644
index 00000000..1e831794
--- /dev/null
+++ b/src/components/tinymce/index.js
@@ -0,0 +1,3 @@
+import Index from './index.vue'
+
+export default Index
diff --git a/src/components/tinymce/index.vue b/src/components/tinymce/index.vue
new file mode 100644
index 00000000..2eda1a1c
--- /dev/null
+++ b/src/components/tinymce/index.vue
@@ -0,0 +1,88 @@
+
+
+
+
+
diff --git a/src/components/tinymce/package.json b/src/components/tinymce/package.json
new file mode 100644
index 00000000..3d2d2d4b
--- /dev/null
+++ b/src/components/tinymce/package.json
@@ -0,0 +1,28 @@
+{
+ "name": "form-gen-tinymce",
+ "version": "1.0.0",
+ "description": "富文本编辑器tinymce的一个vue版本封装。使用cdn动态脚本引入的方式加载。",
+ "main": "lib/form-gen-tinymce.umd.js",
+ "directories": {
+ "example": "example"
+ },
+ "scripts": {
+ "test": "echo \"Error: no test specified\" && exit 1"
+ },
+ "repository": {
+ "type": "git",
+ "url": "git+https://github.com/JakHuang/form-generator.git"
+ },
+ "keywords": [
+ "tinymce-vue"
+ ],
+ "dependencies": {
+ "throttle-debounce": "^2.1.0"
+ },
+ "author": "jakHuang",
+ "license": "MIT",
+ "bugs": {
+ "url": "https://github.com/JakHuang/form-generator/issues"
+ },
+ "homepage": "https://github.com/JakHuang/form-generator/blob/dev/src/components/tinymce"
+}
diff --git a/src/components/tinymce/zh_CN.js b/src/components/tinymce/zh_CN.js
new file mode 100644
index 00000000..4f494d63
--- /dev/null
+++ b/src/components/tinymce/zh_CN.js
@@ -0,0 +1,420 @@
+/* eslint-disable */
+tinymce.addI18n('zh_CN',{
+"Redo": "\u91cd\u505a",
+"Undo": "\u64a4\u9500",
+"Cut": "\u526a\u5207",
+"Copy": "\u590d\u5236",
+"Paste": "\u7c98\u8d34",
+"Select all": "\u5168\u9009",
+"New document": "\u65b0\u6587\u4ef6",
+"Ok": "\u786e\u5b9a",
+"Cancel": "\u53d6\u6d88",
+"Visual aids": "\u7f51\u683c\u7ebf",
+"Bold": "\u7c97\u4f53",
+"Italic": "\u659c\u4f53",
+"Underline": "\u4e0b\u5212\u7ebf",
+"Strikethrough": "\u5220\u9664\u7ebf",
+"Superscript": "\u4e0a\u6807",
+"Subscript": "\u4e0b\u6807",
+"Clear formatting": "\u6e05\u9664\u683c\u5f0f",
+"Align left": "\u5de6\u8fb9\u5bf9\u9f50",
+"Align center": "\u4e2d\u95f4\u5bf9\u9f50",
+"Align right": "\u53f3\u8fb9\u5bf9\u9f50",
+"Justify": "\u4e24\u7aef\u5bf9\u9f50",
+"Bullet list": "\u9879\u76ee\u7b26\u53f7",
+"Numbered list": "\u7f16\u53f7\u5217\u8868",
+"Decrease indent": "\u51cf\u5c11\u7f29\u8fdb",
+"Increase indent": "\u589e\u52a0\u7f29\u8fdb",
+"Close": "\u5173\u95ed",
+"Formats": "\u683c\u5f0f",
+"Your browser doesn't support direct access to the clipboard. Please use the Ctrl+X\/C\/V keyboard shortcuts instead.": "\u4f60\u7684\u6d4f\u89c8\u5668\u4e0d\u652f\u6301\u6253\u5f00\u526a\u8d34\u677f\uff0c\u8bf7\u4f7f\u7528Ctrl+X\/C\/V\u7b49\u5feb\u6377\u952e\u3002",
+"Headers": "\u6807\u9898",
+"Header 1": "\u6807\u98981",
+"Header 2": "\u6807\u98982",
+"Header 3": "\u6807\u98983",
+"Header 4": "\u6807\u98984",
+"Header 5": "\u6807\u98985",
+"Header 6": "\u6807\u98986",
+"Headings": "\u6807\u9898",
+"Heading 1": "\u6807\u98981",
+"Heading 2": "\u6807\u98982",
+"Heading 3": "\u6807\u98983",
+"Heading 4": "\u6807\u98984",
+"Heading 5": "\u6807\u98985",
+"Heading 6": "\u6807\u98986",
+"Preformatted": "\u9884\u5148\u683c\u5f0f\u5316\u7684",
+"Div": "Div",
+"Pre": "Pre",
+"Code": "\u4ee3\u7801",
+"Paragraph": "\u6bb5\u843d",
+"Blockquote": "\u5f15\u6587\u533a\u5757",
+"Inline": "\u6587\u672c",
+"Blocks": "\u57fa\u5757",
+"Paste is now in plain text mode. Contents will now be pasted as plain text until you toggle this option off.": "\u5f53\u524d\u4e3a\u7eaf\u6587\u672c\u7c98\u8d34\u6a21\u5f0f\uff0c\u518d\u6b21\u70b9\u51fb\u53ef\u4ee5\u56de\u5230\u666e\u901a\u7c98\u8d34\u6a21\u5f0f\u3002",
+"Fonts": "\u5b57\u4f53",
+"Font Sizes": "\u5b57\u53f7",
+"Class": "\u7c7b\u578b",
+"Browse for an image": "\u6d4f\u89c8\u56fe\u50cf",
+"OR": "\u6216",
+"Drop an image here": "\u62d6\u653e\u4e00\u5f20\u56fe\u50cf\u81f3\u6b64",
+"Upload": "\u4e0a\u4f20",
+"Block": "\u5757",
+"Align": "\u5bf9\u9f50",
+"Default": "\u9ed8\u8ba4",
+"Circle": "\u7a7a\u5fc3\u5706",
+"Disc": "\u5b9e\u5fc3\u5706",
+"Square": "\u65b9\u5757",
+"Lower Alpha": "\u5c0f\u5199\u82f1\u6587\u5b57\u6bcd",
+"Lower Greek": "\u5c0f\u5199\u5e0c\u814a\u5b57\u6bcd",
+"Lower Roman": "\u5c0f\u5199\u7f57\u9a6c\u5b57\u6bcd",
+"Upper Alpha": "\u5927\u5199\u82f1\u6587\u5b57\u6bcd",
+"Upper Roman": "\u5927\u5199\u7f57\u9a6c\u5b57\u6bcd",
+"Anchor...": "\u951a\u70b9...",
+"Name": "\u540d\u79f0",
+"Id": "\u6807\u8bc6\u7b26",
+"Id should start with a letter, followed only by letters, numbers, dashes, dots, colons or underscores.": "\u6807\u8bc6\u7b26\u5e94\u8be5\u4ee5\u5b57\u6bcd\u5f00\u5934\uff0c\u540e\u8ddf\u5b57\u6bcd\u3001\u6570\u5b57\u3001\u7834\u6298\u53f7\u3001\u70b9\u3001\u5192\u53f7\u6216\u4e0b\u5212\u7ebf\u3002",
+"You have unsaved changes are you sure you want to navigate away?": "\u4f60\u8fd8\u6709\u6587\u6863\u5c1a\u672a\u4fdd\u5b58\uff0c\u786e\u5b9a\u8981\u79bb\u5f00\uff1f",
+"Restore last draft": "\u6062\u590d\u4e0a\u6b21\u7684\u8349\u7a3f",
+"Special character...": "\u7279\u6b8a\u5b57\u7b26...",
+"Source code": "\u6e90\u4ee3\u7801",
+"Insert\/Edit code sample": "\u63d2\u5165\/\u7f16\u8f91\u4ee3\u7801\u793a\u4f8b",
+"Language": "\u8bed\u8a00",
+"Code sample...": "\u793a\u4f8b\u4ee3\u7801...",
+"Color Picker": "\u9009\u8272\u5668",
+"R": "R",
+"G": "G",
+"B": "B",
+"Left to right": "\u4ece\u5de6\u5230\u53f3",
+"Right to left": "\u4ece\u53f3\u5230\u5de6",
+"Emoticons...": "\u8868\u60c5\u7b26\u53f7...",
+"Metadata and Document Properties": "\u5143\u6570\u636e\u548c\u6587\u6863\u5c5e\u6027",
+"Title": "\u6807\u9898",
+"Keywords": "\u5173\u952e\u8bcd",
+"Description": "\u63cf\u8ff0",
+"Robots": "\u673a\u5668\u4eba",
+"Author": "\u4f5c\u8005",
+"Encoding": "\u7f16\u7801",
+"Fullscreen": "\u5168\u5c4f",
+"Action": "\u64cd\u4f5c",
+"Shortcut": "\u5feb\u6377\u952e",
+"Help": "\u5e2e\u52a9",
+"Address": "\u5730\u5740",
+"Focus to menubar": "\u79fb\u52a8\u7126\u70b9\u5230\u83dc\u5355\u680f",
+"Focus to toolbar": "\u79fb\u52a8\u7126\u70b9\u5230\u5de5\u5177\u680f",
+"Focus to element path": "\u79fb\u52a8\u7126\u70b9\u5230\u5143\u7d20\u8def\u5f84",
+"Focus to contextual toolbar": "\u79fb\u52a8\u7126\u70b9\u5230\u4e0a\u4e0b\u6587\u83dc\u5355",
+"Insert link (if link plugin activated)": "\u63d2\u5165\u94fe\u63a5 (\u5982\u679c\u94fe\u63a5\u63d2\u4ef6\u5df2\u6fc0\u6d3b)",
+"Save (if save plugin activated)": "\u4fdd\u5b58(\u5982\u679c\u4fdd\u5b58\u63d2\u4ef6\u5df2\u6fc0\u6d3b)",
+"Find (if searchreplace plugin activated)": "\u67e5\u627e(\u5982\u679c\u67e5\u627e\u66ff\u6362\u63d2\u4ef6\u5df2\u6fc0\u6d3b)",
+"Plugins installed ({0}):": "\u5df2\u5b89\u88c5\u63d2\u4ef6 ({0}):",
+"Premium plugins:": "\u4f18\u79c0\u63d2\u4ef6\uff1a",
+"Learn more...": "\u4e86\u89e3\u66f4\u591a...",
+"You are using {0}": "\u4f60\u6b63\u5728\u4f7f\u7528 {0}",
+"Plugins": "\u63d2\u4ef6",
+"Handy Shortcuts": "\u5feb\u6377\u952e",
+"Horizontal line": "\u6c34\u5e73\u5206\u5272\u7ebf",
+"Insert\/edit image": "\u63d2\u5165\/\u7f16\u8f91\u56fe\u7247",
+"Image description": "\u56fe\u7247\u63cf\u8ff0",
+"Source": "\u5730\u5740",
+"Dimensions": "\u5927\u5c0f",
+"Constrain proportions": "\u4fdd\u6301\u7eb5\u6a2a\u6bd4",
+"General": "\u666e\u901a",
+"Advanced": "\u9ad8\u7ea7",
+"Style": "\u6837\u5f0f",
+"Vertical space": "\u5782\u76f4\u8fb9\u8ddd",
+"Horizontal space": "\u6c34\u5e73\u8fb9\u8ddd",
+"Border": "\u8fb9\u6846",
+"Insert image": "\u63d2\u5165\u56fe\u7247",
+"Image...": "\u56fe\u7247...",
+"Image list": "\u56fe\u7247\u5217\u8868",
+"Rotate counterclockwise": "\u9006\u65f6\u9488\u65cb\u8f6c",
+"Rotate clockwise": "\u987a\u65f6\u9488\u65cb\u8f6c",
+"Flip vertically": "\u5782\u76f4\u7ffb\u8f6c",
+"Flip horizontally": "\u6c34\u5e73\u7ffb\u8f6c",
+"Edit image": "\u7f16\u8f91\u56fe\u7247",
+"Image options": "\u56fe\u7247\u9009\u9879",
+"Zoom in": "\u653e\u5927",
+"Zoom out": "\u7f29\u5c0f",
+"Crop": "\u88c1\u526a",
+"Resize": "\u8c03\u6574\u5927\u5c0f",
+"Orientation": "\u65b9\u5411",
+"Brightness": "\u4eae\u5ea6",
+"Sharpen": "\u9510\u5316",
+"Contrast": "\u5bf9\u6bd4\u5ea6",
+"Color levels": "\u989c\u8272\u5c42\u6b21",
+"Gamma": "\u4f3d\u9a6c\u503c",
+"Invert": "\u53cd\u8f6c",
+"Apply": "\u5e94\u7528",
+"Back": "\u540e\u9000",
+"Insert date\/time": "\u63d2\u5165\u65e5\u671f\/\u65f6\u95f4",
+"Date\/time": "\u65e5\u671f\/\u65f6\u95f4",
+"Insert\/Edit Link": "\u63d2\u5165\/\u7f16\u8f91\u94fe\u63a5",
+"Insert\/edit link": "\u63d2\u5165\/\u7f16\u8f91\u94fe\u63a5",
+"Text to display": "\u663e\u793a\u6587\u5b57",
+"Url": "\u5730\u5740",
+"Open link in...": "\u94fe\u63a5\u6253\u5f00\u4f4d\u7f6e...",
+"Current window": "\u5f53\u524d\u7a97\u53e3",
+"None": "\u65e0",
+"New window": "\u5728\u65b0\u7a97\u53e3\u6253\u5f00",
+"Remove link": "\u5220\u9664\u94fe\u63a5",
+"Anchors": "\u951a\u70b9",
+"Link...": "\u94fe\u63a5...",
+"Paste or type a link": "\u7c98\u8d34\u6216\u8f93\u5165\u94fe\u63a5",
+"The URL you entered seems to be an email address. Do you want to add the required mailto: prefix?": "\u4f60\u6240\u586b\u5199\u7684URL\u5730\u5740\u4e3a\u90ae\u4ef6\u5730\u5740\uff0c\u9700\u8981\u52a0\u4e0amailto:\u524d\u7f00\u5417\uff1f",
+"The URL you entered seems to be an external link. Do you want to add the required http:\/\/ prefix?": "\u4f60\u6240\u586b\u5199\u7684URL\u5730\u5740\u5c5e\u4e8e\u5916\u90e8\u94fe\u63a5\uff0c\u9700\u8981\u52a0\u4e0ahttp:\/\/:\u524d\u7f00\u5417\uff1f",
+"Link list": "\u94fe\u63a5\u5217\u8868",
+"Insert video": "\u63d2\u5165\u89c6\u9891",
+"Insert\/edit video": "\u63d2\u5165\/\u7f16\u8f91\u89c6\u9891",
+"Insert\/edit media": "\u63d2\u5165\/\u7f16\u8f91\u5a92\u4f53",
+"Alternative source": "\u955c\u50cf",
+"Alternative source URL": "\u66ff\u4ee3\u6765\u6e90\u7f51\u5740",
+"Media poster (Image URL)": "\u5c01\u9762(\u56fe\u7247\u5730\u5740)",
+"Paste your embed code below:": "\u5c06\u5185\u5d4c\u4ee3\u7801\u7c98\u8d34\u5728\u4e0b\u9762:",
+"Embed": "\u5185\u5d4c",
+"Media...": "\u591a\u5a92\u4f53...",
+"Nonbreaking space": "\u4e0d\u95f4\u65ad\u7a7a\u683c",
+"Page break": "\u5206\u9875\u7b26",
+"Paste as text": "\u7c98\u8d34\u4e3a\u6587\u672c",
+"Preview": "\u9884\u89c8",
+"Print...": "\u6253\u5370...",
+"Save": "\u4fdd\u5b58",
+"Find": "\u67e5\u627e",
+"Replace with": "\u66ff\u6362\u4e3a",
+"Replace": "\u66ff\u6362",
+"Replace all": "\u5168\u90e8\u66ff\u6362",
+"Previous": "\u4e0a\u4e00\u4e2a",
+"Next": "\u4e0b\u4e00\u4e2a",
+"Find and replace...": "\u67e5\u627e\u5e76\u66ff\u6362...",
+"Could not find the specified string.": "\u672a\u627e\u5230\u641c\u7d22\u5185\u5bb9.",
+"Match case": "\u533a\u5206\u5927\u5c0f\u5199",
+"Find whole words only": "\u5168\u5b57\u5339\u914d",
+"Spell check": "\u62fc\u5199\u68c0\u67e5",
+"Ignore": "\u5ffd\u7565",
+"Ignore all": "\u5168\u90e8\u5ffd\u7565",
+"Finish": "\u5b8c\u6210",
+"Add to Dictionary": "\u6dfb\u52a0\u5230\u5b57\u5178",
+"Insert table": "\u63d2\u5165\u8868\u683c",
+"Table properties": "\u8868\u683c\u5c5e\u6027",
+"Delete table": "\u5220\u9664\u8868\u683c",
+"Cell": "\u5355\u5143\u683c",
+"Row": "\u884c",
+"Column": "\u5217",
+"Cell properties": "\u5355\u5143\u683c\u5c5e\u6027",
+"Merge cells": "\u5408\u5e76\u5355\u5143\u683c",
+"Split cell": "\u62c6\u5206\u5355\u5143\u683c",
+"Insert row before": "\u5728\u4e0a\u65b9\u63d2\u5165",
+"Insert row after": "\u5728\u4e0b\u65b9\u63d2\u5165",
+"Delete row": "\u5220\u9664\u884c",
+"Row properties": "\u884c\u5c5e\u6027",
+"Cut row": "\u526a\u5207\u884c",
+"Copy row": "\u590d\u5236\u884c",
+"Paste row before": "\u7c98\u8d34\u5230\u4e0a\u65b9",
+"Paste row after": "\u7c98\u8d34\u5230\u4e0b\u65b9",
+"Insert column before": "\u5728\u5de6\u4fa7\u63d2\u5165",
+"Insert column after": "\u5728\u53f3\u4fa7\u63d2\u5165",
+"Delete column": "\u5220\u9664\u5217",
+"Cols": "\u5217",
+"Rows": "\u884c",
+"Width": "\u5bbd",
+"Height": "\u9ad8",
+"Cell spacing": "\u5355\u5143\u683c\u5916\u95f4\u8ddd",
+"Cell padding": "\u5355\u5143\u683c\u5185\u8fb9\u8ddd",
+"Show caption": "\u663e\u793a\u6807\u9898",
+"Left": "\u5de6\u5bf9\u9f50",
+"Center": "\u5c45\u4e2d",
+"Right": "\u53f3\u5bf9\u9f50",
+"Cell type": "\u5355\u5143\u683c\u7c7b\u578b",
+"Scope": "\u8303\u56f4",
+"Alignment": "\u5bf9\u9f50\u65b9\u5f0f",
+"H Align": "\u6c34\u5e73\u5bf9\u9f50",
+"V Align": "\u5782\u76f4\u5bf9\u9f50",
+"Top": "\u9876\u90e8\u5bf9\u9f50",
+"Middle": "\u5782\u76f4\u5c45\u4e2d",
+"Bottom": "\u5e95\u90e8\u5bf9\u9f50",
+"Header cell": "\u8868\u5934\u5355\u5143\u683c",
+"Row group": "\u884c\u7ec4",
+"Column group": "\u5217\u7ec4",
+"Row type": "\u884c\u7c7b\u578b",
+"Header": "\u8868\u5934",
+"Body": "\u8868\u4f53",
+"Footer": "\u8868\u5c3e",
+"Border color": "\u8fb9\u6846\u989c\u8272",
+"Insert template...": "\u63d2\u5165\u6a21\u677f...",
+"Templates": "\u6a21\u677f",
+"Template": "\u6a21\u677f",
+"Text color": "\u6587\u5b57\u989c\u8272",
+"Background color": "\u80cc\u666f\u8272",
+"Custom...": "\u81ea\u5b9a\u4e49...",
+"Custom color": "\u81ea\u5b9a\u4e49\u989c\u8272",
+"No color": "\u65e0",
+"Remove color": "\u79fb\u9664\u989c\u8272",
+"Table of Contents": "\u5185\u5bb9\u5217\u8868",
+"Show blocks": "\u663e\u793a\u533a\u5757\u8fb9\u6846",
+"Show invisible characters": "\u663e\u793a\u4e0d\u53ef\u89c1\u5b57\u7b26",
+"Word count": "\u5b57\u6570",
+"Count": "\u8ba1\u6570",
+"Document": "\u6587\u6863",
+"Selection": "\u9009\u62e9",
+"Words": "\u5355\u8bcd",
+"Words: {0}": "\u5b57\u6570\uff1a{0}",
+"{0} words": "{0} \u5b57",
+"File": "\u6587\u4ef6",
+"Edit": "\u7f16\u8f91",
+"Insert": "\u63d2\u5165",
+"View": "\u89c6\u56fe",
+"Format": "\u683c\u5f0f",
+"Table": "\u8868\u683c",
+"Tools": "\u5de5\u5177",
+"Powered by {0}": "\u7531{0}\u9a71\u52a8",
+"Rich Text Area. Press ALT-F9 for menu. Press ALT-F10 for toolbar. Press ALT-0 for help": "\u5728\u7f16\u8f91\u533a\u6309ALT-F9\u6253\u5f00\u83dc\u5355\uff0c\u6309ALT-F10\u6253\u5f00\u5de5\u5177\u680f\uff0c\u6309ALT-0\u67e5\u770b\u5e2e\u52a9",
+"Image title": "\u56fe\u7247\u6807\u9898",
+"Border width": "\u8fb9\u6846\u5bbd\u5ea6",
+"Border style": "\u8fb9\u6846\u6837\u5f0f",
+"Error": "\u9519\u8bef",
+"Warn": "\u8b66\u544a",
+"Valid": "\u6709\u6548",
+"To open the popup, press Shift+Enter": "\u6309Shitf+Enter\u952e\u6253\u5f00\u5bf9\u8bdd\u6846",
+"Rich Text Area. Press ALT-0 for help.": "\u7f16\u8f91\u533a\u3002\u6309Alt+0\u952e\u6253\u5f00\u5e2e\u52a9\u3002",
+"System Font": "\u7cfb\u7edf\u5b57\u4f53",
+"Failed to upload image: {0}": "\u56fe\u7247\u4e0a\u4f20\u5931\u8d25: {0}",
+"Failed to load plugin: {0} from url {1}": "\u63d2\u4ef6\u52a0\u8f7d\u5931\u8d25: {0} \u6765\u81ea\u94fe\u63a5 {1}",
+"Failed to load plugin url: {0}": "\u63d2\u4ef6\u52a0\u8f7d\u5931\u8d25 \u94fe\u63a5: {0}",
+"Failed to initialize plugin: {0}": "\u63d2\u4ef6\u521d\u59cb\u5316\u5931\u8d25: {0}",
+"example": "\u793a\u4f8b",
+"Search": "\u641c\u7d22",
+"All": "\u5168\u90e8",
+"Currency": "\u8d27\u5e01",
+"Text": "\u6587\u5b57",
+"Quotations": "\u5f15\u7528",
+"Mathematical": "\u6570\u5b66",
+"Extended Latin": "\u62c9\u4e01\u8bed\u6269\u5145",
+"Symbols": "\u7b26\u53f7",
+"Arrows": "\u7bad\u5934",
+"User Defined": "\u81ea\u5b9a\u4e49",
+"dollar sign": "\u7f8e\u5143\u7b26\u53f7",
+"currency sign": "\u8d27\u5e01\u7b26\u53f7",
+"euro-currency sign": "\u6b27\u5143\u7b26\u53f7",
+"colon sign": "\u5192\u53f7",
+"cruzeiro sign": "\u514b\u9c81\u8d5b\u7f57\u5e01\u7b26\u53f7",
+"french franc sign": "\u6cd5\u90ce\u7b26\u53f7",
+"lira sign": "\u91cc\u62c9\u7b26\u53f7",
+"mill sign": "\u5bc6\u5c14\u7b26\u53f7",
+"naira sign": "\u5948\u62c9\u7b26\u53f7",
+"peseta sign": "\u6bd4\u585e\u5854\u7b26\u53f7",
+"rupee sign": "\u5362\u6bd4\u7b26\u53f7",
+"won sign": "\u97e9\u5143\u7b26\u53f7",
+"new sheqel sign": "\u65b0\u8c22\u514b\u5c14\u7b26\u53f7",
+"dong sign": "\u8d8a\u5357\u76fe\u7b26\u53f7",
+"kip sign": "\u8001\u631d\u57fa\u666e\u7b26\u53f7",
+"tugrik sign": "\u56fe\u683c\u91cc\u514b\u7b26\u53f7",
+"drachma sign": "\u5fb7\u62c9\u514b\u9a6c\u7b26\u53f7",
+"german penny symbol": "\u5fb7\u56fd\u4fbf\u58eb\u7b26\u53f7",
+"peso sign": "\u6bd4\u7d22\u7b26\u53f7",
+"guarani sign": "\u74dc\u62c9\u5c3c\u7b26\u53f7",
+"austral sign": "\u6fb3\u5143\u7b26\u53f7",
+"hryvnia sign": "\u683c\u91cc\u592b\u5c3c\u4e9a\u7b26\u53f7",
+"cedi sign": "\u585e\u5730\u7b26\u53f7",
+"livre tournois sign": "\u91cc\u5f17\u5f17\u5c14\u7b26\u53f7",
+"spesmilo sign": "spesmilo\u7b26\u53f7",
+"tenge sign": "\u575a\u6208\u7b26\u53f7",
+"indian rupee sign": "\u5370\u5ea6\u5362\u6bd4",
+"turkish lira sign": "\u571f\u8033\u5176\u91cc\u62c9",
+"nordic mark sign": "\u5317\u6b27\u9a6c\u514b",
+"manat sign": "\u9a6c\u7eb3\u7279\u7b26\u53f7",
+"ruble sign": "\u5362\u5e03\u7b26\u53f7",
+"yen character": "\u65e5\u5143\u5b57\u6837",
+"yuan character": "\u4eba\u6c11\u5e01\u5143\u5b57\u6837",
+"yuan character, in hong kong and taiwan": "\u5143\u5b57\u6837\uff08\u6e2f\u53f0\u5730\u533a\uff09",
+"yen\/yuan character variant one": "\u5143\u5b57\u6837\uff08\u5927\u5199\uff09",
+"Loading emoticons...": "\u52a0\u8f7d\u8868\u60c5\u7b26\u53f7...",
+"Could not load emoticons": "\u4e0d\u80fd\u52a0\u8f7d\u8868\u60c5\u7b26\u53f7",
+"People": "\u4eba\u7c7b",
+"Animals and Nature": "\u52a8\u7269\u548c\u81ea\u7136",
+"Food and Drink": "\u98df\u7269\u548c\u996e\u54c1",
+"Activity": "\u6d3b\u52a8",
+"Travel and Places": "\u65c5\u6e38\u548c\u5730\u70b9",
+"Objects": "\u7269\u4ef6",
+"Flags": "\u65d7\u5e1c",
+"Characters": "\u5b57\u7b26",
+"Characters (no spaces)": "\u5b57\u7b26(\u65e0\u7a7a\u683c)",
+"{0} characters": "{0} \u4e2a\u5b57\u7b26",
+"Error: Form submit field collision.": "\u9519\u8bef: \u8868\u5355\u63d0\u4ea4\u5b57\u6bb5\u51b2\u7a81\u3002",
+"Error: No form element found.": "\u9519\u8bef: \u6ca1\u6709\u8868\u5355\u63a7\u4ef6\u3002",
+"Update": "\u66f4\u65b0",
+"Color swatch": "\u989c\u8272\u6837\u672c",
+"Turquoise": "\u9752\u7eff\u8272",
+"Green": "\u7eff\u8272",
+"Blue": "\u84dd\u8272",
+"Purple": "\u7d2b\u8272",
+"Navy Blue": "\u6d77\u519b\u84dd",
+"Dark Turquoise": "\u6df1\u84dd\u7eff\u8272",
+"Dark Green": "\u6df1\u7eff\u8272",
+"Medium Blue": "\u4e2d\u84dd\u8272",
+"Medium Purple": "\u4e2d\u7d2b\u8272",
+"Midnight Blue": "\u6df1\u84dd\u8272",
+"Yellow": "\u9ec4\u8272",
+"Orange": "\u6a59\u8272",
+"Red": "\u7ea2\u8272",
+"Light Gray": "\u6d45\u7070\u8272",
+"Gray": "\u7070\u8272",
+"Dark Yellow": "\u6697\u9ec4\u8272",
+"Dark Orange": "\u6df1\u6a59\u8272",
+"Dark Red": "\u6df1\u7ea2\u8272",
+"Medium Gray": "\u4e2d\u7070\u8272",
+"Dark Gray": "\u6df1\u7070\u8272",
+"Light Green": "\u6d45\u7eff\u8272",
+"Light Yellow": "\u6d45\u9ec4\u8272",
+"Light Red": "\u6d45\u7ea2\u8272",
+"Light Purple": "\u6d45\u7d2b\u8272",
+"Light Blue": "\u6d45\u84dd\u8272",
+"Dark Purple": "\u6df1\u7d2b\u8272",
+"Dark Blue": "\u6df1\u84dd\u8272",
+"Black": "\u9ed1\u8272",
+"White": "\u767d\u8272",
+"Switch to or from fullscreen mode": "\u5207\u6362\u5168\u5c4f\u6a21\u5f0f",
+"Open help dialog": "\u6253\u5f00\u5e2e\u52a9\u5bf9\u8bdd\u6846",
+"history": "\u5386\u53f2",
+"styles": "\u6837\u5f0f",
+"formatting": "\u683c\u5f0f\u5316",
+"alignment": "\u5bf9\u9f50",
+"indentation": "\u7f29\u8fdb",
+"permanent pen": "\u8bb0\u53f7\u7b14",
+"comments": "\u5907\u6ce8",
+"Format Painter": "\u683c\u5f0f\u5237",
+"Insert\/edit iframe": "\u63d2\u5165\/\u7f16\u8f91\u6846\u67b6",
+"Capitalization": "\u5927\u5199",
+"lowercase": "\u5c0f\u5199",
+"UPPERCASE": "\u5927\u5199",
+"Title Case": "\u9996\u5b57\u6bcd\u5927\u5199",
+"Permanent Pen Properties": "\u6c38\u4e45\u7b14\u5c5e\u6027",
+"Permanent pen properties...": "\u6c38\u4e45\u7b14\u5c5e\u6027...",
+"Font": "\u5b57\u4f53",
+"Size": "\u5b57\u53f7",
+"More...": "\u66f4\u591a...",
+"Spellcheck Language": "\u62fc\u5199\u68c0\u67e5\u8bed\u8a00",
+"Select...": "\u9009\u62e9...",
+"Preferences": "\u9996\u9009\u9879",
+"Yes": "\u662f",
+"No": "\u5426",
+"Keyboard Navigation": "\u952e\u76d8\u6307\u5f15",
+"Version": "\u7248\u672c",
+"Anchor": "\u951a\u70b9",
+"Special character": "\u7279\u6b8a\u7b26\u53f7",
+"Code sample": "\u4ee3\u7801\u793a\u4f8b",
+"Color": "\u989c\u8272",
+"Emoticons": "\u8868\u60c5",
+"Document properties": "\u6587\u6863\u5c5e\u6027",
+"Image": "\u56fe\u7247",
+"Insert link": "\u63d2\u5165\u94fe\u63a5",
+"Target": "\u6253\u5f00\u65b9\u5f0f",
+"Link": "\u94fe\u63a5",
+"Poster": "\u5c01\u9762",
+"Media": "\u5a92\u4f53",
+"Print": "\u6253\u5370",
+"Prev": "\u4e0a\u4e00\u4e2a",
+"Find and replace": "\u67e5\u627e\u548c\u66ff\u6362",
+"Whole words": "\u5168\u5b57\u5339\u914d",
+"Spellcheck": "\u62fc\u5199\u68c0\u67e5",
+"Caption": "\u6807\u9898",
+"Insert template": "\u63d2\u5165\u6a21\u677f"
+});
\ No newline at end of file
diff --git a/src/utils/loadBeautifier.js b/src/utils/loadBeautifier.js
new file mode 100644
index 00000000..cec9ccd2
--- /dev/null
+++ b/src/utils/loadBeautifier.js
@@ -0,0 +1,28 @@
+import loadScript from './loadScript'
+import ELEMENT from 'element-ui'
+import pluginsConfig from './pluginsConfig'
+
+let beautifierObj
+
+export default function loadBeautifier(cb) {
+ const { beautifierUrl } = pluginsConfig
+ if (beautifierObj) {
+ cb(beautifierObj)
+ return
+ }
+
+ const loading = ELEMENT.Loading.service({
+ fullscreen: true,
+ lock: true,
+ text: '格式化资源加载中...',
+ spinner: 'el-icon-loading',
+ background: 'rgba(255, 255, 255, 0.5)'
+ })
+
+ loadScript(beautifierUrl, () => {
+ loading.close()
+ // eslint-disable-next-line no-undef
+ beautifierObj = beautifier
+ cb(beautifierObj)
+ })
+}
diff --git a/src/utils/loadMonaco.js b/src/utils/loadMonaco.js
new file mode 100644
index 00000000..9bb64f14
--- /dev/null
+++ b/src/utils/loadMonaco.js
@@ -0,0 +1,40 @@
+import loadScript from './loadScript'
+import ELEMENT from 'element-ui'
+import pluginsConfig from './pluginsConfig'
+
+// monaco-editor单例
+let monacoEidtor
+
+/**
+ * 动态加载monaco-editor cdn资源
+ * @param {Function} cb 回调,必填
+ */
+export default function loadMonaco(cb) {
+ if (monacoEidtor) {
+ cb(monacoEidtor)
+ return
+ }
+
+ const { monacoEditorUrl: vs } = pluginsConfig
+
+ // 使用element ui实现加载提示
+ const loading = ELEMENT.Loading.service({
+ fullscreen: true,
+ lock: true,
+ text: '编辑器资源初始化中...',
+ spinner: 'el-icon-loading',
+ background: 'rgba(255, 255, 255, 0.5)'
+ })
+
+ !window.require && (window.require = {})
+ !window.require.paths && (window.require.paths = {})
+ window.require.paths.vs = vs
+
+ loadScript(`${vs}/loader.js`, () => {
+ window.require(['vs/editor/editor.main'], () => {
+ loading.close()
+ monacoEidtor = window.monaco
+ cb(monacoEidtor)
+ })
+ })
+}
diff --git a/src/utils/loadScript.js b/src/utils/loadScript.js
new file mode 100644
index 00000000..18112fd7
--- /dev/null
+++ b/src/utils/loadScript.js
@@ -0,0 +1,60 @@
+const callbacks = {}
+
+/**
+ * 加载一个远程脚本
+ * @param {String} src 一个远程脚本
+ * @param {Function} callback 回调
+ */
+function loadScript(src, callback) {
+ const existingScript = document.getElementById(src)
+ const cb = callback || (() => {})
+ if (!existingScript) {
+ callbacks[src] = []
+ const $script = document.createElement('script')
+ $script.src = src
+ $script.id = src
+ $script.async = 1
+ document.body.appendChild($script)
+ const onEnd = 'onload' in $script ? stdOnEnd.bind($script) : ieOnEnd.bind($script)
+ onEnd($script)
+ }
+
+ callbacks[src].push(cb)
+
+ function stdOnEnd(script) {
+ script.onload = () => {
+ this.onerror = this.onload = null
+ callbacks[src].forEach(item => {
+ item(null, script)
+ })
+ delete callbacks[src]
+ }
+ script.onerror = () => {
+ this.onerror = this.onload = null
+ cb(new Error(`Failed to load ${src}`), script)
+ }
+ }
+
+ function ieOnEnd(script) {
+ script.onreadystatechange = () => {
+ if (this.readyState !== 'complete' && this.readyState !== 'loaded') return
+ this.onreadystatechange = null
+ callbacks[src].forEach(item => {
+ item(null, script)
+ })
+ delete callbacks[src]
+ }
+ }
+}
+
+/**
+ * 顺序加载一组远程脚本
+ * @param {Array} list 一组远程脚本
+ * @param {Function} cb 回调
+ */
+export function loadScriptQueue(list, cb) {
+ const first = list.shift()
+ list.length ? loadScript(first, () => loadScriptQueue(list, cb)) : loadScript(first, cb)
+}
+
+export default loadScript
diff --git a/src/utils/loadTinymce.js b/src/utils/loadTinymce.js
new file mode 100644
index 00000000..e2455fc8
--- /dev/null
+++ b/src/utils/loadTinymce.js
@@ -0,0 +1,29 @@
+import loadScript from './loadScript'
+import ELEMENT from 'element-ui'
+import pluginsConfig from './pluginsConfig'
+
+let tinymceObj
+
+export default function loadTinymce(cb) {
+ const { tinymceUrl } = pluginsConfig
+
+ if (tinymceObj) {
+ cb(tinymceObj)
+ return
+ }
+
+ const loading = ELEMENT.Loading.service({
+ fullscreen: true,
+ lock: true,
+ text: '富文本资源加载中...',
+ spinner: 'el-icon-loading',
+ background: 'rgba(255, 255, 255, 0.5)'
+ })
+
+ loadScript(tinymceUrl, () => {
+ loading.close()
+ // eslint-disable-next-line no-undef
+ tinymceObj = tinymce
+ cb(tinymceObj)
+ })
+}
diff --git a/src/utils/pluginsConfig.js b/src/utils/pluginsConfig.js
new file mode 100644
index 00000000..e7f4882a
--- /dev/null
+++ b/src/utils/pluginsConfig.js
@@ -0,0 +1,13 @@
+const CDN = 'https://lib.baomitu.com/' // CDN Homepage: https://cdn.baomitu.com/
+const publicPath = process.env.BASE_URL
+
+function splicingPluginUrl(PluginName, version, fileName) {
+ return `${CDN}${PluginName}/${version}/${fileName}`
+}
+
+export default {
+ beautifierUrl: splicingPluginUrl('js-beautify', '1.13.5', 'beautifier.min.js'),
+ // monacoEditorUrl: splicingPluginUrl('monaco-editor', '0.19.3', 'min/vs'), // 使用 monaco-editor CDN 链接
+ monacoEditorUrl: `${publicPath}libs/monaco-editor/vs`, // 使用 monaco-editor 本地代码
+ tinymceUrl: splicingPluginUrl('tinymce', '5.7.0', 'tinymce.min.js')
+}