mirror of
https://github.com/Tencent/tmagic-editor.git
synced 2025-04-05 19:41:40 +08:00
parent
b9248498eb
commit
cb99304993
@ -44,6 +44,10 @@ http://localhost:8098/tmagic-editor/playground/
|
||||
|
||||
即可得到一个魔方编辑器示例项目
|
||||
|
||||
## magic-admin
|
||||
|
||||
已迁移至 https://github.com/vft-magic/tmagic-admin
|
||||
|
||||
## 项目介绍
|
||||
在本项目中,我们核心内容,是包含在 `packages/editor` 中的编辑器,以及 `runtime` 和 `packages/ui` 提供的各个前端框架相关的 runtime 和 ui。
|
||||
|
||||
|
@ -1,223 +0,0 @@
|
||||
# 介绍
|
||||
|
||||
我们提供了与编辑器、表单等配套的管理端供开发者直接使用。管理端(magic-admin)代码存放于[开源仓库](https://github.com/Tencent/tmagic-editor)"magic-admin"目录下,可作为一个独立项目运行。我们提供这个管理端一方面期望开发者可以更清晰的了解一个项目从编辑到生成的整个流程,另一方面,开发者也可以
|
||||
在 magic-admin 的基础上快速搭建适合自己业务的管理平台。
|
||||
|
||||
管理端提供了如下能力:
|
||||
|
||||
- 项目列表展示,查询
|
||||
- 项目创建,复制
|
||||
- 项目编辑以及 AB TEST 配置能力
|
||||
- 项目发布以及发布状态查看和管理
|
||||
|
||||
<img src="https://vfiles.gtimg.cn/vupload/20211129/81d34a1638168945248.png">
|
||||
|
||||
## 开发调试
|
||||
|
||||
magic-admin 管理端分为 web 端和 server 端,目录结构如下:
|
||||
|
||||
**web 目录结构**
|
||||
|
||||
```
|
||||
.
|
||||
├── babel.config.js
|
||||
├── jest.config.js
|
||||
├── package.json
|
||||
├── package-lock.json
|
||||
├── public
|
||||
│ ├── favicon.ico
|
||||
│ └── index.html
|
||||
├── README.md
|
||||
├── src
|
||||
│ ├── api(web 端接口文件)
|
||||
│ ├── App.vue
|
||||
│ ├── assets
|
||||
│ ├── components(组件文件)
|
||||
│ ├── config(表单和状态配置文件)
|
||||
│ ├── main.ts(入口文件)
|
||||
│ ├── plugins(插件)
|
||||
│ ├── router(路由)
|
||||
│ ├── shims-vue.d.ts
|
||||
│ ├── store(全局变量的封装)
|
||||
│ ├── typings
|
||||
│ ├── use(核心逻辑)
|
||||
│ ├── util(公共方法)
|
||||
│ └── views
|
||||
├── tests
|
||||
│ ├── unit(测试用例文件)
|
||||
│ └── utils.ts
|
||||
├── tsconfig.json
|
||||
├── types(声明文件)
|
||||
│ ├── axios-jsonp
|
||||
│ ├── index.d.ts
|
||||
│ └── shims-vue.d.ts
|
||||
└── vue.config.js
|
||||
```
|
||||
|
||||
**server 目录结构**
|
||||
|
||||
```
|
||||
.
|
||||
├── jest.config.ts
|
||||
├── package.json
|
||||
├── package-lock.json
|
||||
├── pm2.config.js
|
||||
├── src
|
||||
│ ├── config(配置文件)
|
||||
│ ├── controller(控制器)
|
||||
│ ├── database(数据库初始化 sql 文件)
|
||||
│ ├── index.ts(入口文件)
|
||||
│ ├── models(数据库模型定义,使用`sequelize`)
|
||||
│ ├── routers(路由文件)
|
||||
│ ├── sequelize(数据库实例初始化文件)
|
||||
│ ├── service(service 文件)
|
||||
│ ├── template(发布所需模板文件)
|
||||
│ ├── typings(声明文件)
|
||||
│ └── utils(公共方法文件)
|
||||
├── tests
|
||||
│ └── unit(测试用例)
|
||||
└── tsconfig.json
|
||||
```
|
||||
|
||||
**开发者本地调试 magic-admin 请按照如下步骤:**
|
||||
|
||||
- 数据库:
|
||||
我们在 magic-admin/server/src/database/init.sql 中准备了库表初始化文件,开发者首先需要创建所需数据表
|
||||
|
||||
- 表名:magic_act_info
|
||||
项目基础信息表,包含项目 ID,项目名称,项目负责人,项目时间等项目基础信息。
|
||||
- 表名:magic_ui_config
|
||||
页面配置表,magic-admin 支持了一个项目中包含多个项目页面的能力,因此每个页面的组件配置信息将分别存储。
|
||||
|
||||
- 启动 web 端:
|
||||
|
||||
```bash
|
||||
$ cd magic-admin/web
|
||||
$ npm i
|
||||
$ npm run serve
|
||||
```
|
||||
|
||||
- 启动 server 端
|
||||
|
||||
```bash
|
||||
$ cd magic-admin/server
|
||||
$ npm i
|
||||
$ npm run dev
|
||||
```
|
||||
|
||||
server 文件夹下面这些敏感文件,需要开发者参考示例进行替换:
|
||||
|
||||
```
|
||||
.
|
||||
├── src
|
||||
│ ├── config
|
||||
│ │ ├── databaseExample.ts(数据库配置文件)
|
||||
│ │ ├── keyExample.ts(加密秘钥配置)
|
||||
```
|
||||
|
||||
- 关于登录态:
|
||||
magic-admin 在库表中为开发者预留了用户信息字段(项目负责人),开发者可以根据自身业务需要,实现用户登录态
|
||||
|
||||
```js
|
||||
// web/src/App.vue
|
||||
watchEffect(async () => {
|
||||
// 登录态获取交由开发者实现
|
||||
const userName = process.env.VUE_APP_USER_NAME || "defaultName";
|
||||
Cookies.set("userName", userName);
|
||||
});
|
||||
```
|
||||
|
||||
## 管理端能力
|
||||
|
||||
- **项目状态**
|
||||
|
||||
我们将项目的状态分为三种:修改中,部分页面已发布,全部页面已发布。在项目列表页面,可以展开查看每个项目页面的状态。
|
||||
|
||||
修改中:项目所有页面均在编辑状态
|
||||
|
||||
部分已发布:项目的一些页面在编辑状态,一些页面已发布
|
||||
|
||||
已发布:项目所有页面均已发布
|
||||
|
||||
- **在管理端引入 runtime**
|
||||
|
||||
在管理端中我们提供了一个可视化的模拟画布,他需要依赖 runtime 核心库,因此我们需要先在 magic 根目录下运行
|
||||
|
||||
```js
|
||||
cd magic
|
||||
npm run build:runtime:admin
|
||||
```
|
||||
|
||||
将 /runtime/vue3|vue2|react/dist 文件夹复制到 /magic-admin/serve/static 目录下。
|
||||
|
||||
```
|
||||
mkdir ./magic-admin/server/static/vue3
|
||||
cp -rf ./runtime/vue3/dist/* ./magic-admin/server/static/vue3
|
||||
```
|
||||
|
||||
上面的操作我们提供了/magic-admin/setup.sh 脚本文件来实现,开发者可以参考该脚本文件来搭建流水线。
|
||||
|
||||
[runtime 详细介绍](https://tencent.github.io/tmagic-editor/docs/page/introduction.html#runtime)
|
||||
|
||||
- **AB TEST**
|
||||
|
||||
当项目开发者需要对页面进行 AB TEST 测试时,可以在项目中创建多个项目页面,并且在项目配置中进行配置
|
||||
<img src="https://vfiles.gtimg.cn/vupload/20211129/c11fa81638173475771.png">
|
||||
这里仅为管理端的配置,通过这里我们将在项目配置文件中得到类似如下结构的 abtest 信息,开发者可以在页面加载时根据 DSL 的 abtest 字段进行判断。
|
||||
|
||||
```js
|
||||
abTest: [
|
||||
{
|
||||
name: "abtest1",
|
||||
type: ["pgv_pvid"],
|
||||
pageList: [
|
||||
{
|
||||
pageName: "page_1",
|
||||
proportion: "50",
|
||||
},
|
||||
{
|
||||
pageName: "page_2",
|
||||
proportion: "50",
|
||||
},
|
||||
],
|
||||
},
|
||||
];
|
||||
```
|
||||
|
||||
- **项目保存**
|
||||
|
||||
项目创建之后的配置信息分为两部份:项目基础信息,页面组件配置信息。
|
||||
项目基础信息是整个项目共用的配置项,对应 magic_act_info 数据表
|
||||
页面组件配置信息是一个项目中单个页面的配置项,对应 magic_ui_config 数据表
|
||||
|
||||
项目基础信息不在 magic-editor 支持范围以内,需要开发者自行结合 @tmagic/form 按需开发。管理端示例中这部分内容在页面右上角【项目配置】抽屉页
|
||||
|
||||
页面组件配置信息是指 magic-editor 中的 modelValue,他是一份 js schema 包含了页面内组件的配置内容,也是页面渲染的关键依赖文件。[DSL 概念参考](https://tencent.github.io/tmagic-editor/docs/page/introduction.html#%E7%BC%96%E8%BE%91%E5%99%A8%E4%BA%A7%E7%89%A9-DSL)
|
||||
|
||||
magic-admin 支持一个项目中创建多个项目页面的能力,因此,在项目保存的时候,我们的做法是将每个页面单独作为一条记录保存,比如项目 A 中包含页面 1 和页面 2
|
||||
保存之后我们将得到
|
||||
|
||||
magic_act_info 表
|
||||
| act_id | act_name | operator | act_status | abtest_raw | ... |
|
||||
| --- | --- | --- | --- |--- |--- |
|
||||
| 123 |项目 A | username | 修改中| []|... |
|
||||
|
||||
magic_ui_config 表
|
||||
| id | act_id | c_dist_code | page_title | page_publish_status |... |
|
||||
| --- | --- | --- | --- |--- | --- |
|
||||
| 1 |123 | 页面 1 的 DSL 配置 |页面 1 |修改中|...|
|
||||
| 2 |123 | 页面 2 的 DSL 配置 |页面 2|修改中|...|
|
||||
|
||||
- **项目发布**
|
||||
|
||||
管理端的项目发布是对[页面发布](https://tencent.github.io/tmagic-editor/docs/page/introduction.html#%E9%A1%B5%E9%9D%A2%E5%8F%91%E5%B8%83) 的实践。
|
||||
原始的页面框架 page.html 需要通过 runtime 打包生成,注入的 DSL 保存在 magic_ui_config 表 c_dist_code 字段中。
|
||||
发布时将 DSL 文件注入到 page.html 中,写入 server/assets/publish 目录下,访问路径: http://localhost/publish/${page_name}.html
|
||||
|
||||
## 部署
|
||||
|
||||
::: tip
|
||||
前提条件:node 环境>=14.15
|
||||
:::
|
||||
|
||||
如需使用流水线部署,请参考 /magic-admin/setup.sh
|
@ -1,13 +0,0 @@
|
||||
{
|
||||
"name": "magic-admin",
|
||||
"version": "1.0.0",
|
||||
"description": "Magic Admin 可视化搭建平台管理端。magic-admin 为独立项目目录,分为 web 端和 server 端,请按照下面指引操作",
|
||||
"main": "index.js",
|
||||
"scripts": {
|
||||
"admin:run": "chmod u+x setup.sh && ./setup.sh",
|
||||
"init": "npm run web:install && npm run server:install",
|
||||
"web:install": "cd web && npm install",
|
||||
"server:install": "cd server && npm install"
|
||||
},
|
||||
"author": ""
|
||||
}
|
@ -1,10 +0,0 @@
|
||||
{
|
||||
"presets": [
|
||||
[
|
||||
"@babel/preset-env"
|
||||
],
|
||||
[
|
||||
"@babel/preset-typescript"
|
||||
]
|
||||
]
|
||||
}
|
@ -1,3 +0,0 @@
|
||||
dist
|
||||
node_modules
|
||||
pm2.config.js
|
@ -1,51 +0,0 @@
|
||||
module.exports = {
|
||||
env: {
|
||||
node: true,
|
||||
browser: true,
|
||||
},
|
||||
globals: {
|
||||
describe: true,
|
||||
it: true,
|
||||
expect: true,
|
||||
jest: true,
|
||||
beforeEach: true,
|
||||
},
|
||||
parser: '@typescript-eslint/parser',
|
||||
extends: [
|
||||
'eslint-config-tencent',
|
||||
'eslint-config-tencent/ts',
|
||||
'./prettier',
|
||||
],
|
||||
plugins: [
|
||||
'@typescript-eslint',
|
||||
'simple-import-sort',
|
||||
],
|
||||
ignorePatterns: ['.eslintrc.js','/assets/*','/tests/*','/coverage/*'],
|
||||
rules: {
|
||||
'no-param-reassign': 'off',
|
||||
'simple-import-sort/imports': [
|
||||
'error', {
|
||||
groups: [
|
||||
// Node.js builtins. You could also generate this regex if you use a `.js` config.
|
||||
// For example: `^(${require('module').builtinModules.join('|')})(/|$)`
|
||||
[
|
||||
'^(assert|buffer|child_process|cluster|console|constants|crypto|dgram|dns|domain|events|fs|http|https|module|net|os|path|punycode|querystring|readline|repl|stream|string_decoder|sys|timers|tls|tty|url|util|vm|zlib|freelist|v8|process|async_hooks|http2|perf_hooks)(/.*|$)',
|
||||
],
|
||||
// Packages. `react|vue` related packages come first.
|
||||
['^(react|vue|vite)', '^@?\\w'],
|
||||
['^(@tmagic)(/.*|$)'],
|
||||
// Internal packages.
|
||||
['^(@|@src|@tests)(/.*|$)'],
|
||||
// Side effect imports.
|
||||
['^\\u0000'],
|
||||
// Parent imports. Put `..` last.
|
||||
['^\\.\\.(?!/?$)', '^\\.\\./?$'],
|
||||
// Other relative imports. Put same-folder imports and `.` last.
|
||||
['^\\./(?=.*/)(?!/?$)', '^\\.(?!/?$)', '^\\./?$'],
|
||||
// Style imports.
|
||||
['^.+\\.s?css$'],
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
};
|
10
magic-admin/server/.gitignore
vendored
10
magic-admin/server/.gitignore
vendored
@ -1,10 +0,0 @@
|
||||
node_modules
|
||||
dist
|
||||
yarn.lock
|
||||
package.json.lock
|
||||
coverage
|
||||
.vscode
|
||||
assets
|
||||
!static/**/assets
|
||||
src/config/database.ts
|
||||
src/config/key.ts
|
@ -1,7 +0,0 @@
|
||||
export default {
|
||||
collectCoverage: true,
|
||||
coverageProvider: 'v8',
|
||||
moduleNameMapper: {
|
||||
'^@src/(.*)$': '<rootDir>/src/$1',
|
||||
},
|
||||
};
|
28165
magic-admin/server/package-lock.json
generated
28165
magic-admin/server/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -1,76 +0,0 @@
|
||||
{
|
||||
"name": "magic-admin",
|
||||
"version": "1.0.0",
|
||||
"description": "",
|
||||
"tags": [
|
||||
"orm",
|
||||
"typescript",
|
||||
"koa"
|
||||
],
|
||||
"scripts": {
|
||||
"dev": "nodemon",
|
||||
"lint": "eslint . --ext .js,.ts --fix",
|
||||
"test": "jest"
|
||||
},
|
||||
"_moduleAliases": {
|
||||
"@": "dist/src"
|
||||
},
|
||||
"nodemonConfig": {
|
||||
"watch": [
|
||||
"src"
|
||||
],
|
||||
"ext": "ts",
|
||||
"env": {
|
||||
"NODE_ENV": "development",
|
||||
"PORT": 3001
|
||||
},
|
||||
"exec": "ts-node -r tsconfig-paths/register src/index.ts --files"
|
||||
},
|
||||
"dependencies": {
|
||||
"dayjs": "^1.11.7",
|
||||
"koa": "^2.7.0",
|
||||
"koa-bodyparser": "^4.2.1",
|
||||
"koa-router": "^8.0.6",
|
||||
"koa-send": "^5.0.1",
|
||||
"log4js": "^6.3.0",
|
||||
"mysql2": "^2.3.3",
|
||||
"reflect-metadata": "^0.1.13",
|
||||
"sequelize": "^6.29.0",
|
||||
"sequelize-typescript": "^2.1.0",
|
||||
"tslib": "^2.4.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@babel/core": "^7.20.12",
|
||||
"@babel/preset-env": "^7.20.2",
|
||||
"@babel/preset-typescript": "^7.18.6",
|
||||
"@types/axios": "^0.14.0",
|
||||
"@types/babel-core": "^6.25.7",
|
||||
"@types/fs-extra": "^9.0.13",
|
||||
"@types/jest": "^26.0.24",
|
||||
"@types/koa": "^2.0.48",
|
||||
"@types/koa-bodyparser": "^4.2.2",
|
||||
"@types/koa-router": "^7.0.40",
|
||||
"@types/koa-send": "^4.1.3",
|
||||
"@types/node": "^12.0.0",
|
||||
"@typescript-eslint/eslint-plugin": "^4.28.1",
|
||||
"@typescript-eslint/parser": "^4.28.1",
|
||||
"axios": "^0.24.0",
|
||||
"babel-jest": "^27.0.6",
|
||||
"eslint": "^7.29.0",
|
||||
"eslint-config-tencent": "^1.0.2",
|
||||
"eslint-plugin-prettier": "^3.4.0",
|
||||
"eslint-plugin-simple-import-sort": "^7.0.0",
|
||||
"fs-extra": "^10.0.0",
|
||||
"jest": "^27.0.6",
|
||||
"lodash": "^4.17.21",
|
||||
"minimatch": "^5.1.0",
|
||||
"module-alias": "^2.2.2",
|
||||
"nodemon": "^1.19.0",
|
||||
"prettier": "^2.3.2",
|
||||
"serialize-javascript": "^6.0.0",
|
||||
"ts-node": "^8.1.0",
|
||||
"tsconfig-paths": "^3.8.0",
|
||||
"typescript": "^4.7.4",
|
||||
"uglify-js": "^3.14.1"
|
||||
}
|
||||
}
|
@ -1,76 +0,0 @@
|
||||
module.exports = {
|
||||
plugins: ['prettier'],
|
||||
rules: {
|
||||
'wrap-iife': 'off',
|
||||
'template-curly-spacing': 'off',
|
||||
'space-infix-ops': 'off',
|
||||
'space-in-parens': 'off',
|
||||
'space-before-blocks': 'off',
|
||||
'quote-props': 'off',
|
||||
'padded-blocks': 'off',
|
||||
'operator-linebreak': 'off',
|
||||
'object-curly-spacing': 'off',
|
||||
'nonblock-statement-body-position': 'off',
|
||||
'no-whitespace-before-property': 'off',
|
||||
'no-trailing-spaces': 'off',
|
||||
'no-multiple-empty-lines': 'off',
|
||||
'no-mixed-operators': 'off',
|
||||
'no-confusing-arrow': 'off',
|
||||
'newline-per-chained-call': 'off',
|
||||
'max-len': 'off',
|
||||
'linebreak-style': 'off',
|
||||
'key-spacing': 'off',
|
||||
'implicit-arrow-linebreak': 'off',
|
||||
'generator-star-spacing': 'off',
|
||||
'function-paren-newline': 'off',
|
||||
'eol-last': 'off',
|
||||
'computed-property-spacing': 'off',
|
||||
'comma-style': 'off',
|
||||
'comma-dangle': 'off',
|
||||
'block-spacing': 'off',
|
||||
'arrow-spacing': 'off',
|
||||
'arrow-parens': 'off',
|
||||
'array-bracket-spacing': 'off',
|
||||
|
||||
'@typescript-eslint/type-annotation-spacing': 'off',
|
||||
'@typescript-eslint/space-before-function-paren': 'off',
|
||||
'@typescript-eslint/semi': 'off',
|
||||
'@typescript-eslint/quotes': 'off',
|
||||
'@typescript-eslint/keyword-spacing': 'off',
|
||||
'@typescript-eslint/indent': 'off',
|
||||
'@typescript-eslint/func-call-spacing': 'off',
|
||||
'@typescript-eslint/comma-spacing': 'off',
|
||||
'@typescript-eslint/brace-style': 'off',
|
||||
/**
|
||||
* prettier 格式错误
|
||||
*/
|
||||
'prettier/prettier': [
|
||||
'warn',
|
||||
{
|
||||
printWidth: 120,
|
||||
tabWidth: 2,
|
||||
useTabs: false,
|
||||
semi: true,
|
||||
singleQuote: true,
|
||||
quoteProps: 'as-needed',
|
||||
jsxSingleQuote: false,
|
||||
trailingComma: 'all',
|
||||
bracketSpacing: true,
|
||||
jsxBracketSameLine: false,
|
||||
arrowParens: 'always',
|
||||
rangeStart: 0,
|
||||
rangeEnd: null,
|
||||
requirePragma: false,
|
||||
insertPragma: false,
|
||||
proseWrap: 'preserve',
|
||||
htmlWhitespaceSensitivity: 'css',
|
||||
vueIndentScriptAndStyle: false,
|
||||
endOfLine: 'lf',
|
||||
embeddedLanguageFormatting: 'auto',
|
||||
},
|
||||
{
|
||||
usePrettierrc: false,
|
||||
},
|
||||
],
|
||||
},
|
||||
};
|
@ -1,42 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import path from 'path';
|
||||
// 活动状态
|
||||
export enum ActStatus {
|
||||
ALL = -1, // 查询传参使用:全部状态占位
|
||||
MODIFYING, // 修改中
|
||||
PART_PUBLISHED, // 部分页面已发布
|
||||
PUBLISHED, // 全部页面已发布
|
||||
}
|
||||
|
||||
// 页面状态
|
||||
export enum PageStatus {
|
||||
MODIFYING = 0, // 修改中
|
||||
PUBLISHED, // 已预发布
|
||||
}
|
||||
|
||||
// 静态资源根目录
|
||||
export const StaticPath = {
|
||||
ASSETS: path.resolve(__dirname, '../../assets'),
|
||||
TEMPLATE: path.resolve(__dirname, '../template'),
|
||||
STATIC: path.resolve(__dirname, '../../static'),
|
||||
PUBLISH: path.resolve(__dirname, '../../assets/publish'),
|
||||
};
|
||||
|
||||
export const UiRuntimeJS = '<script src="https://unpkg.com/vue@next/dist/vue.runtime.global.js"></script>';
|
@ -1,27 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 数据库配置文件,需开发者自行替换并更名为database.ts
|
||||
export default {
|
||||
connectionLimit: 10,
|
||||
host: '1.2.3.4',
|
||||
port: 36000,
|
||||
user: 'database_username',
|
||||
password: 'database_password',
|
||||
database: 'database_name',
|
||||
};
|
@ -1,22 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// crypto加密key,请开发者自行替换。需要 32 位
|
||||
export default {
|
||||
key: 'crypto_algorithm_aes-256-cbc1111',
|
||||
};
|
@ -1,120 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 处理活动请求
|
||||
import Koa from 'koa';
|
||||
|
||||
import ActService, { ActInfoDetail, ActListQuery, CopyInfo } from '@src/service/act';
|
||||
|
||||
class ActController {
|
||||
private service: ActService = new ActService();
|
||||
|
||||
// 获取活动列表
|
||||
getList = async (ctx: Koa.Context) => {
|
||||
try {
|
||||
const query: ActListQuery = JSON.parse(ctx.request.body.data);
|
||||
const [actList, count] = await Promise.all([this.service.getActList(query), this.service.getCount(query)]);
|
||||
ctx.body = {
|
||||
data: actList,
|
||||
total: count,
|
||||
fetch: true,
|
||||
errorMsg: '',
|
||||
};
|
||||
} catch (e) {
|
||||
ctx.logger.error(e);
|
||||
ctx.body = {
|
||||
data: [],
|
||||
total: 0,
|
||||
fetch: false,
|
||||
errorMsg: (e as Error).message,
|
||||
};
|
||||
}
|
||||
};
|
||||
|
||||
// 新建活动
|
||||
create = async (ctx: Koa.Context) => {
|
||||
try {
|
||||
const actInfo: ActInfoDetail = JSON.parse(ctx.request.body.data);
|
||||
const actId = await this.service.create(actInfo);
|
||||
ctx.body = {
|
||||
ret: 0,
|
||||
msg: '新建活动成功',
|
||||
data: { actId },
|
||||
};
|
||||
} catch (e) {
|
||||
ctx.body = {
|
||||
ret: -1,
|
||||
msg: (e as Error).message,
|
||||
};
|
||||
}
|
||||
};
|
||||
|
||||
// 复制活动
|
||||
copy = async (ctx: Koa.Context) => {
|
||||
try {
|
||||
const copyInfo: CopyInfo = JSON.parse(ctx.request.body.data);
|
||||
await this.service.copy(copyInfo);
|
||||
ctx.body = {
|
||||
ret: 0,
|
||||
msg: '复制成功',
|
||||
};
|
||||
} catch (e) {
|
||||
ctx.body = {
|
||||
ret: -1,
|
||||
msg: (e as Error).message,
|
||||
};
|
||||
}
|
||||
};
|
||||
|
||||
// 根据id查询活动详情
|
||||
getInfo = async (ctx: Koa.Context) => {
|
||||
try {
|
||||
const id = Number(ctx.query.id);
|
||||
const act = await this.service.getActInfo(id);
|
||||
ctx.body = {
|
||||
ret: 0,
|
||||
msg: '获取活动信息成功',
|
||||
data: act,
|
||||
};
|
||||
} catch (e) {
|
||||
ctx.body = {
|
||||
ret: -1,
|
||||
msg: (e as Error).message,
|
||||
};
|
||||
}
|
||||
};
|
||||
|
||||
// 根据页面id删除活动页面
|
||||
removePage = async (ctx: Koa.Context) => {
|
||||
try {
|
||||
const { pageId } = JSON.parse(ctx.request.body.data);
|
||||
await this.service.removePage(pageId);
|
||||
ctx.body = {
|
||||
ret: 0,
|
||||
msg: '删除活动页面成功',
|
||||
};
|
||||
} catch (e) {
|
||||
ctx.body = {
|
||||
ret: -1,
|
||||
msg: (e as Error).message,
|
||||
};
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
export default new ActController();
|
@ -1,41 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 处理编辑器请求
|
||||
import Koa from 'koa';
|
||||
|
||||
import EditorService from '@src/service/editor';
|
||||
|
||||
class EditorController {
|
||||
private service: EditorService = new EditorService();
|
||||
|
||||
// 拉取编辑器左侧展示的组件列表
|
||||
getComponentList = async (ctx: Koa.Context) => {
|
||||
ctx.body = {
|
||||
ret: 0,
|
||||
msg: '获取组件列表成功',
|
||||
data: await this.service.getComponentList(),
|
||||
};
|
||||
};
|
||||
// 拉取编辑器右边活动配置的web插件
|
||||
getWebPlugins = async (ctx: Koa.Context) => {
|
||||
ctx.body = await this.service.getWebPlugins();
|
||||
};
|
||||
}
|
||||
|
||||
export default new EditorController();
|
@ -1,65 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 处理保存发布请求
|
||||
import Koa from 'koa';
|
||||
|
||||
import PublishService from '@src/service/publish';
|
||||
|
||||
class PublishController {
|
||||
private service: PublishService = new PublishService();
|
||||
|
||||
// 保存活动基础信息
|
||||
saveActInfo = async (ctx: Koa.Context) => {
|
||||
try {
|
||||
// data不是真正的json对象,可能包含组件自定义code代码
|
||||
/* eslint-disable-next-line */
|
||||
const { actInfo,rootInfo } = eval(`(${ctx.request.body.data})`);
|
||||
const res = await this.service.saveActInfo({ actInfo, rootInfo });
|
||||
ctx.body = {
|
||||
ret: res.ret,
|
||||
msg: res.msg,
|
||||
};
|
||||
} catch (e) {
|
||||
ctx.body = {
|
||||
ret: -1,
|
||||
msg: (e as Error).message,
|
||||
};
|
||||
}
|
||||
};
|
||||
// 发布
|
||||
publish = async (ctx: Koa.Context) => {
|
||||
try {
|
||||
// data不是真正的json对象,可能包含组件自定义code代码
|
||||
/* eslint-disable-next-line */
|
||||
const { actId, publishPages,rootInfo } = eval(`(${ctx.request.body.data})`);
|
||||
const operator = ctx.cookies.get('userName');
|
||||
const res = await this.service.publish({ actId, publishPages, rootInfo, operator });
|
||||
ctx.body = {
|
||||
ret: res.ret,
|
||||
msg: res.msg,
|
||||
};
|
||||
} catch (e) {
|
||||
ctx.body = {
|
||||
ret: -1,
|
||||
msg: (e as Error).message,
|
||||
};
|
||||
}
|
||||
};
|
||||
}
|
||||
export default new PublishController();
|
@ -1,35 +0,0 @@
|
||||
-- 活动基础信息表
|
||||
CREATE TABLE `magic_act_info` (
|
||||
`act_id` int(20) NOT NULL AUTO_INCREMENT COMMENT '活动id',
|
||||
`act_crypto_id` varchar(128) NOT NULL COMMENT '活动加密ID',
|
||||
`act_name` varchar(128) NOT NULL COMMENT '活动名称',
|
||||
`act_begin_time` varchar(128) NOT NULL COMMENT '活动开始时间',
|
||||
`act_end_time` varchar(128) NOT NULL COMMENT '活动结束时间',
|
||||
`act_modify_time` varchar(128) DEFAULT NULL COMMENT '活动修改时间',
|
||||
`act_create_time` varchar(128) NOT NULL COMMENT '活动创建时间',
|
||||
`operator` varchar(512) DEFAULT NULL COMMENT '负责人',
|
||||
`locker` varchar(128) DEFAULT NULL COMMENT '当前正在编辑的人',
|
||||
`lock_time` datetime DEFAULT NULL COMMENT '锁定时间',
|
||||
`act_status` int(11) DEFAULT NULL COMMENT '活动状态:0-修改中,1-部分已发布,2-已发布',
|
||||
`abtest_raw` mediumtext COMMENT 'serialize后的abtest',
|
||||
PRIMARY KEY (`act_id`),
|
||||
KEY `act_name` (`act_name`)
|
||||
) ENGINE = InnoDB AUTO_INCREMENT = 1 DEFAULT CHARSET = utf8mb4 COMMENT = '魔方开源-活动信息表';
|
||||
|
||||
-- 页面配置表
|
||||
CREATE TABLE `magic_ui_config` (
|
||||
`id` int(8) unsigned NOT NULL AUTO_INCREMENT COMMENT '页面id',
|
||||
`act_id` int(8) NOT NULL COMMENT '活动id',
|
||||
`c_dist_code` mediumblob COMMENT 'babel编译后的config',
|
||||
`c_src_code` mediumblob COMMENT 'config 源码',
|
||||
`c_c_time` varchar(128) DEFAULT NULL COMMENT 'config创建时间',
|
||||
`c_m_time` varchar(128) DEFAULT NULL COMMENT 'config修改时间',
|
||||
`c_ui_version` varchar(64) DEFAULT NULL COMMENT 'magic-ui 版本',
|
||||
`page_title` varchar(128) NOT NULL COMMENT '活动页面标题(H5顶部展示)',
|
||||
`page_publish_time` varchar(128) DEFAULT NULL COMMENT '页面发布时间',
|
||||
`page_publish_status` int(11) NOT NULL DEFAULT '0' COMMENT '页面发布状态:修改中0,已发布1',
|
||||
`publish_operator` varchar(128) DEFAULT NULL COMMENT '发布人',
|
||||
`web_plugin` varchar(255) DEFAULT NULL COMMENT 'web插件',
|
||||
`page_name` varchar(128) DEFAULT NULL COMMENT '页面名称(编辑器页面唯一标识)',
|
||||
PRIMARY KEY (`id`)
|
||||
) ENGINE = InnoDB AUTO_INCREMENT = 1 DEFAULT CHARSET = utf8mb4 COMMENT = '魔方开源-uiconfig表';
|
@ -1,48 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import Koa from 'koa';
|
||||
import bodyParser from 'koa-bodyparser';
|
||||
import { getLogger } from 'log4js';
|
||||
|
||||
import routers from '@src/routers';
|
||||
import staticRouters from '@src/routers/static';
|
||||
const app = new Koa();
|
||||
const { PORT } = process.env;
|
||||
|
||||
app.use(
|
||||
bodyParser({
|
||||
formLimit: '10mb',
|
||||
jsonLimit: '10mb',
|
||||
}),
|
||||
);
|
||||
app.use(async (ctx, next) => {
|
||||
ctx.logger = getLogger();
|
||||
ctx.logger.level = 'debug';
|
||||
await next();
|
||||
});
|
||||
app.use(async (ctx, next) => {
|
||||
ctx.logger.debug(ctx.url);
|
||||
await next();
|
||||
});
|
||||
// 初始化路由中间件
|
||||
app.use(routers.routes()).use(routers.allowedMethods());
|
||||
app.use(staticRouters.routes()).use(staticRouters.allowedMethods());
|
||||
app.listen(PORT);
|
||||
|
||||
console.log(`server启动成功 端口:${PORT}`);
|
@ -1,73 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { AllowNull, Column, HasMany, Model, Table } from 'sequelize-typescript';
|
||||
|
||||
import { Page } from '@src/models/page';
|
||||
// 活动基础信息表
|
||||
@Table({
|
||||
tableName: 'magic_act_info',
|
||||
})
|
||||
export class ActInfo extends Model<ActInfo> {
|
||||
@Column({
|
||||
primaryKey: true,
|
||||
autoIncrement: true,
|
||||
field: 'act_id',
|
||||
})
|
||||
actId: number;
|
||||
|
||||
@Column({ field: 'act_crypto_id' })
|
||||
actCryptoId: string;
|
||||
|
||||
@Column({ field: 'act_name' })
|
||||
actName: string;
|
||||
|
||||
@Column({ field: 'act_begin_time' })
|
||||
actBeginTime: string;
|
||||
|
||||
@Column({ field: 'act_end_time' })
|
||||
actEndTime: string;
|
||||
|
||||
@Column({ field: 'act_modify_time' })
|
||||
actModifyTime: string;
|
||||
|
||||
@Column({ field: 'act_create_time' })
|
||||
actCreateTime: string;
|
||||
|
||||
@AllowNull
|
||||
@Column
|
||||
operator?: string;
|
||||
|
||||
@AllowNull
|
||||
@Column
|
||||
locker?: string;
|
||||
|
||||
@Column({ field: 'lock_time' })
|
||||
lockTime: string;
|
||||
|
||||
@AllowNull
|
||||
@Column({ field: 'act_status' })
|
||||
actStatus: number; // 0:修改中 1:部分已发布 2:已发布
|
||||
|
||||
@HasMany(() => Page)
|
||||
pages: Page[];
|
||||
|
||||
@AllowNull
|
||||
@Column({ field: 'abtest_raw' })
|
||||
abTestRaw?: string;
|
||||
}
|
@ -1,22 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 模型汇总
|
||||
import { ActInfo } from '@src/models/act';
|
||||
import { Page } from '@src/models/page';
|
||||
export default [ActInfo, Page];
|
@ -1,76 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { AllowNull, BelongsTo, Column, ForeignKey, Model, Table } from 'sequelize-typescript';
|
||||
|
||||
import { ActInfo } from '@src/models/act';
|
||||
// 页面信息表
|
||||
@Table({
|
||||
tableName: 'magic_ui_config',
|
||||
})
|
||||
export class Page extends Model<Page> {
|
||||
@Column({
|
||||
primaryKey: true,
|
||||
autoIncrement: true,
|
||||
})
|
||||
id: string;
|
||||
|
||||
@ForeignKey(() => ActInfo)
|
||||
@Column({ field: 'act_id' })
|
||||
actId: number;
|
||||
|
||||
@BelongsTo(() => ActInfo)
|
||||
act: ActInfo;
|
||||
|
||||
@AllowNull
|
||||
@Column({ field: 'c_dist_code' })
|
||||
distCode: string;
|
||||
|
||||
@AllowNull
|
||||
@Column({ field: 'c_src_code' })
|
||||
srcCode: string;
|
||||
|
||||
@AllowNull
|
||||
@Column({ field: 'c_c_time' })
|
||||
pageCreateTime: string;
|
||||
|
||||
@AllowNull
|
||||
@Column({ field: 'c_m_time' })
|
||||
pageModifyTime: string;
|
||||
|
||||
@AllowNull
|
||||
@Column({ field: 'c_ui_version' })
|
||||
pagePublishUiVersion: string;
|
||||
|
||||
@Column({ field: 'page_title' })
|
||||
pageTitle: string;
|
||||
|
||||
@Column({ field: 'page_name' })
|
||||
pageName: string;
|
||||
|
||||
@AllowNull
|
||||
@Column({ field: 'page_publish_time' })
|
||||
pagePublishTime: string;
|
||||
|
||||
@Column({ field: 'page_publish_status' })
|
||||
pagePublishStatus: number; // 0:修改中 1:已发布
|
||||
|
||||
@AllowNull
|
||||
@Column({ field: 'publish_operator' })
|
||||
pagePublishOperator: string;
|
||||
}
|
@ -1,40 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 活动列表相关路由
|
||||
import Router from 'koa-router';
|
||||
|
||||
import actController from '@src/controller/act';
|
||||
|
||||
const router = new Router();
|
||||
// 拉取编辑器左侧展示的组件列表
|
||||
router.post('/getList', actController.getList);
|
||||
|
||||
// 创建活动
|
||||
router.post('/create', actController.create);
|
||||
|
||||
// 复制活动
|
||||
router.post('/copy', actController.copy);
|
||||
|
||||
// 根据id获取活动信息
|
||||
router.get('/get', actController.getInfo);
|
||||
|
||||
// 删除活动页面
|
||||
router.post('/removePage', actController.removePage);
|
||||
|
||||
export default router;
|
@ -1,31 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 编辑器相关路由
|
||||
import Router from 'koa-router';
|
||||
|
||||
import editorController from '@src/controller/editor';
|
||||
|
||||
const router = new Router();
|
||||
// 拉取编辑器左侧展示的组件列表
|
||||
router.get('/getComponentList', editorController.getComponentList);
|
||||
|
||||
// 拉取活动配置的web插件
|
||||
router.get('/getWebPlugins', editorController.getWebPlugins);
|
||||
|
||||
export default router;
|
@ -1,38 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 整合所有子路由
|
||||
import Router from 'koa-router';
|
||||
|
||||
import act from '@src/routers/act';
|
||||
import editor from '@src/routers/editor';
|
||||
import publish from '@src/routers/publish';
|
||||
|
||||
const router = new Router({
|
||||
prefix: '/api',
|
||||
});
|
||||
// 编辑器相关路由
|
||||
router.use('/editor', editor.routes(), editor.allowedMethods());
|
||||
|
||||
// 活动列表相关路由
|
||||
router.use('/act', act.routes(), act.allowedMethods());
|
||||
|
||||
// 保存发布相关路由
|
||||
router.use('/publish', publish.routes(), publish.allowedMethods());
|
||||
|
||||
export default router;
|
@ -1,31 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 保存发布的相关路由
|
||||
import Router from 'koa-router';
|
||||
|
||||
import PublishController from '@src/controller/publish';
|
||||
|
||||
const router = new Router();
|
||||
// 保存活动基础信息
|
||||
router.post('/saveActInfo', PublishController.saveActInfo);
|
||||
|
||||
// 发布
|
||||
router.post('/publish', PublishController.publish);
|
||||
|
||||
export default router;
|
@ -1,48 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// web静态资源相关路由
|
||||
import { pathExistsSync } from 'fs-extra';
|
||||
import Router from 'koa-router';
|
||||
import send from 'koa-send';
|
||||
|
||||
import { StaticPath } from '@src/config/config';
|
||||
|
||||
const router = new Router();
|
||||
const options = { root: '/', gzip: true, maxage: 36000 };
|
||||
router.get('/', async (ctx) => {
|
||||
await send(ctx, `${StaticPath.ASSETS}/index.html`, options);
|
||||
});
|
||||
|
||||
router.get('/static/*', async (ctx) => {
|
||||
const file = `${StaticPath.STATIC}/${ctx.params[0]}`;
|
||||
if (pathExistsSync(file)) {
|
||||
await send(ctx, file, options);
|
||||
}
|
||||
});
|
||||
|
||||
router.get('/*', async (ctx) => {
|
||||
const file = `${StaticPath.ASSETS}/${ctx.params[0]}`;
|
||||
if (pathExistsSync(file)) {
|
||||
await send(ctx, file, options);
|
||||
} else {
|
||||
await send(ctx, `${StaticPath.ASSETS}/index.html`, options);
|
||||
}
|
||||
});
|
||||
|
||||
export default router;
|
@ -1,41 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { Sequelize } from 'sequelize-typescript';
|
||||
|
||||
import sqlConf from '@src/config/database';
|
||||
import models from '@src/models/index';
|
||||
// 数据库初始化
|
||||
export default class SequelizeHelper {
|
||||
private static instance;
|
||||
public static getInstance() {
|
||||
if (!SequelizeHelper.instance) {
|
||||
const sequelize = new Sequelize(sqlConf.database, sqlConf.user, sqlConf.password, {
|
||||
host: sqlConf.host,
|
||||
port: sqlConf.port,
|
||||
dialect: 'mysql',
|
||||
define: {
|
||||
timestamps: false,
|
||||
},
|
||||
});
|
||||
sequelize.addModels(models);
|
||||
SequelizeHelper.instance = sequelize;
|
||||
}
|
||||
return SequelizeHelper.instance;
|
||||
}
|
||||
}
|
@ -1,383 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { cloneDeep } from 'lodash';
|
||||
import { Op } from 'sequelize';
|
||||
|
||||
import { ActStatus } from '@src/config/config';
|
||||
import { ActInfo } from '@src/models/act';
|
||||
import { Page } from '@src/models/page';
|
||||
import SequelizeHelper from '@src/sequelize/index';
|
||||
import PageService from '@src/service/page';
|
||||
import type { ABTest, ActBaseInfo, ActInfoIncludePage, PageInfo } from '@src/typings';
|
||||
import Crypto from '@src/utils/crypto/crypto';
|
||||
import { getFormatTime } from '@src/utils/index';
|
||||
import logger from '@src/utils/logger';
|
||||
|
||||
export interface OrderItem {
|
||||
columnName: string;
|
||||
direction: string;
|
||||
}
|
||||
|
||||
export interface ActListQuery {
|
||||
where: {
|
||||
onlySelf: boolean;
|
||||
search: string;
|
||||
pageTitle: string;
|
||||
actStatus: number;
|
||||
};
|
||||
orderBy: OrderItem[];
|
||||
pgIndex: number;
|
||||
pgSize: number;
|
||||
userName: string;
|
||||
}
|
||||
|
||||
// 新建活动参数
|
||||
export interface ActInfoDetail {
|
||||
actName: string;
|
||||
actBeginTime: string;
|
||||
actEndTime: string;
|
||||
operator: string;
|
||||
}
|
||||
|
||||
// 复制活动参数
|
||||
export interface CopyInfo {
|
||||
actId: number;
|
||||
userName: string;
|
||||
}
|
||||
|
||||
export type FormatType = 'search' | 'actId' | 'status' | 'order' | 'title';
|
||||
|
||||
// 活动列表查询体的构建方法
|
||||
const getActListQueryConstructor = ({ attributes = [], query }: { attributes?: string[]; query: ActListQuery }) => ({
|
||||
attributes: [...attributes],
|
||||
where: {
|
||||
operator: { [Op.substring]: query.where.onlySelf ? query.userName : '' },
|
||||
actStatus: formatQuery(query.where.actStatus, 'status'),
|
||||
[Op.or]: [
|
||||
{ actName: { [Op.substring]: formatQuery(query.where.search, 'search') } },
|
||||
{ operator: { [Op.substring]: formatQuery(query.where.search, 'search') } },
|
||||
{ actCryptoId: query.where.search },
|
||||
formatQuery(query.where.search, 'actId'),
|
||||
],
|
||||
},
|
||||
});
|
||||
|
||||
export default class ActService {
|
||||
private pageService: PageService = new PageService();
|
||||
|
||||
/**
|
||||
* 查询活动列表
|
||||
* @param {ActListQuery} query 查询语句
|
||||
* @returns {ActInfoIncludePage[]} 活动列表数据
|
||||
*/
|
||||
getActList = async (query: ActListQuery) => {
|
||||
SequelizeHelper.getInstance();
|
||||
// 构建查询体
|
||||
const attributes = ['actId', 'actName', 'actBeginTime', 'actEndTime', 'actStatus', 'operator', 'actCryptoId'];
|
||||
const queryCond = getActListQueryConstructor({ attributes, query });
|
||||
const actList = await ActInfo.findAll({
|
||||
...queryCond,
|
||||
order: formatQuery(query.orderBy[0], 'order'),
|
||||
limit: query.pgSize,
|
||||
offset: query.pgIndex * query.pgSize,
|
||||
include: [
|
||||
{
|
||||
model: Page,
|
||||
attributes: ['pageTitle', 'pagePublishTime', 'pagePublishStatus', 'pagePublishOperator'],
|
||||
where: formatQuery(query.where.pageTitle, 'title') as {
|
||||
pageTitle: {
|
||||
[Op.substring]: string;
|
||||
};
|
||||
} | null,
|
||||
},
|
||||
],
|
||||
});
|
||||
|
||||
const formatActList = actList.map((act) => this.formatPages(act));
|
||||
|
||||
return formatActList;
|
||||
};
|
||||
|
||||
/**
|
||||
* 获取查询结果总数
|
||||
* @param {ActListQuery} query 查询语句
|
||||
* @returns {number} 查询结果总数
|
||||
*/
|
||||
getCount = async (query: ActListQuery) => {
|
||||
SequelizeHelper.getInstance();
|
||||
|
||||
// 构建查询体
|
||||
const queryCond = getActListQueryConstructor({ query });
|
||||
const actList = await ActInfo.findAll({
|
||||
...queryCond,
|
||||
include: [
|
||||
{
|
||||
model: Page,
|
||||
attributes: [],
|
||||
where: formatQuery(query.where.pageTitle, 'title'),
|
||||
},
|
||||
],
|
||||
});
|
||||
|
||||
return actList.length;
|
||||
};
|
||||
|
||||
/**
|
||||
* 根据id查询活动详情
|
||||
* @param {number} actId 活动id
|
||||
* @returns {ActInfoIncludePage} 活动基本信息含页面配置
|
||||
*/
|
||||
getActInfo = async (actId: number) => {
|
||||
const act = await getActInfoHandler(actId);
|
||||
const formatAct = this.formatPages(act);
|
||||
let abTestArray: ABTest[] = [];
|
||||
try {
|
||||
if (formatAct.abTestRaw) {
|
||||
abTestArray = JSON.parse(formatAct.abTestRaw);
|
||||
}
|
||||
formatAct.abTest = abTestArray;
|
||||
delete formatAct.abTestRaw;
|
||||
return formatAct;
|
||||
} catch (error) {
|
||||
logger.error(error);
|
||||
throw new Error('根据id查询活动详情失败');
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* 新建活动
|
||||
* @param {ActInfoDetail} actInfo 新建活动所需信息
|
||||
* @returns {number} 活动id
|
||||
*/
|
||||
create = async (actInfo: ActInfoDetail) => {
|
||||
SequelizeHelper.getInstance();
|
||||
|
||||
// 新增活动
|
||||
const newAct = {
|
||||
actModifyTime: getFormatTime(),
|
||||
actCreateTime: getFormatTime(),
|
||||
actStatus: ActStatus.MODIFYING,
|
||||
actCryptoId: '', // 数据库不能为空,先做个占位符
|
||||
...actInfo,
|
||||
};
|
||||
const act = await ActInfo.create<ActInfo>(newAct as ActInfo);
|
||||
|
||||
// 更新加密id
|
||||
const cryptoId = Crypto.encode(act.actId.toString());
|
||||
await ActInfo.update(
|
||||
{ actCryptoId: cryptoId },
|
||||
{
|
||||
where: { actId: act.actId },
|
||||
},
|
||||
);
|
||||
|
||||
// 添加默认活动页
|
||||
const defaultPage = await this.pageService.create({
|
||||
actId: act.actId,
|
||||
});
|
||||
await act.$add<Page>('Pages', defaultPage);
|
||||
|
||||
return act.actId;
|
||||
};
|
||||
|
||||
/**
|
||||
* 复制活动
|
||||
* @param {CopyInfo} copyInfo 复制活动所需信息
|
||||
* @returns void
|
||||
*/
|
||||
copy = async (copyInfo: CopyInfo) => {
|
||||
SequelizeHelper.getInstance();
|
||||
|
||||
const targetAct = await ActInfo.findOne({
|
||||
where: { actId: copyInfo.actId },
|
||||
include: Page,
|
||||
});
|
||||
|
||||
if (!targetAct) throw new Error('源活动不存在');
|
||||
|
||||
const { actName, actBeginTime, actEndTime } = targetAct;
|
||||
const newAct = {
|
||||
actName: `【复制】${actName}`,
|
||||
actBeginTime,
|
||||
actEndTime,
|
||||
actModifyTime: getFormatTime(),
|
||||
actCreateTime: getFormatTime(),
|
||||
operator: copyInfo.userName,
|
||||
actStatus: ActStatus.MODIFYING,
|
||||
};
|
||||
|
||||
const act = await ActInfo.create<ActInfo>(newAct as ActInfo);
|
||||
|
||||
// 更新加密id
|
||||
const cryptoId = Crypto.encode(act.actId.toString());
|
||||
await ActInfo.update(
|
||||
{ actCryptoId: cryptoId },
|
||||
{
|
||||
where: { actId: act.actId },
|
||||
},
|
||||
);
|
||||
|
||||
// 复制page
|
||||
if (targetAct.pages) {
|
||||
await Promise.all(
|
||||
targetAct.pages.map(async (page) => {
|
||||
const copyPage = await this.pageService.create(page);
|
||||
await act.$add<Page>('Pages', copyPage);
|
||||
}),
|
||||
);
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* 更新活动信息
|
||||
* @param {ActBaseInfo} actInfo 需要更新的活动信息
|
||||
* @returns {Res} 活动基本信息
|
||||
*/
|
||||
update = async (actInfo: ActBaseInfo) => {
|
||||
try {
|
||||
if (actInfo.abTest?.length) {
|
||||
actInfo.abTestRaw = JSON.stringify(actInfo.abTest);
|
||||
}
|
||||
SequelizeHelper.getInstance();
|
||||
const updateActInfo = Object.assign(actInfo, {
|
||||
actModifyTime: getFormatTime(),
|
||||
actStatus: ActStatus.MODIFYING,
|
||||
});
|
||||
|
||||
await ActInfo.update(
|
||||
{
|
||||
...updateActInfo,
|
||||
},
|
||||
{
|
||||
where: { actId: actInfo.actId },
|
||||
},
|
||||
);
|
||||
} catch (error) {
|
||||
logger.error(`活动基础信息保存失败:${error}`);
|
||||
throw new Error('活动基础信息保存失败');
|
||||
}
|
||||
return { ret: 0, msg: '活动基础信息保存成功' };
|
||||
};
|
||||
|
||||
/**
|
||||
* 发布活动信息
|
||||
* @param {ActBaseInfo} actInfo 需要发布的活动信息
|
||||
* @param {number} actStatus 活动状态
|
||||
* @returns {Res} 活动基本信息
|
||||
*/
|
||||
publish = async (actInfo: ActBaseInfo, actStatus: number) => {
|
||||
try {
|
||||
const updateActInfo = Object.assign(actInfo, {
|
||||
actModifyTime: getFormatTime(),
|
||||
actStatus,
|
||||
});
|
||||
await ActInfo.update(
|
||||
{
|
||||
...updateActInfo,
|
||||
},
|
||||
{
|
||||
where: { actId: actInfo.actId },
|
||||
},
|
||||
);
|
||||
} catch (error) {
|
||||
logger.error(`活动基础信息发布失败:${error}`);
|
||||
throw new Error('活动基础信息发布失败');
|
||||
}
|
||||
return { ret: 0, msg: '活动基础信息发布成功' };
|
||||
};
|
||||
|
||||
/**
|
||||
* 根据页面ID删除活动页面
|
||||
* @param {number} pageId 页面ID
|
||||
* @returns void
|
||||
*/
|
||||
removePage = async (pageId: number) => {
|
||||
try {
|
||||
await Page.destroy({
|
||||
where: {
|
||||
id: pageId,
|
||||
},
|
||||
});
|
||||
} catch (error) {
|
||||
logger.error(error);
|
||||
throw new Error('删除活动页面失败');
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* 格式化pages的code
|
||||
* @param {ActInfo} act 待格式化内容
|
||||
* @returns {ActInfo} 格式化之后的结果
|
||||
*/
|
||||
formatPages = (act: ActInfoIncludePage): ActInfoIncludePage => {
|
||||
const newAct: ActInfoIncludePage = cloneDeep(act);
|
||||
if (act.pages) {
|
||||
newAct.pages = act.pages.map((page: PageInfo) => this.pageService.formatCode(page));
|
||||
}
|
||||
return newAct;
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* 按活动号查询,关联page数据
|
||||
* @param {number} actId 活动id
|
||||
* @returns {ActInfoIncludePage} 查询结果
|
||||
*/
|
||||
const getActInfoHandler = async (actId: number): Promise<ActInfoIncludePage> => {
|
||||
SequelizeHelper.getInstance();
|
||||
const act = await ActInfo.findOne({
|
||||
where: { actId },
|
||||
include: Page,
|
||||
});
|
||||
if (!act) throw new Error('活动不存在');
|
||||
return act.toJSON() as ActInfoIncludePage;
|
||||
};
|
||||
|
||||
// 将原始查询参数转化为构建查询体的参数
|
||||
const formatSearch = (search: string) => search || '';
|
||||
|
||||
const formatActIdQuery = (search: string) => {
|
||||
const isNumber = search && !isNaN(parseInt(search, 10));
|
||||
return isNumber ? { actId: parseInt(search, 10) } : {};
|
||||
};
|
||||
|
||||
const formatActStatusQuery = (status: number) =>
|
||||
status === ActStatus.ALL ? { [Op.in]: [ActStatus.MODIFYING, ActStatus.PART_PUBLISHED, ActStatus.PUBLISHED] } : status;
|
||||
|
||||
const formatOrder = (order: OrderItem) =>
|
||||
order.columnName ? [[order.columnName, order.direction === 'descending' ? 'DESC' : 'ASC']] : [];
|
||||
|
||||
const formatPageTitle = (title: string) =>
|
||||
title
|
||||
? {
|
||||
pageTitle: { [Op.substring]: title },
|
||||
}
|
||||
: null;
|
||||
|
||||
// 根据查询参数类型返回对应的转化结果
|
||||
const formatQuery = (query: string | OrderItem | number, type: FormatType) => {
|
||||
const formats = {
|
||||
search: formatSearch,
|
||||
actId: formatActIdQuery,
|
||||
status: formatActStatusQuery,
|
||||
order: formatOrder,
|
||||
title: formatPageTitle,
|
||||
};
|
||||
return formats[type]?.call(this, query);
|
||||
};
|
@ -1,27 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import componetList from '@src/template/editor/get-component-list';
|
||||
import webPlugins from '@src/template/editor/get-web-plugins';
|
||||
|
||||
export default class EditorService {
|
||||
// 获取组件列表
|
||||
getComponentList = () => componetList;
|
||||
// 获取插件列表
|
||||
getWebPlugins = () => webPlugins;
|
||||
}
|
@ -1,193 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { cloneDeep } from 'lodash';
|
||||
|
||||
import { PageStatus } from '@src/config/config';
|
||||
import { Page } from '@src/models/page';
|
||||
import SequelizeHelper from '@src/sequelize/index';
|
||||
import type { PageInfo } from '@src/typings';
|
||||
import { configTransformDist, getFormatTime, serializeConfig } from '@src/utils/index';
|
||||
import logger from '@src/utils/logger';
|
||||
|
||||
export default class PageService {
|
||||
/**
|
||||
* 新建页面
|
||||
* @param {Page} page 页面参数
|
||||
* @returns {Page} 新建页面
|
||||
*/
|
||||
create = (page?: Partial<Page>) => {
|
||||
const newPage = {
|
||||
pageCreateTime: getFormatTime(),
|
||||
pageModifyTime: getFormatTime(),
|
||||
pagePublishStatus: PageStatus.MODIFYING,
|
||||
pageTitle: 'index',
|
||||
pageName: 'index',
|
||||
...page,
|
||||
};
|
||||
return Page.create(newPage as Page);
|
||||
};
|
||||
|
||||
/**
|
||||
* 判断是否buffer类型
|
||||
* @param {any} val 待判定参数
|
||||
* @returns {Boolean} 判断结果
|
||||
*/
|
||||
isBuffer = (val: any) => val && typeof val === 'object' && Buffer.isBuffer(val);
|
||||
|
||||
/**
|
||||
* 格式化code,将srcCode和distCode转化为string
|
||||
* @param {PageInfo} page 待格式化的页面对象
|
||||
* @returns {PageInfo} 格式化之后的结果
|
||||
*/
|
||||
formatCode = (page?: PageInfo): PageInfo => {
|
||||
const newPage: PageInfo = cloneDeep(page);
|
||||
if (this.isBuffer(page.distCode)) newPage.distCode = page.distCode.toString();
|
||||
if (this.isBuffer(page.srcCode)) newPage.srcCode = page.srcCode.toString();
|
||||
if (page.pageCreateTime) newPage.pageCreateTime = getFormatTime(page.pageCreateTime);
|
||||
if (page.pageModifyTime) newPage.pageModifyTime = getFormatTime(page.pageModifyTime);
|
||||
if (page.pagePublishTime) newPage.pagePublishTime = getFormatTime(page.pagePublishTime);
|
||||
return newPage;
|
||||
};
|
||||
|
||||
/**
|
||||
* 更新页面信息
|
||||
* @param {PageInfo} page 待更新页面数组
|
||||
* @param {Boolean} isPublish 是否发布
|
||||
* @param {String} operator 操作人
|
||||
* @returns {Res} 结果返回
|
||||
*/
|
||||
update = async (pages: PageInfo[], isPublish = false, operator = '') => {
|
||||
const sequelize = SequelizeHelper.getInstance();
|
||||
const pageColUpdate: PageInfo = {};
|
||||
if (!isPublish) {
|
||||
pageColUpdate.pagePublishStatus = PageStatus.MODIFYING;
|
||||
pageColUpdate.pageModifyTime = getFormatTime();
|
||||
} else {
|
||||
pageColUpdate.pagePublishStatus = PageStatus.PUBLISHED;
|
||||
pageColUpdate.pagePublishTime = getFormatTime();
|
||||
pageColUpdate.pagePublishOperator = operator;
|
||||
}
|
||||
try {
|
||||
await sequelize.transaction(() => {
|
||||
Promise.all(
|
||||
pages.map(async (page: PageInfo) => {
|
||||
// 如果page.id不是纯数字,说明是新建的页面,需要补入一些字段
|
||||
const isNewPage = !/^\d+$/.test(page.id);
|
||||
if (isNewPage) {
|
||||
pageColUpdate.pageCreateTime = getFormatTime();
|
||||
pageColUpdate.pageTitle = page.pageName;
|
||||
}
|
||||
const upsertPage = Object.assign(page, pageColUpdate) as Page;
|
||||
// page更新到数据库
|
||||
await Page.upsert(upsertPage);
|
||||
if (isNewPage) {
|
||||
// 将新分配的Pageid回写至srcCode,distCode
|
||||
const newPageIdRes = await Page.findOne({
|
||||
where: {
|
||||
actId: page.actId,
|
||||
pageTitle: page.pageTitle,
|
||||
},
|
||||
attributes: ['id'],
|
||||
});
|
||||
const newPageId = newPageIdRes.toJSON();
|
||||
await this.updateCode(newPageId.id, page);
|
||||
}
|
||||
}),
|
||||
);
|
||||
});
|
||||
return { ret: 0, msg: '页面配置更新成功' };
|
||||
} catch (e) {
|
||||
logger.error(`页面配置更新失败:${e}`);
|
||||
throw new Error('页面配置更新失败');
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* 查询页面
|
||||
* @param {number} actId 活动Id
|
||||
* @param {string[]} publishPages 待发布页面数组
|
||||
* @returns {PageInfo[]} 查询结果
|
||||
*/
|
||||
getPages = async (actId: number, publishPages: string[]): Promise<PageInfo[]> => {
|
||||
SequelizeHelper.getInstance();
|
||||
const pages = await Page.findAll(
|
||||
publishPages.length > 0
|
||||
? {
|
||||
where: {
|
||||
actId,
|
||||
pageName: publishPages,
|
||||
},
|
||||
raw: true,
|
||||
}
|
||||
: {
|
||||
where: {
|
||||
actId,
|
||||
},
|
||||
raw: true,
|
||||
},
|
||||
);
|
||||
|
||||
return pages.map((page) => this.formatCode(page));
|
||||
};
|
||||
|
||||
/**
|
||||
* 查询页面数量
|
||||
* @param {number} actId 活动Id
|
||||
* @returns {number} 页面数量
|
||||
*/
|
||||
getPagesCount = async (actId: number): Promise<number> => {
|
||||
SequelizeHelper.getInstance();
|
||||
const pages = await Page.findAll({
|
||||
where: {
|
||||
actId,
|
||||
},
|
||||
raw: true,
|
||||
});
|
||||
return pages.length;
|
||||
};
|
||||
|
||||
/**
|
||||
* 将新分配的Pageid回写至srcCode,distCode
|
||||
* @param {string} pageId 页面id
|
||||
* @param {PageInfo} page 页面配置
|
||||
* @returns void
|
||||
*/
|
||||
updateCode = async (pageId: string, page: PageInfo) => {
|
||||
try {
|
||||
// 不是真正的json对象,可能包含组件自定义code代码
|
||||
// eslint-disable-next-line no-eval
|
||||
const srcCode = eval(`(${page.srcCode})`);
|
||||
srcCode.items[0].id = pageId;
|
||||
const newSrcCode = serializeConfig(srcCode);
|
||||
const newDistCode = configTransformDist(newSrcCode);
|
||||
await Page.update(
|
||||
{
|
||||
srcCode: newSrcCode,
|
||||
distCode: newDistCode,
|
||||
},
|
||||
{
|
||||
where: { id: pageId },
|
||||
},
|
||||
);
|
||||
} catch (error) {
|
||||
logger.error(error);
|
||||
throw new Error('保存新添加的页面失败');
|
||||
}
|
||||
};
|
||||
}
|
@ -1,273 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { copySync, emptyDir, outputFileSync, readFileSync } from 'fs-extra';
|
||||
import { cloneDeep } from 'lodash';
|
||||
|
||||
import { ActStatus, StaticPath, UiRuntimeJS } from '@src/config/config';
|
||||
import ActService from '@src/service/act';
|
||||
import PageService from '@src/service/page';
|
||||
import type { ActBaseInfo, PageInfo, UiConfig } from '@src/typings';
|
||||
import { configTransformDist, processTransConfig, serializeConfig } from '@src/utils/index';
|
||||
import logger from '@src/utils/logger';
|
||||
|
||||
const actService = new ActService();
|
||||
const pageService = new PageService();
|
||||
type DivideConfig = {
|
||||
pageName: string;
|
||||
pageConfig: UiConfig;
|
||||
};
|
||||
type DivideConfigTrans = {
|
||||
pageName: string;
|
||||
pageConfigStr: string;
|
||||
};
|
||||
|
||||
export default class PublishService {
|
||||
/**
|
||||
* 保存
|
||||
* @param {ActBaseInfo} actInfo 活动信息
|
||||
* @param {string} rootInfo uiconfig页面配置信息
|
||||
* @returns {Res} 保存结果
|
||||
*/
|
||||
saveActInfo = async ({ actInfo, rootInfo }: { actInfo: ActBaseInfo; rootInfo: string }) => {
|
||||
// 保存活动基础信息
|
||||
await actService.update(actInfo);
|
||||
try {
|
||||
// 不是真正的json对象,可能包含组件自定义code代码
|
||||
// eslint-disable-next-line no-eval
|
||||
const uiConfig = eval(`(${rootInfo})`);
|
||||
// 按页面拆分 abtest等发布时处理
|
||||
const divideConfigs = dividePageProcessor(uiConfig);
|
||||
// 处理为保存所需格式
|
||||
const pagesForSave = pageSaveProcessor(divideConfigs, actInfo);
|
||||
// 保存到magic_ui_config表
|
||||
await pageService.update(pagesForSave);
|
||||
return { ret: 0, msg: '保存成功' };
|
||||
} catch (error) {
|
||||
logger.error(error);
|
||||
throw new Error('保存活动失败');
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* 发布
|
||||
* @param {number} actId 活动Id
|
||||
* @param {string[]} publishPages 待发布页面
|
||||
* @param {string} operator 发布人
|
||||
* @returns {Res} 发布结果
|
||||
*/
|
||||
publish = async ({
|
||||
actId,
|
||||
publishPages,
|
||||
operator,
|
||||
}: {
|
||||
actId: number;
|
||||
publishPages: string[];
|
||||
rootInfo: string;
|
||||
operator: string;
|
||||
}) => {
|
||||
try {
|
||||
// 查询活动基础信息
|
||||
const actInfo = await actService.getActInfo(actId);
|
||||
// 查询需要发布的页面
|
||||
const pagesToPublish = await pageService.getPages(actId, publishPages);
|
||||
let publishPagesArr: DivideConfigTrans[] = pagesToPublish.map((pageItem) => ({
|
||||
pageName: pageItem.pageName,
|
||||
pageConfigStr: pageItem.distCode,
|
||||
}));
|
||||
// 处理abtest页面
|
||||
const abTestConfig: DivideConfigTrans[] = await getAbTestConfig(publishPages, actInfo.abTest, actId);
|
||||
publishPagesArr = publishPagesArr.concat(abTestConfig);
|
||||
// 发布
|
||||
await publishToFiles(publishPagesArr);
|
||||
// 更新页面状态
|
||||
await pageService.update(pagesToPublish, true, operator);
|
||||
// 确认页面是否全部发布
|
||||
let actStatus = ActStatus.PART_PUBLISHED;
|
||||
const singlePageCount = await pageService.getPagesCount(actId);
|
||||
const abTestCount = actInfo.abTest.length;
|
||||
const allPagesCount = singlePageCount + abTestCount;
|
||||
if (allPagesCount === publishPages.length) {
|
||||
actStatus = ActStatus.PUBLISHED;
|
||||
}
|
||||
await actService.publish(actInfo, actStatus);
|
||||
return { ret: 0, msg: '发布成功' };
|
||||
} catch (error) {
|
||||
logger.error(error);
|
||||
throw new Error('发布失败');
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* 配置信息拆分
|
||||
* @param {UiConfig} uiConfig 按页面拆分后的配置数组,每个都是完整的uiconfig,可直接发布
|
||||
* @returns {DivideConfig[]} 拆分后的页面数组
|
||||
*/
|
||||
const dividePageProcessor = (uiConfig: UiConfig): DivideConfig[] => {
|
||||
const deployPages: DivideConfig[] = [];
|
||||
// 按页面拆分uiConfig
|
||||
for (let i = 0; i < uiConfig.items.length; i++) {
|
||||
// 深复制一份
|
||||
const pageConfig = cloneDeep(uiConfig);
|
||||
pageConfig.items = [pageConfig.items[i]];
|
||||
deployPages.push({
|
||||
pageName: uiConfig.items[i].name,
|
||||
pageConfig,
|
||||
});
|
||||
}
|
||||
return deployPages;
|
||||
};
|
||||
|
||||
/**
|
||||
* 获取abtest包含的页面配置
|
||||
* @param {ActBaseInfo.abTest} abTest abtest配置
|
||||
* @param {string[]} publishPages 待发布页面
|
||||
* @param {number} actId 活动ID
|
||||
* @returns {DivideConfig[]} abtest拆分后的页面数组
|
||||
*/
|
||||
const getAbTestConfig = async (
|
||||
publishPages: string[],
|
||||
abTest: ActBaseInfo['abTest'],
|
||||
actId: number,
|
||||
): Promise<DivideConfigTrans[]> => {
|
||||
const filterArr = filterIncludeAbTest(publishPages, abTest);
|
||||
if (filterArr.length === 0) {
|
||||
return [];
|
||||
}
|
||||
return await Promise.all(
|
||||
filterArr.map(async (test) => {
|
||||
let pagesToPublish: UiConfig;
|
||||
await Promise.all(
|
||||
test.pageList.map(async (testItem) => {
|
||||
const [pageRecord] = await pageService.getPages(actId, [testItem.pageName]);
|
||||
// 不是真正的json对象,可能包含组件自定义code代码
|
||||
// eslint-disable-next-line no-eval
|
||||
const pageConfig = eval(`(${pageRecord.distCode})`);
|
||||
if (!pagesToPublish) {
|
||||
pagesToPublish = pageConfig;
|
||||
} else {
|
||||
pagesToPublish.items.push(pageConfig?.items[0]);
|
||||
}
|
||||
}),
|
||||
);
|
||||
// 将config对象序列化并转译
|
||||
const srcCode = serializeConfig(pagesToPublish);
|
||||
const distCode = configTransformDist(srcCode);
|
||||
return {
|
||||
pageName: test.name,
|
||||
pageConfigStr: distCode,
|
||||
};
|
||||
}),
|
||||
);
|
||||
};
|
||||
|
||||
/**
|
||||
* 过滤出发布页面包含的abtest页面
|
||||
* @param {ActBaseInfo.abTest} abTest abtest配置
|
||||
* @param {string[]} publishPages 待发布页面
|
||||
* @returns {ActBaseInfo['abTest']} 过滤结果
|
||||
*/
|
||||
const filterIncludeAbTest = (publishPages: string[], abTest: ActBaseInfo['abTest']) => {
|
||||
if (abTest.length === 0) {
|
||||
return [];
|
||||
}
|
||||
const filterArr = [];
|
||||
abTest.forEach((test) => {
|
||||
if (publishPages.includes(test.name)) {
|
||||
filterArr.push(test);
|
||||
}
|
||||
});
|
||||
return filterArr;
|
||||
};
|
||||
|
||||
/**
|
||||
* 处理为保存所需格式
|
||||
* @param {DivideConfig[]} divideConfigs 按页面拆分后的配置数组,每个都是完整的uiconfig,可直接发布
|
||||
* @param {ActBaseInfo} actInfo 活动基本信息
|
||||
* @returns {PageInfo[]} 保存所需的PageInfo数组
|
||||
*/
|
||||
const pageSaveProcessor = (divideConfigs: DivideConfig[], actInfo: ActBaseInfo): PageInfo[] =>
|
||||
divideConfigs.map((divideItem) => {
|
||||
const srcCode = serializeConfig(divideItem.pageConfig);
|
||||
const distCode = configTransformDist(srcCode);
|
||||
return {
|
||||
pageTitle: divideItem.pageConfig.items[0].title || divideItem.pageConfig.items[0].name,
|
||||
pageName: divideItem.pageName,
|
||||
srcCode,
|
||||
distCode,
|
||||
actId: actInfo.actId,
|
||||
id: divideItem.pageConfig.items[0].id,
|
||||
};
|
||||
});
|
||||
|
||||
/**
|
||||
* 发布活动配置文件
|
||||
* @param {DivideConfigTrans} pageConfig 发布页面的UiConfig活动配置
|
||||
* @returns void
|
||||
*/
|
||||
const publishUiconfig = (pageConfig: DivideConfigTrans) => {
|
||||
try {
|
||||
const { pageName, pageConfigStr } = pageConfig;
|
||||
const distJs = `${StaticPath.PUBLISH}/uiconfig_${pageName}.js`;
|
||||
const code = processTransConfig(pageConfigStr);
|
||||
outputFileSync(distJs, code);
|
||||
logger.debug(`create ${distJs} success!`);
|
||||
} catch (error) {
|
||||
logger.error(error);
|
||||
throw new Error('发布活动配置文件失败');
|
||||
}
|
||||
};
|
||||
/**
|
||||
* 发布活动html文件
|
||||
* @param {DivideConfigTrans} pageConfig 发布页面的UiConfig活动配置
|
||||
* @returns void
|
||||
*/
|
||||
const publishHtml = (pageConfig: DivideConfigTrans) => {
|
||||
const { pageName } = pageConfig;
|
||||
try {
|
||||
// 复制html模板
|
||||
const distHtml = `${StaticPath.PUBLISH}/${pageName}.html`;
|
||||
const tmpHtml = `${StaticPath.STATIC}/vue3/runtime/page/index.html`;
|
||||
copySync(tmpHtml, distHtml);
|
||||
// 注入活动配置文件
|
||||
const configScript = `<script type='module' src='./uiconfig_${pageName}.js'></script>\n\t${UiRuntimeJS}`;
|
||||
const data = readFileSync(distHtml, 'utf-8');
|
||||
const newData = data.replace(UiRuntimeJS, configScript);
|
||||
outputFileSync(distHtml, newData, 'utf-8');
|
||||
logger.debug(`create ${distHtml} success!`);
|
||||
} catch (error) {
|
||||
throw new Error('发布活动html文件失败');
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* 发布文件
|
||||
* @param {DivideConfigTrans[]} publishPagesArr 每个发布页面的UiConfig
|
||||
* @returns void
|
||||
*/
|
||||
const publishToFiles = async (publishPagesArr: DivideConfigTrans[]) => {
|
||||
// 1. 文件夹清空并重新创建
|
||||
await emptyDir(StaticPath.PUBLISH);
|
||||
publishPagesArr.forEach((divideConfig) => {
|
||||
// 2、发布uiconfig.js
|
||||
publishUiconfig(divideConfig);
|
||||
// 3、发布html模板
|
||||
publishHtml(divideConfig);
|
||||
});
|
||||
};
|
@ -1,45 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
export default [
|
||||
{
|
||||
title: '容器',
|
||||
items: [
|
||||
{
|
||||
icon: 'folder-opened',
|
||||
text: '组',
|
||||
type: 'container',
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
title: '示例组件',
|
||||
items: [
|
||||
{
|
||||
icon: 'tickets',
|
||||
text: '文本',
|
||||
type: 'text',
|
||||
},
|
||||
{
|
||||
icon: 'switch-button',
|
||||
text: '按钮',
|
||||
type: 'button',
|
||||
},
|
||||
],
|
||||
},
|
||||
];
|
@ -1,26 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
export default {
|
||||
options: [
|
||||
{
|
||||
text: '会员登录认证',
|
||||
value: '会员登录认证',
|
||||
},
|
||||
],
|
||||
};
|
98
magic-admin/server/src/typings/index.d.ts
vendored
98
magic-admin/server/src/typings/index.d.ts
vendored
@ -1,98 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 活动信息定义
|
||||
export interface ActBaseInfo {
|
||||
actId: number;
|
||||
actCryptoId: string;
|
||||
actName: string;
|
||||
operator?: string;
|
||||
actStatus: number;
|
||||
actBeginTime?: string;
|
||||
actEndTime?: string;
|
||||
actModifyTime?: string;
|
||||
actCreateTime?: string;
|
||||
locker?: string;
|
||||
lockTime?: string;
|
||||
abTest?: ABTest[];
|
||||
abTestRaw?: string;
|
||||
plugins?: string[];
|
||||
id?: string;
|
||||
name?: string;
|
||||
}
|
||||
|
||||
// 查询处理后的活动信息,含页面配置
|
||||
export interface ActInfoIncludePage extends ActBaseInfo {
|
||||
pages: PageInfo[];
|
||||
}
|
||||
// 编辑器组件配置定义
|
||||
export interface UiConfig {
|
||||
actId: number;
|
||||
type?: string;
|
||||
id?: string;
|
||||
name?: string;
|
||||
operator?: string;
|
||||
items?: PageInfo[];
|
||||
abTest?: ABTest[];
|
||||
useLastPackage?: string;
|
||||
}
|
||||
|
||||
// 活动页面abtest定义
|
||||
export interface ABTest {
|
||||
name: string;
|
||||
type: string;
|
||||
pageList?: AbTestPageList[];
|
||||
}
|
||||
|
||||
// 活动页面abtest pagelist定义
|
||||
export interface AbTestPageList {
|
||||
pageName: string;
|
||||
proportion: string;
|
||||
}
|
||||
|
||||
// 活动页面信息定义
|
||||
export interface PageInfo {
|
||||
id?: string;
|
||||
pageTitle?: string;
|
||||
title?: string;
|
||||
name?: string;
|
||||
pageName?: string;
|
||||
actId?: number;
|
||||
pageCreateTime?: string;
|
||||
pageModifyTime?: string;
|
||||
pagePublishStatus?: number;
|
||||
pagePublishTime?: string;
|
||||
pagePublishOperator?: string;
|
||||
pagePublishUiVersion?: string;
|
||||
srcCode?: string;
|
||||
distCode?: string;
|
||||
plugins?: string[];
|
||||
}
|
||||
|
||||
// 从editor拿到的活动页面信息
|
||||
export interface EditorInfo {
|
||||
type: string;
|
||||
items?: PageInfo[];
|
||||
}
|
||||
|
||||
// 接口返回
|
||||
export interface Res<T = any> {
|
||||
ret: number;
|
||||
msg: string;
|
||||
data?: T;
|
||||
}
|
@ -1,52 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
// 活动加密
|
||||
import myCrypto from 'crypto';
|
||||
|
||||
import config from '@src/config/key';
|
||||
|
||||
const algorithm = 'aes-256-cbc';
|
||||
const keyStr = config.key;
|
||||
const ivByte = Buffer.from(keyStr.substr(0, 16));
|
||||
|
||||
function encrypt(text: string) {
|
||||
const cipher = myCrypto.createCipheriv(algorithm, Buffer.from(keyStr), ivByte);
|
||||
let encrypted = cipher.update(text);
|
||||
encrypted = Buffer.concat([encrypted, cipher.final()]);
|
||||
return encrypted.toString('hex');
|
||||
}
|
||||
|
||||
function decrypt(text: string) {
|
||||
const encryptedData = text;
|
||||
const encryptedText = Buffer.from(encryptedData, 'hex');
|
||||
const decipher = myCrypto.createDecipheriv(algorithm, Buffer.from(keyStr), ivByte);
|
||||
let decrypted = decipher.update(encryptedText);
|
||||
decrypted = Buffer.concat([decrypted, decipher.final()]);
|
||||
return decrypted.toString();
|
||||
}
|
||||
|
||||
export default {
|
||||
encode(text: string): string {
|
||||
return encrypt(text);
|
||||
},
|
||||
|
||||
decode(text: string): string {
|
||||
return decrypt(text);
|
||||
},
|
||||
};
|
@ -1,101 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import path from 'path';
|
||||
|
||||
import axios from 'axios';
|
||||
import dayjs from 'dayjs';
|
||||
import timezone from 'dayjs/plugin/timezone';
|
||||
import utc from 'dayjs/plugin/utc';
|
||||
import { createWriteStream, emptyDir } from 'fs-extra';
|
||||
import serialize from 'serialize-javascript';
|
||||
import uglifyJS from 'uglify-js';
|
||||
|
||||
import type { UiConfig } from '@src/typings';
|
||||
import { babelTransform } from '@src/utils/transform';
|
||||
|
||||
dayjs.extend(utc);
|
||||
dayjs.extend(timezone);
|
||||
|
||||
/**
|
||||
* 格式化配置内容
|
||||
* @param {string} value 待格式化内容
|
||||
* @returns {string} 格式化结果
|
||||
*/
|
||||
const serializeConfig = function (value: UiConfig): string {
|
||||
return serialize(value, {
|
||||
space: 2,
|
||||
unsafe: true,
|
||||
}).replace(/"(\w+)":\s/g, '$1: ');
|
||||
};
|
||||
|
||||
/**
|
||||
* 将srccode转换为distcode
|
||||
* @param {string} srcCode srcCode
|
||||
* @returns {string} distcode
|
||||
*/
|
||||
const configTransformDist = (srcCode: string): string => {
|
||||
let babelCode: string = babelTransform(`window.uiConfig=[${srcCode}]`);
|
||||
babelCode = babelCode.replace('window.uiConfig = [', '');
|
||||
return babelCode.substring(0, babelCode.length - 2);
|
||||
};
|
||||
|
||||
/**
|
||||
* uglifyJS处理distcode
|
||||
* @param {string} transConfig transConfig
|
||||
* @returns {string} 处理结果
|
||||
*/
|
||||
const processTransConfig = (transConfig) => {
|
||||
const code = `window.magicDSL = [${transConfig}]`;
|
||||
return uglifyJS.minify(`${code}`).code;
|
||||
};
|
||||
|
||||
/**
|
||||
* 下载文件到本地
|
||||
* @param {string} url 文件下载地址
|
||||
* @param {string} filePath 文件保存目录
|
||||
* @param {string} fileName 文件名
|
||||
* @returns {Promise} 处理结果
|
||||
*/
|
||||
const getFileFromUrl = async ({ url, filePath, fileName }) => {
|
||||
// 1. 文件夹清空并重新创建
|
||||
await emptyDir(filePath);
|
||||
const distPath = path.resolve(filePath, fileName);
|
||||
const writer = createWriteStream(distPath);
|
||||
const response = await axios({
|
||||
url,
|
||||
method: 'GET',
|
||||
responseType: 'stream',
|
||||
});
|
||||
response.data.pipe(writer);
|
||||
return new Promise((resolve, reject) => {
|
||||
writer.on('finish', resolve);
|
||||
writer.on('error', reject);
|
||||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* 格式化时间(上海时区)
|
||||
* @param {string} [time] 待转换时间戳,不传则获取当前时间
|
||||
* @param {string} [formatTmp] 转换格式,不传则默认使用'YYYY-MM-DD HH:mm:ss'
|
||||
* @returns {string} 格式化之后的时间
|
||||
*/
|
||||
const getFormatTime = (time: string | number = Date.now(), formatTmp = 'YYYY-MM-DD HH:mm:ss') =>
|
||||
dayjs.tz(time, 'Asia/Shanghai').format(formatTmp);
|
||||
|
||||
export { serializeConfig, configTransformDist, processTransConfig, getFileFromUrl, getFormatTime };
|
@ -1,39 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { getLogger } from 'log4js';
|
||||
|
||||
class Logger {
|
||||
public logger;
|
||||
constructor() {
|
||||
this.logger = getLogger();
|
||||
}
|
||||
info = (message: string) => {
|
||||
this.logger.info(message);
|
||||
};
|
||||
debug = (message: string) => {
|
||||
this.logger.debug(message);
|
||||
};
|
||||
warn = (message: string) => {
|
||||
this.logger.warn(message);
|
||||
};
|
||||
error = (message: string) => {
|
||||
this.logger.error(message);
|
||||
};
|
||||
}
|
||||
export default new Logger();
|
@ -1,43 +0,0 @@
|
||||
/*
|
||||
* Tencent is pleased to support the open source community by making MagicEditor available.
|
||||
*
|
||||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { transformSync } from '@babel/core';
|
||||
|
||||
import logger from '@src/utils/logger';
|
||||
export function babelTransform(value) {
|
||||
const options = {
|
||||
compact: false,
|
||||
presets: [
|
||||
[
|
||||
'@babel/preset-env',
|
||||
{
|
||||
modules: false,
|
||||
targets: {
|
||||
browsers: ['> 1%', 'last 2 versions', 'not ie <= 8'],
|
||||
},
|
||||
},
|
||||
],
|
||||
],
|
||||
};
|
||||
try {
|
||||
return transformSync(value, options)?.code || '';
|
||||
} catch (e) {
|
||||
logger.error(e);
|
||||
throw new Error('babel 编译失败');
|
||||
}
|
||||
}
|
@ -1,119 +0,0 @@
|
||||
(function (global, factory) {
|
||||
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
|
||||
typeof define === 'function' && define.amd ? define(factory) :
|
||||
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.magicPresetConfigs = factory());
|
||||
})(this, (function () { 'use strict';
|
||||
|
||||
const page = [
|
||||
{
|
||||
text: "\u9875\u9762\u6807\u8BC6",
|
||||
name: "name",
|
||||
disabled: true,
|
||||
extra: "\u5728\u591A\u9875\u9762\u7684\u60C5\u51B5\u4E0B\u7528\u6765\u6307\u5B9A\u8981\u6253\u5F00\u7684\u9875\u9762"
|
||||
},
|
||||
{
|
||||
text: "\u9875\u9762\u6807\u9898",
|
||||
name: "title"
|
||||
},
|
||||
{
|
||||
name: "layout",
|
||||
text: "\u5BB9\u5668\u5E03\u5C40",
|
||||
type: "select",
|
||||
defaultValue: "absolute",
|
||||
options: [
|
||||
{ value: "absolute", text: "\u7EDD\u5BF9\u5B9A\u4F4D" },
|
||||
{ value: "relative", text: "\u6D41\u5F0F\u5E03\u5C40" }
|
||||
],
|
||||
onChange: (formState, v, { model }) => {
|
||||
if (!model.style)
|
||||
return v;
|
||||
if (v === "relative") {
|
||||
model.style.height = "auto";
|
||||
} else {
|
||||
const el = formState.stage?.renderer?.contentWindow.document.getElementById(model.id);
|
||||
if (el) {
|
||||
model.style.height = el.getBoundingClientRect().height;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
];
|
||||
|
||||
const container = [
|
||||
{
|
||||
name: "layout",
|
||||
text: "\u5BB9\u5668\u5E03\u5C40",
|
||||
type: "select",
|
||||
defaultValue: "absolute",
|
||||
options: [
|
||||
{ value: "absolute", text: "\u7EDD\u5BF9\u5B9A\u4F4D" },
|
||||
{ value: "relative", text: "\u6D41\u5F0F\u5E03\u5C40" }
|
||||
],
|
||||
onChange: (formState, v, { model }) => {
|
||||
if (!model.style)
|
||||
return v;
|
||||
if (v === "relative") {
|
||||
model.style.height = "auto";
|
||||
} else {
|
||||
const el = formState.stage?.renderer?.contentWindow.document.getElementById(model.id);
|
||||
if (el) {
|
||||
model.style.height = el.getBoundingClientRect().height;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
];
|
||||
|
||||
const button = [
|
||||
{
|
||||
text: "\u6587\u672C",
|
||||
name: "text"
|
||||
}
|
||||
];
|
||||
|
||||
const text = [
|
||||
{
|
||||
name: "text",
|
||||
text: "\u6587\u672C"
|
||||
},
|
||||
{
|
||||
name: "multiple",
|
||||
text: "\u591A\u884C\u6587\u672C",
|
||||
type: "switch"
|
||||
}
|
||||
];
|
||||
|
||||
const img = [
|
||||
{
|
||||
text: "\u56FE\u7247",
|
||||
name: "src"
|
||||
},
|
||||
{
|
||||
text: "\u94FE\u63A5",
|
||||
name: "url"
|
||||
}
|
||||
];
|
||||
|
||||
const qrcode = [
|
||||
{
|
||||
text: "\u94FE\u63A5",
|
||||
name: "url"
|
||||
}
|
||||
];
|
||||
|
||||
const overlay = [];
|
||||
|
||||
const configs = {
|
||||
"page": page,
|
||||
"container": container,
|
||||
"button": button,
|
||||
"text": text,
|
||||
"img": img,
|
||||
"qrcode": qrcode,
|
||||
"overlay": overlay
|
||||
};
|
||||
|
||||
return configs;
|
||||
|
||||
}));
|
||||
//# sourceMappingURL=index.umd.cjs.map
|
File diff suppressed because one or more lines are too long
@ -1,47 +0,0 @@
|
||||
(function (global, factory) {
|
||||
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
|
||||
typeof define === 'function' && define.amd ? define(factory) :
|
||||
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.magicPresetEvents = factory());
|
||||
})(this, (function () { 'use strict';
|
||||
|
||||
const page = {
|
||||
methods: [
|
||||
{
|
||||
label: "\u5237\u65B0\u9875\u9762",
|
||||
value: "refresh"
|
||||
}
|
||||
]
|
||||
};
|
||||
|
||||
const overlay = {
|
||||
methods: [
|
||||
{
|
||||
label: "\u6253\u5F00\u8499\u5C42",
|
||||
value: "openOverlay"
|
||||
},
|
||||
{
|
||||
label: "\u5173\u95ED\u8499\u5C42",
|
||||
value: "closeOverlay"
|
||||
}
|
||||
],
|
||||
events: [
|
||||
{
|
||||
label: "\u6253\u5F00\u8499\u5C42",
|
||||
value: "overlay:open"
|
||||
},
|
||||
{
|
||||
label: "\u5173\u95ED\u8499\u5C42",
|
||||
value: "overlay:close"
|
||||
}
|
||||
]
|
||||
};
|
||||
|
||||
const events = {
|
||||
"page": page,
|
||||
"overlay": overlay
|
||||
};
|
||||
|
||||
return events;
|
||||
|
||||
}));
|
||||
//# sourceMappingURL=index.umd.cjs.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index.umd.cjs","sources":["../../../../../packages/ui/src/page/src/event.ts","../../../../../packages/ui/src/overlay/src/event.ts","../../../.tmagic/event-entry.ts"],"sourcesContent":["export default {\n methods: [\n {\n label: '刷新页面',\n value: 'refresh',\n },\n ],\n};\n","export default {\n methods: [\n {\n label: '打开蒙层',\n value: 'openOverlay',\n },\n {\n label: '关闭蒙层',\n value: 'closeOverlay',\n },\n ],\n events: [\n {\n label: '打开蒙层',\n value: 'overlay:open',\n },\n {\n label: '关闭蒙层',\n value: 'overlay:close',\n },\n ],\n};\n","import page from '/parisma/github/tmagic-editor/packages/ui/src/page/src/event';\nimport overlay from '/parisma/github/tmagic-editor/packages/ui/src/overlay/src/event';\n\nconst events: Record<string, any> = {\n 'page': page,\n 'overlay': overlay,\n};\n\nexport default events;"],"names":[],"mappings":";;;;;;AAAA,eAAe;EAAA,EACb,OAAS,EAAA;EAAA,IACP;EAAA,MACE,KAAO,EAAA,0BAAA;EAAA,MACP,KAAO,EAAA,SAAA;EAAA,KACT;EAAA,GACF;EACF,CAAA;;ACPA,kBAAe;EAAA,EACb,OAAS,EAAA;EAAA,IACP;EAAA,MACE,KAAO,EAAA,0BAAA;EAAA,MACP,KAAO,EAAA,aAAA;EAAA,KACT;EAAA,IACA;EAAA,MACE,KAAO,EAAA,0BAAA;EAAA,MACP,KAAO,EAAA,cAAA;EAAA,KACT;EAAA,GACF;EAAA,EACA,MAAQ,EAAA;EAAA,IACN;EAAA,MACE,KAAO,EAAA,0BAAA;EAAA,MACP,KAAO,EAAA,cAAA;EAAA,KACT;EAAA,IACA;EAAA,MACE,KAAO,EAAA,0BAAA;EAAA,MACP,KAAO,EAAA,eAAA;EAAA,KACT;EAAA,GACF;EACF,CAAA;;AClBA,QAAM,MAA8B,GAAA;EAAA,EAClC,MAAQ,EAAA,IAAA;EAAA,EACR,SAAW,EAAA,OAAA;EACb;;;;;;;;"}
|
@ -1,90 +0,0 @@
|
||||
(function (global, factory) {
|
||||
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
|
||||
typeof define === 'function' && define.amd ? define(factory) :
|
||||
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.magicPresetValues = factory());
|
||||
})(this, (function () { 'use strict';
|
||||
|
||||
const page = {
|
||||
items: [],
|
||||
style: {
|
||||
width: "100%",
|
||||
height: "100%"
|
||||
}
|
||||
};
|
||||
|
||||
const container = {
|
||||
items: [],
|
||||
style: {
|
||||
width: "375",
|
||||
height: "100"
|
||||
}
|
||||
};
|
||||
|
||||
const button = {
|
||||
text: "\u8BF7\u8F93\u5165\u6587\u672C\u5185\u5BB9",
|
||||
multiple: true,
|
||||
style: {
|
||||
width: "270",
|
||||
height: "37.5",
|
||||
border: 0,
|
||||
backgroundColor: "#fb6f00"
|
||||
}
|
||||
};
|
||||
|
||||
const text = {
|
||||
type: "text",
|
||||
text: "\u8BF7\u8F93\u5165\u6587\u672C\u5185\u5BB9",
|
||||
multiple: true,
|
||||
style: {
|
||||
width: "100",
|
||||
height: "auto"
|
||||
}
|
||||
};
|
||||
|
||||
const img = {
|
||||
src: "https://puui.qpic.cn/vupload/0/1573555382625_bhp0wud8l6w.png/0",
|
||||
url: "",
|
||||
style: {
|
||||
position: "absolute",
|
||||
left: "57",
|
||||
width: "176",
|
||||
height: "176"
|
||||
}
|
||||
};
|
||||
|
||||
const qrcode = {
|
||||
url: "https://m.film.qq.com",
|
||||
style: {
|
||||
position: "absolute",
|
||||
left: "57",
|
||||
width: "176",
|
||||
height: "176"
|
||||
}
|
||||
};
|
||||
|
||||
const overlay = {
|
||||
style: {
|
||||
position: "fixed",
|
||||
width: "100%",
|
||||
height: "100%",
|
||||
top: 0,
|
||||
left: 0,
|
||||
backgroundColor: "rgba(0, 0, 0, 0.8)"
|
||||
},
|
||||
items: []
|
||||
};
|
||||
|
||||
const values = {
|
||||
"page": page,
|
||||
"container": container,
|
||||
"button": button,
|
||||
"text": text,
|
||||
"img": img,
|
||||
"qrcode": qrcode,
|
||||
"overlay": overlay
|
||||
};
|
||||
|
||||
return values;
|
||||
|
||||
}));
|
||||
//# sourceMappingURL=index.umd.cjs.map
|
File diff suppressed because one or more lines are too long
@ -1,2 +0,0 @@
|
||||
System.register(["./index-legacy.1dc0be94.js"],(function(e,t){"use strict";var n,c;return{setters:[e=>{n=e.t,c=e._}],execute:function(){const t=Vue.defineComponent({props:{config:{type:Object,default:()=>({})}},setup(e){const t=Vue.getCurrentInstance()?.proxy,c=Vue.inject("app");return Vue.provide("hoc",t),{tagName:Vue.computed((()=>`magic-ui-${n(e.config.type)}`)),style:Vue.computed((()=>c?.transformStyle(e.config.style))),display:()=>{const t=e.config?.display;return"function"==typeof t?t(c):!1!==t}}}});e("C",c(t,[["render",function(e,t,n,c,o,i){return e.display()?(Vue.openBlock(),Vue.createBlock(Vue.resolveDynamicComponent(e.tagName),{key:0,id:e.config.id,class:Vue.normalizeClass("magic-ui-component"+(e.config.className?` ${e.config.className}`:"")),style:Vue.normalizeStyle(e.style),config:e.config},null,8,["id","class","style","config"])):Vue.createCommentVNode("v-if",!0)}],["__file","/parisma/github/tmagic-editor/packages/ui/src/Component.vue"]]))}}}));
|
||||
//# sourceMappingURL=Component-legacy.8992014a.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"Component-legacy.8992014a.js","sources":["../../../../../../packages/ui/src/Component.vue"],"sourcesContent":["<template>\n <component\n v-if=\"display()\"\n :is=\"tagName\"\n :id=\"config.id\"\n :class=\"`magic-ui-component${config.className ? ` ${config.className}` : ''}`\"\n :style=\"style\"\n :config=\"config\"\n ></component>\n</template>\n\n<script lang=\"ts\">\nimport { computed, defineComponent, getCurrentInstance, inject, provide } from 'vue';\n\nimport Core from '@tmagic/core';\nimport { toLine } from '@tmagic/utils';\n\nexport default defineComponent({\n props: {\n config: {\n type: Object,\n default: () => ({}),\n },\n },\n\n setup(props) {\n const vm = getCurrentInstance()?.proxy;\n const app: Core | undefined = inject('app');\n\n provide('hoc', vm);\n\n return {\n tagName: computed(() => `magic-ui-${toLine(props.config.type)}`),\n style: computed(() => app?.transformStyle(props.config.style)),\n\n display: () => {\n const displayCfg = props.config?.display;\n\n if (typeof displayCfg === 'function') {\n return displayCfg(app);\n }\n return displayCfg !== false;\n },\n };\n },\n});\n</script>\n"],"names":["Vue","defineComponent","props","config","type","Object","default","setup","vm","app","provide","tagName","style","display","displayCfg","_ctx","_cache","$props","$setup","$data","$options","createBlock","resolveDynamicComponent","key","id","class","normalizeClass","className","normalizeStyle"],"mappings":"wIAiBe,QAAAA,IAAgBC,gBAAA,CAAAC,MAAA,CACtBC,OAAA,CACGC,KAAAC,OACAC,QAAA,KAAA,CAAA,KAGVC,MAAAL,GAGE,MAAAM,EAAWR,gCACXS,EAA8BT,kBAI9B,OAFAA,IAAAU,QAAA,MAAAF,GAEA,CAAOG,QACIX,mDAAsDY,MACxDZ,sDAAsDa,QAAA,KAG3D,MAAAC,EAAAZ,EAAAC,QAAAU,QAEA,MAAA,mBAAAC,EACEA,EAAAL,IAEF,IAAAK,CAAA,EAEJ,yBA1CF,SAAAC,EAAAC,EAAAC,EAAAC,EAAAC,EAAAC,uCAGgBpB,IAAAqB,YAAArB,IAAAsB,wBAAAP,EAAAJ,SAAA,CAAAY,IAAA,EACRC,GAAAT,EAAAZ,OAAAqB,GACAC,MACCzB,IAAQ0B,eAAA,sBAAAX,EAAAZ,OAAAwB,UAAA,IAAAZ,EAAAZ,OAAAwB,YAAA,KAAAf,MAAAZ,IAAA4B,eAAAb,EAAAH"}
|
@ -1,2 +0,0 @@
|
||||
import{t as a,_ as c}from"./index.2a9a64ca.js";const r=Vue.defineComponent({props:{config:{type:Object,default:()=>({})}},setup(e){var t;const s=(t=Vue.getCurrentInstance())==null?void 0:t.proxy,n=Vue.inject("app");return Vue.provide("hoc",s),{tagName:Vue.computed(()=>`magic-ui-${a(e.config.type)}`),style:Vue.computed(()=>n==null?void 0:n.transformStyle(e.config.style)),display:()=>{var i;const o=(i=e.config)==null?void 0:i.display;return typeof o=="function"?o(n):o!==!1}}}});function u(e,s,n,t,o,i){return e.display()?(Vue.openBlock(),Vue.createBlock(Vue.resolveDynamicComponent(e.tagName),{key:0,id:e.config.id,class:Vue.normalizeClass(`magic-ui-component${e.config.className?` ${e.config.className}`:""}`),style:Vue.normalizeStyle(e.style),config:e.config},null,8,["id","class","style","config"])):Vue.createCommentVNode("v-if",!0)}const f=c(r,[["render",u],["__file","/parisma/github/tmagic-editor/packages/ui/src/Component.vue"]]);export{f as C};
|
||||
//# sourceMappingURL=Component.aeed2f79.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"Component.aeed2f79.js","sources":["../../../../../../packages/ui/src/Component.vue"],"sourcesContent":["<template>\n <component\n v-if=\"display()\"\n :is=\"tagName\"\n :id=\"config.id\"\n :class=\"`magic-ui-component${config.className ? ` ${config.className}` : ''}`\"\n :style=\"style\"\n :config=\"config\"\n ></component>\n</template>\n\n<script lang=\"ts\">\nimport { computed, defineComponent, getCurrentInstance, inject, provide } from 'vue';\n\nimport Core from '@tmagic/core';\nimport { toLine } from '@tmagic/utils';\n\nexport default defineComponent({\n props: {\n config: {\n type: Object,\n default: () => ({}),\n },\n },\n\n setup(props) {\n const vm = getCurrentInstance()?.proxy;\n const app: Core | undefined = inject('app');\n\n provide('hoc', vm);\n\n return {\n tagName: computed(() => `magic-ui-${toLine(props.config.type)}`),\n style: computed(() => app?.transformStyle(props.config.style)),\n\n display: () => {\n const displayCfg = props.config?.display;\n\n if (typeof displayCfg === 'function') {\n return displayCfg(app);\n }\n return displayCfg !== false;\n },\n };\n },\n});\n</script>\n"],"names":["_sfc_main","props","vm","_a","app","displayCfg","_sfc_render","_ctx","_cache","$props","$setup","$data","$options"],"mappings":"+CAiBA,MAAeA,EAAA,IAAgB,gBAAA,CAAA,MAAA,CACtB,OAAA,CACG,KAAA,OACA,QAAA,KAAA,CAAA,EACW,CACnB,EACF,MAAAC,EAAA,OAGE,MAAAC,GAAWC,EAAA,2BAAA,YAAAA,QACXC,EAA8B,kBAE9B,WAAA,QAAA,MAAAF,CAAA,EAEA,CAAO,QACI,iDAAsD,MACxD,kEAAsD,QAAA,IAAA,OAG3D,MAAAG,GAAAF,EAAAF,EAAA,SAAA,YAAAE,EAAA,QAEA,OAAA,OAAAE,GAAA,WACEA,EAAAD,CAAA,EAEFC,IAAA,EAAsB,CACxB,CACF,CAEJ,CAAA,EA5CE,SAAAC,EAAAC,EAAAC,EAAAC,EAAAC,EAAAC,EAAAC,EAAA,qCAGgB,IAAA,YAAA,IAAA,wBAAAL,EAAA,OAAA,EAAA,CAAA,IAAA,EACR,GAAAA,EAAA,OAAA,GACA,MACC,IAAQ,eAAA,qBAAAA,EAAA,OAAA,UAAA,IAAAA,EAAA,OAAA,YAAA,IAAA,EAAA,MAAA,IAAA,eAAAA,EAAA,KAAA"}
|
@ -1,2 +0,0 @@
|
||||
System.register(["./Component-legacy.8992014a.js","./useApp-legacy.d06aeb34.js","./index-legacy.1dc0be94.js"],(function(e,n){"use strict";var t,o,i;return{setters:[e=>{t=e.C},e=>{o=e.u},e=>{i=e._}],execute:function(){const n=e=>({show:()=>{e.config.style.display="initial"},hide:()=>{e.config.style.display="none"}}),c=Vue.defineComponent({components:{"magic-ui-component":t},props:{config:{type:Object,default:()=>({})}},setup(e){const t=o(e);return{style:Vue.computed((()=>t?.transformStyle(e.config.style||{}))),display:()=>{const n=e.config?.display;return"function"==typeof n?n(t):!1!==n},...n(e)}}}),s=["id"];e("default",i(c,[["render",function(e,n,t,o,i,c){const l=Vue.resolveComponent("magic-ui-component");return e.display()?(Vue.openBlock(),Vue.createElementBlock("div",{key:0,id:`${e.config.id||""}`,class:Vue.normalizeClass(`magic-ui-container magic-layout-${e.config.layout}${e.config.className?` ${e.config.className}`:""}`),style:Vue.normalizeStyle(e.style)},[Vue.renderSlot(e.$slots,"default"),(Vue.openBlock(!0),Vue.createElementBlock(Vue.Fragment,null,Vue.renderList(e.config.items,(e=>(Vue.openBlock(),Vue.createBlock(l,{key:e.id,config:e},null,8,["config"])))),128))],14,s)):Vue.createCommentVNode("v-if",!0)}],["__file","/parisma/github/tmagic-editor/packages/ui/src/container/src/Container.vue"]]))}}}));
|
||||
//# sourceMappingURL=Container-legacy.3766957a.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"Container-legacy.3766957a.js","sources":["../../../../../../packages/ui/src/useCommonMethod.ts","../../../../../../packages/ui/src/container/src/Container.vue"],"sourcesContent":["/*\n * Tencent is pleased to support the open source community by making TMagicEditor available.\n *\n * Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nexport default (props: any) => ({\n show: () => {\n props.config.style.display = 'initial';\n },\n hide: () => {\n props.config.style.display = 'none';\n },\n});\n","<template>\n <div\n v-if=\"display()\"\n :id=\"`${config.id || ''}`\"\n :class=\"`magic-ui-container magic-layout-${config.layout}${config.className ? ` ${config.className}` : ''}`\"\n :style=\"style\"\n >\n <slot></slot>\n <magic-ui-component v-for=\"item in config.items\" :key=\"item.id\" :config=\"item\"></magic-ui-component>\n </div>\n</template>\n\n<script lang=\"ts\">\nimport { computed, defineComponent, PropType } from 'vue';\n\nimport type { MContainer } from '@tmagic/schema';\n\nimport Component from '../../Component.vue';\nimport useApp from '../../useApp';\nimport useCommonMethod from '../../useCommonMethod';\n\nexport default defineComponent({\n components: {\n 'magic-ui-component': Component,\n },\n\n props: {\n config: {\n type: Object as PropType<MContainer>,\n default: () => ({}),\n },\n },\n\n setup(props) {\n const app = useApp(props);\n\n return {\n style: computed(() => app?.transformStyle(props.config.style || {})),\n\n display: () => {\n const displayCfg = props.config?.display;\n\n if (typeof displayCfg === 'function') {\n return displayCfg(app);\n }\n return displayCfg !== false;\n },\n ...useCommonMethod(props),\n };\n },\n});\n</script>\n"],"names":["props","show","config","style","display","hide","_sfc_main","Vue","defineComponent","components","Component","type","Object","default","setup","app","useApp","computed","transformStyle","displayCfg","useCommonMethod","_component_magic_ui_component","resolveComponent","createElementBlock","key","id","_ctx","class","normalizeClass","layout","className","normalizeStyle","renderSlot","$slots","Fragment","renderList","items","item","openBlock","createBlock"],"mappings":"yNAkBe,QAACA,IAAgB,CAC9BC,KAAM,KACED,EAAAE,OAAOC,MAAMC,QAAU,SAAvB,EAERC,KAAM,KACEL,EAAAE,OAAOC,MAAMC,QAAU,MAAvB,ICFKE,EAAAC,IAAgBC,gBAAA,CAAAC,WAAA,CACjB,qBAAAC,GAEZV,MAAA,CAEOE,OAAA,CACGS,KAAAC,OACAC,QAAA,KAAA,CAAA,KAGVC,MAAAd,GAGE,MAAAe,EAAAC,EAAAhB,GAEA,MAAA,CAAOG,MACEI,IAASU,UAAA,IAAAF,GAAAG,eAAAlB,EAAAE,OAAAC,OAAA,CAAA,KAAmDC,QAAA,KAGjE,MAAAe,EAAAnB,EAAAE,QAAAE,QAEA,MAAA,mBAAAe,EACEA,EAAAJ,IAEF,IAAAI,CAAA,KACFC,EAAApB,GAEF,8DA9Ca,MAAAqB,EAAAd,IAAAe,iBAAA,0DACIf,IAAAgB,mBAAA,MAAA,CAAAC,IAAA,EACXC,GAAA,GAAAC,EAAAxB,OAAAuB,IAAA,KACEE,MAAApB,IAAAqB,eAAA,mCAAAF,EAAAxB,OAAA2B,SAAAH,EAAAxB,OAAA4B,UAAA,IAAAJ,EAAAxB,OAAA4B,YAAA,YAEKvB,IAAAwB,eAAAL,EAAAvB,QAAA,CACbI,IAAAyB,WAAAN,EAAAO,OAAA,YAAA1B,eAA8D,GAAAA,IAAAgB,mBAAAhB,IAAA2B,cAAA3B,IAAA4B,WAAAT,EAAAxB,OAAAkC,OAAAC,QAASC,YAAM/B,IAAAgC,YAAAlB,EAAA,CAAAG,IAAAa,EAAAZ"}
|
@ -1,2 +0,0 @@
|
||||
import{C as c}from"./Component.aeed2f79.js";import{u as a}from"./useApp.8bd6dbd4.js";import{_ as r}from"./index.2a9a64ca.js";const u=e=>({show:()=>{e.config.style.display="initial"},hide:()=>{e.config.style.display="none"}}),l=Vue.defineComponent({components:{"magic-ui-component":c},props:{config:{type:Object,default:()=>({})}},setup(e){const o=a(e);return{style:Vue.computed(()=>o==null?void 0:o.transformStyle(e.config.style||{})),display:()=>{var t;const n=(t=e.config)==null?void 0:t.display;return typeof n=="function"?n(o):n!==!1},...u(e)}}}),m=["id"];function f(e,o,n,t,d,p){const s=Vue.resolveComponent("magic-ui-component");return e.display()?(Vue.openBlock(),Vue.createElementBlock("div",{key:0,id:`${e.config.id||""}`,class:Vue.normalizeClass(`magic-ui-container magic-layout-${e.config.layout}${e.config.className?` ${e.config.className}`:""}`),style:Vue.normalizeStyle(e.style)},[Vue.renderSlot(e.$slots,"default"),(Vue.openBlock(!0),Vue.createElementBlock(Vue.Fragment,null,Vue.renderList(e.config.items,i=>(Vue.openBlock(),Vue.createBlock(s,{key:i.id,config:i},null,8,["config"]))),128))],14,m)):Vue.createCommentVNode("v-if",!0)}const C=r(l,[["render",f],["__file","/parisma/github/tmagic-editor/packages/ui/src/container/src/Container.vue"]]);export{C as default};
|
||||
//# sourceMappingURL=Container.7997a01b.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"Container.7997a01b.js","sources":["../../../../../../packages/ui/src/useCommonMethod.ts","../../../../../../packages/ui/src/container/src/Container.vue"],"sourcesContent":["/*\n * Tencent is pleased to support the open source community by making TMagicEditor available.\n *\n * Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nexport default (props: any) => ({\n show: () => {\n props.config.style.display = 'initial';\n },\n hide: () => {\n props.config.style.display = 'none';\n },\n});\n","<template>\n <div\n v-if=\"display()\"\n :id=\"`${config.id || ''}`\"\n :class=\"`magic-ui-container magic-layout-${config.layout}${config.className ? ` ${config.className}` : ''}`\"\n :style=\"style\"\n >\n <slot></slot>\n <magic-ui-component v-for=\"item in config.items\" :key=\"item.id\" :config=\"item\"></magic-ui-component>\n </div>\n</template>\n\n<script lang=\"ts\">\nimport { computed, defineComponent, PropType } from 'vue';\n\nimport type { MContainer } from '@tmagic/schema';\n\nimport Component from '../../Component.vue';\nimport useApp from '../../useApp';\nimport useCommonMethod from '../../useCommonMethod';\n\nexport default defineComponent({\n components: {\n 'magic-ui-component': Component,\n },\n\n props: {\n config: {\n type: Object as PropType<MContainer>,\n default: () => ({}),\n },\n },\n\n setup(props) {\n const app = useApp(props);\n\n return {\n style: computed(() => app?.transformStyle(props.config.style || {})),\n\n display: () => {\n const displayCfg = props.config?.display;\n\n if (typeof displayCfg === 'function') {\n return displayCfg(app);\n }\n return displayCfg !== false;\n },\n ...useCommonMethod(props),\n };\n },\n});\n</script>\n"],"names":["useCommonMethod","props","_sfc_main","Component","app","useApp","displayCfg","_a","_component_magic_ui_component","_ctx","item"],"mappings":"6HAkBA,MAAeA,EAACC,IAAgB,CAC9B,KAAM,IAAM,CACJA,EAAA,OAAO,MAAM,QAAU,SAC/B,EACA,KAAM,IAAM,CACJA,EAAA,OAAO,MAAM,QAAU,MAC/B,CACF,GCJeC,EAAA,IAAgB,gBAAA,CAAA,WAAA,CACjB,qBAAAC,CACY,EACxB,MAAA,CAEO,OAAA,CACG,KAAA,OACA,QAAA,KAAA,CAAA,EACW,CACnB,EACF,MAAAF,EAAA,CAGE,MAAAG,EAAAC,EAAAJ,CAAA,EAEA,MAAA,CAAO,MACE,IAAS,SAAA,IAAAG,GAAA,YAAAA,EAAA,eAAAH,EAAA,OAAA,OAAA,CAAA,EAAA,EAAmD,QAAA,IAAA,OAGjE,MAAAK,GAAAC,EAAAN,EAAA,SAAA,YAAAM,EAAA,QAEA,OAAA,OAAAD,GAAA,WACEA,EAAAF,CAAA,EAEFE,IAAA,EAAsB,EACxB,GAAAN,EAAAC,CAAA,CACwB,CAC1B,CAEJ,CAAA,mCAhDiB,MAAAO,EAAA,IAAA,iBAAA,oBAAA,sCACI,IAAA,mBAAA,MAAA,CAAA,IAAA,EACX,GAAA,GAAAC,EAAA,OAAA,IAAA,KACE,MAAA,IAAA,eAAA,mCAAAA,EAAA,OAAA,SAAAA,EAAA,OAAA,UAAA,IAAAA,EAAA,OAAA,YAAA,IAAA,QAEK,IAAA,eAAAA,EAAA,KAAA,CAAA,EAAA,CACb,IAAA,WAAAA,EAAA,OAAA,SAAA,GAAA,cAA8D,EAAA,EAAA,IAAA,mBAAA,IAAA,cAAA,IAAA,WAAAA,EAAA,OAAA,MAAAC,IAAS,IAAA,UAAM,EAAA,IAAA,YAAAF,EAAA,CAAA,IAAAE,EAAA"}
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@ -1,2 +0,0 @@
|
||||
System.register(["./Component-legacy.8992014a.js","./useApp-legacy.d06aeb34.js","./index-legacy.1dc0be94.js"],(function(e,n){"use strict";var t,o,c;return{setters:[e=>{t=e.C},e=>{o=e.u},e=>{c=e._}],execute:function(){const n=Vue.defineComponent({components:{"magic-ui-component":t},props:{config:{type:Object,default:()=>({})}},setup(e){const n=o(e);return{style:Vue.computed((()=>n?.transformStyle(e.config.style||{}))),refresh(){window.location.reload()}}}}),i=["id"];e("default",c(n,[["render",function(e,n,t,o,c,u){const a=Vue.resolveComponent("magic-ui-component");return Vue.openBlock(),Vue.createElementBlock("div",{id:`${e.config.id||""}`,class:Vue.normalizeClass(`magic-ui-page magic-ui-container magic-layout-${e.config.layout}${e.config.className?` ${e.config.className}`:""}`),style:Vue.normalizeStyle(e.style)},[Vue.renderSlot(e.$slots,"default"),(Vue.openBlock(!0),Vue.createElementBlock(Vue.Fragment,null,Vue.renderList(e.config.items,(e=>(Vue.openBlock(),Vue.createBlock(a,{key:e.id,config:e},null,8,["config"])))),128))],14,i)}],["__file","/parisma/github/tmagic-editor/packages/ui/src/page/src/index.vue"]]))}}}));
|
||||
//# sourceMappingURL=index-legacy.44438566.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index-legacy.44438566.js","sources":["../../../../../../packages/ui/src/page/src/index.vue"],"sourcesContent":["<template>\n <div\n :id=\"`${config.id || ''}`\"\n :class=\"`magic-ui-page magic-ui-container magic-layout-${config.layout}${\n config.className ? ` ${config.className}` : ''\n }`\"\n :style=\"style\"\n >\n <slot></slot>\n <magic-ui-component v-for=\"item in config.items\" :key=\"item.id\" :config=\"item\"></magic-ui-component>\n </div>\n</template>\n\n<script lang=\"ts\">\nimport { computed, defineComponent, PropType } from 'vue';\n\nimport type { MPage } from '@tmagic/schema';\n\nimport Component from '../../Component.vue';\nimport useApp from '../../useApp';\n\nexport default defineComponent({\n components: {\n 'magic-ui-component': Component,\n },\n\n props: {\n config: {\n type: Object as PropType<MPage>,\n default: () => ({}),\n },\n },\n\n setup(props) {\n const app = useApp(props);\n\n return {\n style: computed(() => app?.transformStyle(props.config.style || {})),\n\n refresh() {\n window.location.reload();\n },\n };\n },\n});\n</script>\n"],"names":["_sfc_main","Vue","defineComponent","components","Component","props","config","type","Object","default","setup","app","useApp","style","computed","transformStyle","refresh","window","location","reload","resolveComponent","openBlock","createElementBlock","id","_ctx","class","normalizeClass","layout","className","normalizeStyle","renderSlot","$slots","Fragment","renderList","items","item","createBlock","_component_magic_ui_component","key"],"mappings":"yNAqBe,MAAAA,EAAAC,IAAgBC,gBAAA,CAAAC,WAAA,CACjB,qBAAAC,GAEZC,MAAA,CAEOC,OAAA,CACGC,KAAAC,OACAC,QAAA,KAAA,CAAA,KAGVC,MAAAL,GAGE,MAAAM,EAAAC,EAAAP,GAEA,MAAA,CAAOQ,MACEZ,IAASa,UAAA,IAAAH,GAAAI,eAAAV,EAAAC,OAAAO,OAAA,CAAA,KAAmDG,UAGjEC,OAAAC,SAAAC,QAAuB,EAE3B,sEAzCFlB,IASMmB,iBAAA,sBAPE,WAAAC,YAAApB,IAAAqB,mBAAA,MAAA,CAAAC,GAAA,GAAAC,EAAAlB,OAAAiB,IAAA,KAA0FE,MAG/FxB,IAAAyB,eAAA,iDAAAF,EAAAlB,OAAAqB,SAAAH,EAAAlB,OAAAsB,UAAA,IAAAJ,EAAAlB,OAAAsB,YAAA,YAEY3B,IAAA4B,eAAAL,EAAAX,QAAA,CACbZ,IAAA6B,WAAAN,EAAAO,OAAA,YAAA9B,eAA8D,GAAAA,IAAAqB,mBAAArB,IAAA+B,cAAA/B,IAAAgC,WAAAT,EAAAlB,OAAA4B,OAAAC,IAASlC,IAAAoB,YAAMpB,IAAAmC,YAAAC,EAAA,CAAAC,IAAAH,EAAAZ"}
|
@ -1,2 +0,0 @@
|
||||
System.register(["./useApp-legacy.d06aeb34.js","./index-legacy.1dc0be94.js"],(function(e,t){"use strict";var i,o;return{setters:[e=>{i=e.u},e=>{o=e._}],execute:function(){const t=Vue.defineComponent({props:{config:{type:Object,default:()=>({})},model:{type:Object,default:()=>({})}},setup(e){const t=Vue.ref(!1),o=i(e),n=o?.page?.getNode(e.config.id),c=()=>{t.value=!0,o&&o.emit("overlay:open",n)},r=()=>{t.value=!1,o&&o.emit("overlay:close",n)};return o?.on("editor:select",((t,i)=>{i.find((t=>t.id===e.config.id))?c():r()})),{visible:t,openOverlay:c,closeOverlay:r}}});e("default",o(t,[["render",function(e,t,i,o,n,c){const r=Vue.resolveComponent("magic-ui-container");return e.visible?(Vue.openBlock(),Vue.createBlock(r,{key:0,class:"magic-ui-overlay",config:{items:e.config.items}},{default:Vue.withCtx((()=>[Vue.renderSlot(e.$slots,"default")])),_:3},8,["config"])):Vue.createCommentVNode("v-if",!0)}],["__file","/parisma/github/tmagic-editor/packages/ui/src/overlay/src/index.vue"]]))}}}));
|
||||
//# sourceMappingURL=index-legacy.5db67b5f.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index-legacy.5db67b5f.js","sources":["../../../../../../packages/ui/src/overlay/src/index.vue"],"sourcesContent":["<template>\n <magic-ui-container v-if=\"visible\" class=\"magic-ui-overlay\" :config=\"{ items: config.items }\">\n <slot></slot>\n </magic-ui-container>\n</template>\n<script lang=\"ts\">\nimport { defineComponent, ref } from 'vue';\n\nimport Core from '@tmagic/core';\nimport type { MNode } from '@tmagic/schema';\n\nimport useApp from '../../useApp';\n\nexport default defineComponent({\n props: {\n config: {\n type: Object,\n default: () => ({}),\n },\n\n model: {\n type: Object,\n default: () => ({}),\n },\n },\n\n setup(props) {\n const visible = ref(false);\n const app: Core | undefined = useApp(props);\n const node = app?.page?.getNode(props.config.id);\n\n const openOverlay = () => {\n visible.value = true;\n if (app) {\n app.emit('overlay:open', node);\n }\n };\n\n const closeOverlay = () => {\n visible.value = false;\n if (app) {\n app.emit('overlay:close', node);\n }\n };\n\n app?.on('editor:select', (info, path) => {\n if (path.find((node: MNode) => node.id === props.config.id)) {\n openOverlay();\n } else {\n closeOverlay();\n }\n });\n\n return {\n visible,\n\n openOverlay,\n closeOverlay,\n };\n },\n});\n</script>\n"],"names":["Vue","defineComponent","props","config","type","Object","default","model","setup","visible","app","useApp","node","page","getNode","id","openOverlay","value","emit","closeOverlay","on","info","path","find","node2","_component_magic_ui_container","resolveComponent","createBlock","key","class","items","_ctx"],"mappings":"2KAae,QAAAA,IAAgBC,gBAAA,CAAAC,MAAA,CACtBC,OAAA,CACGC,KAAAC,OACAC,QAAA,KAAA,CAAA,IAERC,MAAA,CAEOH,KAAAC,OACCC,QAAA,MAAA,KAGVE,MAAAN,GAGE,MAAAO,EAAgBT,YAChBU,EAAAC,EAAAT,GACAU,EAAAF,GAAAG,MAAAC,QAAAZ,EAAAC,OAAAY,IAEAC,EAAA,KACEP,EAAAQ,OAAA,EACAP,GACEA,EAAAQ,KAAA,eAAAN,EAA6B,EAIjCO,EAAA,KACEV,EAAAQ,OAAA,EACAP,GACEA,EAAAQ,KAAA,gBAAAN,EAA8B,EAYlC,OARAF,GAAAU,GAAA,iBAAA,CAAAC,EAAAC,KACEA,EAAAC,MAAAC,GAAAA,EAAAT,KAAAb,EAAAC,OAAAY,KACEC,IAEAG,GAAa,IAIjB,CAAOV,UACLO,cAEAG,eAEF,qDAzD+B,MAAAM,EAAAzB,IAAA0B,iBAAA,wDAA0B1B,IAAA2B,YAAAF,EAAA,CAAAG,IAAA,EAAEC,MAAA,mBAC9C1B,OAAA,CAAA2B,MAAAC,EAAA5B,OAAA2B,QAAA"}
|
@ -1,2 +0,0 @@
|
||||
System.register(["./useApp-legacy.d06aeb34.js","./index-legacy.1dc0be94.js"],(function(e,t){"use strict";var n,o;return{setters:[e=>{n=e.u},e=>{o=e._}],execute:function(){const t=Vue.defineComponent({props:{config:{type:Object,default:()=>({})},model:{type:Object,default:()=>({})}},setup(e){n(e);const t=Vue.getCurrentInstance()?.proxy,o=Vue.reactive([]),c=Vue.computed((()=>["function"==typeof e.config.preAction?e.config.preAction:()=>!0,...o,"function"==typeof e.config.postAction?e.config.postAction:()=>!0]));return{pushAction:function(e){o.push(e)},clickHandler:async function(){for(const n of c.value)if("function"==typeof n&&!1===await n(t,{model:e.model}))break},textConfig:Vue.computed((()=>({type:"text",text:e.config?.text||"",disabledText:e.config?.disabledText||"",html:e.config?.html||""})))}}});e("default",o(t,[["render",function(e,t,n,o,c,i){const u=Vue.resolveComponent("magic-ui-text");return Vue.openBlock(),Vue.createElementBlock("button",{class:"magic-ui-button",onClick:t[0]||(t[0]=(...t)=>e.clickHandler&&e.clickHandler(...t))},[Vue.renderSlot(e.$slots,"default",{},(()=>[Vue.createVNode(u,{config:e.textConfig},null,8,["config"])]))])}],["__file","/parisma/github/tmagic-editor/packages/ui/src/button/src/index.vue"]]))}}}));
|
||||
//# sourceMappingURL=index-legacy.a35443c4.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index-legacy.a35443c4.js","sources":["../../../../../../packages/ui/src/button/src/index.vue"],"sourcesContent":["<template>\n <button class=\"magic-ui-button\" @click=\"clickHandler\">\n <slot>\n <magic-ui-text :config=\"textConfig\"></magic-ui-text>\n </slot>\n </button>\n</template>\n<script lang=\"ts\">\nimport { computed, defineComponent, getCurrentInstance, PropType, reactive } from 'vue';\n\nimport { MButton, MButtonInstance, MText } from '../../../src/types';\nimport useApp from '../../useApp';\n\nexport default defineComponent({\n props: {\n config: {\n type: Object as PropType<MButton>,\n default: () => ({}),\n },\n\n model: {\n type: Object,\n default: () => ({}),\n },\n },\n setup(props) {\n useApp(props);\n const vm: MButtonInstance = getCurrentInstance()?.proxy as MButtonInstance;\n const actions = reactive<Function[]>([]);\n const actualActions = computed(() => [\n typeof props.config.preAction === 'function' ? props.config.preAction : () => true,\n ...actions,\n typeof props.config.postAction === 'function' ? props.config.postAction : () => true,\n ]);\n function pushAction(action: Function): void {\n actions.push(action);\n }\n async function clickHandler(): Promise<void> {\n for (const fn of actualActions.value) {\n if (typeof fn === 'function') {\n const ret = await fn(vm, { model: props.model });\n if (ret === false) {\n break;\n }\n }\n }\n }\n\n const textConfig = computed<MText>(() => ({\n type: 'text',\n text: props.config?.text || '',\n disabledText: props.config?.disabledText || '',\n html: props.config?.html || '',\n }));\n\n return {\n pushAction,\n clickHandler,\n textConfig,\n };\n },\n});\n</script>\n"],"names":["_sfc_main","Vue","defineComponent","props","config","type","Object","default","model","setup","useApp","vm","actions","reactive","actualActions","preAction","postAction","pushAction","action","push","clickHandler","async","value","fn","textConfig","text","disabledText","html","resolveComponent","openBlock","createElementBlock","class"],"mappings":"2KAae,MAAAA,EAAAC,IAAgBC,gBAAA,CAAAC,MAAA,CACtBC,OAAA,CACGC,KAAAC,OACAC,QAAA,MAAA,IAERC,MAAA,CAEOH,KAAAC,OACCC,QAAA,MAAA,KAGVE,MAAAN,GAEEO,EAAAP,GACA,MAAAQ,EAA4BV,gCAC5BW,EAAgBX,IAAqBY,SAAA,IACrCC,EAAsBb,mBAAe,mBAAAE,EAAAC,OAAAW,UAAAZ,EAAAC,OAAAW,UAAA,KAAA,KAC2CH,EAC3E,mBAAAT,EAAAC,OAAAY,WAAAb,EAAAC,OAAAY,WAAA,KAAA,KAwBL,MAAA,CAAOC,WArBP,SAAAC,GACEN,EAAAO,KAAAD,EAAmB,EAqBnBE,aAnBFC,iBACE,IAAA,WAAAP,EAAAQ,MACE,GAAA,mBAAAC,IAEE,UADAA,EAAAZ,EAAA,CAAAH,MAAAL,EAAAK,QAEE,KAGN,EAYAgB,WATiBvB,oBAAuBI,KAAA,OAClCoB,KAAAtB,EAAAC,QAAAqB,MAAA,GACsBC,aAAAvB,EAAAC,QAAAsB,cAAA,GACgBC,KAAAxB,EAAAC,QAAAuB,MAAA,OAQ9C,6DA1DF1B,IAIS2B,iBAAA,iBAJwB,OAAA3B,IAAA4B,YAAO5B,IAAA6B,mBAAA,SAAA,CAAAC,MAAA,qFAG/B"}
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@ -1,2 +0,0 @@
|
||||
System.register(["./useApp-legacy.d06aeb34.js","./index-legacy.1dc0be94.js"],(function(e,t){"use strict";var i,s;return{setters:[e=>{i=e.u},e=>{s=e._}],execute:function(){const t=Vue.defineComponent({props:{config:{type:Object,default:()=>({})},model:{type:Object,default:()=>({})},vars:{type:Object,default:()=>({})}},setup(e){i(e);const t=Vue.getCurrentInstance()?.proxy,s=Vue.inject("hoc");return{displayText:Vue.computed((()=>{let i=e.config?.text||"";const{vars:c}=e;if(s?.disabled&&e.config?.disabledText&&(i=e.config.disabledText),"function"==typeof i)return i.bind(t)(t,{model:e.model});if("[object Object]"===Object.prototype.toString.call(c)){let e=i;return Object.entries(c).forEach((([t,i])=>{e=e.replace(new RegExp(`{{${t}}}`,"g"),i)})),e}return i||""}))}},render(){const e=this.config?.multiple?"magic-ui-text":"magic-ui-text magic-ui-text--single-line";return"function"==typeof this.$slots?.default?Vue.h("div",{class:e},[this.$slots?.default?.()||""]):Vue.h("div",{class:e,...this.displayText?{innerHTML:this.displayText}:{}})}});e("default",s(t,[["__file","/parisma/github/tmagic-editor/packages/ui/src/text/src/index.vue"]]))}}}));
|
||||
//# sourceMappingURL=index-legacy.f6a03169.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index-legacy.f6a03169.js","sources":["../../../../../../packages/ui/src/text/src/index.vue"],"sourcesContent":["<script lang=\"ts\">\nimport { computed, defineComponent, getCurrentInstance, h, inject, PropType } from 'vue';\n\nimport { MComponentInstance, MText, MTextInstance } from '../../../src/types';\nimport useApp from '../../useApp';\n\nexport default defineComponent({\n props: {\n config: {\n type: Object as PropType<MText>,\n default: () => ({}),\n },\n\n model: {\n type: Object,\n default: () => ({}),\n },\n\n vars: {\n type: Object,\n default: () => ({}),\n },\n },\n setup(props) {\n useApp(props);\n const vm: MTextInstance = getCurrentInstance()?.proxy as MTextInstance;\n const hoc: MComponentInstance = inject('hoc');\n const displayText = computed(() => {\n let text = props.config?.text || '';\n const { vars } = props;\n if (hoc?.disabled && props.config?.disabledText) {\n text = props.config.disabledText;\n }\n if (typeof text === 'function') {\n return text.bind(vm)(vm, { model: props.model });\n }\n if (Object.prototype.toString.call(vars) === '[object Object]') {\n let tmp: string = text;\n Object.entries(vars).forEach(([key, value]) => {\n tmp = tmp.replace(new RegExp(`{{${key}}}`, 'g'), value);\n });\n return tmp;\n }\n return text || '';\n });\n\n return {\n displayText,\n };\n },\n\n render() {\n const className = this.config?.multiple ? 'magic-ui-text' : 'magic-ui-text magic-ui-text--single-line';\n if (typeof this.$slots?.default === 'function') {\n return h('div', { class: className }, [this.$slots?.default?.() || '']);\n }\n return h('div', {\n class: className,\n ...(this.displayText ? { innerHTML: this.displayText } : {}),\n });\n },\n});\n</script>\n"],"names":["_sfc_main","Vue","defineComponent","props","config","type","Object","default","model","vars","setup","useApp","vm","hoc","displayText","text","disabled","disabledText","bind","prototype","toString","call","tmp","entries","forEach","key","value","replace","RegExp","render","className","this","multiple","$slots","h","class","innerHTML"],"mappings":"2KAMe,MAAAA,EAAAC,IAAgBC,gBAAA,CAAAC,MAAA,CACtBC,OAAA,CACGC,KAAAC,OACAC,QAAA,KAAA,CAAA,IAERC,MAAA,CAEOH,KAAAC,OACCC,QAAA,KAAA,CAAA,IAERE,KAAA,CAEMJ,KAAAC,OACEC,QAAA,KAAA,CAAA,KAGVG,MAAAP,GAEEQ,EAAAR,GACA,MAAAS,EAA0BX,gCAC1BY,EAAgCZ,kBAoBhC,MAAA,CAAOa,YAnBab,mBAClB,IAAAc,EAAAZ,EAAAC,QAAAW,MAAA,GACA,MAAAN,KAAAA,GAAAN,EAIA,GAHAU,GAAAG,UAAAb,EAAAC,QAAAa,eACEF,EAAAZ,EAAAC,OAAAa,cAEF,qBACE,OAAAF,EAAAG,KAAAN,EAAAG,CAAAH,EAAA,CAAAJ,MAAAL,EAAAK,QAEF,GAAA,oBAAAF,OAAAa,UAAAC,SAAAC,KAAAZ,GAAA,CACE,IAAAa,EAAAP,EAIA,OAHAT,OAAAiB,QAAAd,GAAAe,SAAA,EAAAC,EAAAC,MACEJ,EAAAA,EAAAK,QAAA,IAAAC,OAAA,KAAAH,MAAA,KAAAC,EAAA,IAEFJ,CAAO,CAET,OAAAP,GAAA,EAAA,IAKF,EACFc,SAGE,MAAAC,EAAAC,KAAA3B,QAAA4B,SAAA,gBAAA,2CACA,MAAA,mBAAAC,KAAAA,QAAA1B,QACWN,IAAAiC,EAAA,MAAA,CAAAC,MAAAL,GAAA,CAAAC,KAAAE,QAAA1B,aAAA,kBAEK4B,MAAAL,KACPC,KAAAjB,YAAA,CAAAsB,UAAAL,KAAAjB,aAAA,CAAA,GAER"}
|
@ -1,2 +0,0 @@
|
||||
System.register(["./useApp-legacy.d06aeb34.js","./index-legacy.1dc0be94.js"],(function(e,c){"use strict";var i,n;return{setters:[e=>{i=e.u},e=>{n=e._}],execute:function(){const c=Vue.defineComponent({props:{config:{type:Object,default:()=>({})},model:{type:Object,default:()=>({})}},setup:e=>(i(e),{clickHandler(){e.config.url&&(window.location.href=e.config.url)}})}),t=["src"];e("default",n(c,[["render",function(e,c,i,n,r,l){return Vue.openBlock(),Vue.createElementBlock("img",{class:"magic-ui-img",src:e.config.src,onClick:c[0]||(c[0]=(...c)=>e.clickHandler&&e.clickHandler(...c))},null,8,t)}],["__file","/parisma/github/tmagic-editor/packages/ui/src/img/src/index.vue"]]))}}}));
|
||||
//# sourceMappingURL=index-legacy.fac272b8.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index-legacy.fac272b8.js","sources":["../../../../../../packages/ui/src/img/src/index.vue"],"sourcesContent":["<template>\n <img class=\"magic-ui-img\" :src=\"config.src\" @click=\"clickHandler\" />\n</template>\n<script lang=\"ts\">\nimport { defineComponent, PropType } from 'vue';\n\nimport { MImg } from '../../types';\nimport useApp from '../../useApp';\n\nexport default defineComponent({\n props: {\n config: {\n type: Object as PropType<MImg>,\n default: () => ({}),\n },\n\n model: {\n type: Object,\n default: () => ({}),\n },\n },\n setup(props) {\n useApp(props);\n\n return {\n clickHandler() {\n if (props.config.url) window.location.href = props.config.url;\n },\n };\n },\n});\n</script>\n"],"names":["_sfc_main","Vue","defineComponent","props","config","type","Object","default","model","setup","useApp","clickHandler","url","window","location","href","_ctx","_cache","$props","$setup","$data","$options","openBlock","createElementBlock","class","src"],"mappings":"2KASe,MAAAA,EAAAC,IAAgBC,gBAAA,CAAAC,MAAA,CACtBC,OAAA,CACGC,KAAAC,OACAC,QAAA,KAAA,CAAA,IAERC,MAAA,CAEOH,KAAAC,OACCC,QAAA,KAAA,CAAA,KAGVE,MAAAN,IAEEO,EAAAP,GAEA,CAAOQ,eAEHR,EAAAC,OAAAQ,MAAsBC,OAAAC,SAAAC,KAAAZ,EAAAC,OAAAQ,IAAoC,2CAzBtD,SAAAI,EAAAC,EAAAC,EAAAC,EAAAC,EAAAC,GAAsB,OAAApB,IAAAqB,YAAUrB,IAAAsB,mBAAA,MAAA,CAAAC,MAAA,eAAQC,IAAAT,EAAAZ,OAAAqB"}
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@ -1,2 +0,0 @@
|
||||
import{u as f}from"./useApp.8bd6dbd4.js";import{_ as m}from"./index.2a9a64ca.js";const p=Vue.defineComponent({props:{config:{type:Object,default:()=>({})},model:{type:Object,default:()=>({})},vars:{type:Object,default:()=>({})}},setup(e){var s;f(e);const i=(s=Vue.getCurrentInstance())==null?void 0:s.proxy,t=Vue.inject("hoc");return{displayText:Vue.computed(()=>{var u,l;let c=((u=e.config)==null?void 0:u.text)||"";const{vars:o}=e;if((t==null?void 0:t.disabled)&&((l=e.config)==null?void 0:l.disabledText)&&(c=e.config.disabledText),typeof c=="function")return c.bind(i)(i,{model:e.model});if(Object.prototype.toString.call(o)==="[object Object]"){let a=c;return Object.entries(o).forEach(([r,d])=>{a=a.replace(new RegExp(`{{${r}}}`,"g"),d)}),a}return c||""})}},render(){var i,t,n,s;const e=(i=this.config)!=null&&i.multiple?"magic-ui-text":"magic-ui-text magic-ui-text--single-line";return typeof((t=this.$slots)==null?void 0:t.default)=="function"?Vue.h("div",{class:e},[((s=(n=this.$slots)==null?void 0:n.default)==null?void 0:s.call(n))||""]):Vue.h("div",{class:e,...this.displayText?{innerHTML:this.displayText}:{}})}}),b=m(p,[["__file","/parisma/github/tmagic-editor/packages/ui/src/text/src/index.vue"]]);export{b as default};
|
||||
//# sourceMappingURL=index.55a4739f.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index.55a4739f.js","sources":["../../../../../../packages/ui/src/text/src/index.vue"],"sourcesContent":["<script lang=\"ts\">\nimport { computed, defineComponent, getCurrentInstance, h, inject, PropType } from 'vue';\n\nimport { MComponentInstance, MText, MTextInstance } from '../../../src/types';\nimport useApp from '../../useApp';\n\nexport default defineComponent({\n props: {\n config: {\n type: Object as PropType<MText>,\n default: () => ({}),\n },\n\n model: {\n type: Object,\n default: () => ({}),\n },\n\n vars: {\n type: Object,\n default: () => ({}),\n },\n },\n setup(props) {\n useApp(props);\n const vm: MTextInstance = getCurrentInstance()?.proxy as MTextInstance;\n const hoc: MComponentInstance = inject('hoc');\n const displayText = computed(() => {\n let text = props.config?.text || '';\n const { vars } = props;\n if (hoc?.disabled && props.config?.disabledText) {\n text = props.config.disabledText;\n }\n if (typeof text === 'function') {\n return text.bind(vm)(vm, { model: props.model });\n }\n if (Object.prototype.toString.call(vars) === '[object Object]') {\n let tmp: string = text;\n Object.entries(vars).forEach(([key, value]) => {\n tmp = tmp.replace(new RegExp(`{{${key}}}`, 'g'), value);\n });\n return tmp;\n }\n return text || '';\n });\n\n return {\n displayText,\n };\n },\n\n render() {\n const className = this.config?.multiple ? 'magic-ui-text' : 'magic-ui-text magic-ui-text--single-line';\n if (typeof this.$slots?.default === 'function') {\n return h('div', { class: className }, [this.$slots?.default?.() || '']);\n }\n return h('div', {\n class: className,\n ...(this.displayText ? { innerHTML: this.displayText } : {}),\n });\n },\n});\n</script>\n"],"names":["_sfc_main","props","useApp","vm","_a","hoc","text","vars","_b","tmp","key","value","className","_d","_c"],"mappings":"iFAMA,MAAeA,EAAA,IAAgB,gBAAA,CAAA,MAAA,CACtB,OAAA,CACG,KAAA,OACA,QAAA,KAAA,CAAA,EACW,EACnB,MAAA,CAEO,KAAA,OACC,QAAA,KAAA,CAAA,EACW,EACnB,KAAA,CAEM,KAAA,OACE,QAAA,KAAA,CAAA,EACW,CACnB,EACF,MAAAC,EAAA,OAEEC,EAAAD,CAAA,EACA,MAAAE,GAA0BC,EAAA,2BAAA,YAAAA,QAC1BC,EAAgC,kBAoBhC,MAAA,CAAO,YAnBa,0BAClB,IAAAC,IAAAF,EAAAH,EAAA,SAAA,YAAAG,EAAA,OAAA,GACA,KAAA,CAAA,KAAAG,CAAA,EAAAN,EAIA,IAHAI,GAAA,YAAAA,EAAA,aAAAG,EAAAP,EAAA,SAAA,YAAAO,EAAA,gBACEF,EAAAL,EAAA,OAAA,cAEF,OAAAK,GAAA,WACE,OAAAA,EAAA,KAAAH,CAAA,EAAAA,EAAA,CAAA,MAAAF,EAAA,KAAA,CAAA,EAEF,GAAA,OAAA,UAAA,SAAA,KAAAM,CAAA,IAAA,kBAAA,CACE,IAAAE,EAAAH,EACA,cAAA,QAAAC,CAAA,EAAA,QAAA,CAAA,CAAAG,EAAAC,CAAA,IAAA,CACEF,EAAAA,EAAA,QAAA,IAAA,OAAA,KAAAC,MAAA,GAAA,EAAAC,CAAA,CAAsD,CAAA,EAExDF,CAAO,CAET,OAAAH,GAAA,EAAe,CAAA,CAIf,CACF,EACF,QAAA,aAGE,MAAAM,GAAAR,EAAA,KAAA,SAAA,MAAAA,EAAA,SAAA,gBAAA,2CACA,OAAA,QAAAI,EAAA,KAAA,SAAA,YAAAA,EAAA,UAAA,WACS,IAAE,EAAA,MAAA,CAAA,MAAAI,CAAA,EAAA,GAAAC,GAAAC,EAAA,KAAA,SAAA,YAAAA,EAAA,UAAA,YAAAD,EAAA,KAAAC,KAAA,EAAA,CAAA,EAEJ,aAAS,MAAAF,EACP,GAAA,KAAA,YAAA,CAAA,UAAA,KAAA,WAAA,EAAA,CAAA,CACmD,CAAA,CAC3D,CAEL,CAAA"}
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@ -1,2 +0,0 @@
|
||||
import{u as l}from"./useApp.8bd6dbd4.js";import{_ as d}from"./index.2a9a64ca.js";const m=Vue.defineComponent({props:{config:{type:Object,default:()=>({})},model:{type:Object,default:()=>({})}},setup(t){var o;l(t);const n=(o=Vue.getCurrentInstance())==null?void 0:o.proxy,c=Vue.reactive([]),f=Vue.computed(()=>[typeof t.config.preAction=="function"?t.config.preAction:()=>!0,...c,typeof t.config.postAction=="function"?t.config.postAction:()=>!0]);function a(e){c.push(e)}async function s(){for(const e of f.value)if(typeof e=="function"&&await e(n,{model:t.model})===!1)break}const i=Vue.computed(()=>{var e,u,r;return{type:"text",text:((e=t.config)==null?void 0:e.text)||"",disabledText:((u=t.config)==null?void 0:u.disabledText)||"",html:((r=t.config)==null?void 0:r.html)||""}});return{pushAction:a,clickHandler:s,textConfig:i}}});function g(t,n,c,f,a,s){const i=Vue.resolveComponent("magic-ui-text");return Vue.openBlock(),Vue.createElementBlock("button",{class:"magic-ui-button",onClick:n[0]||(n[0]=(...o)=>t.clickHandler&&t.clickHandler(...o))},[Vue.renderSlot(t.$slots,"default",{},()=>[Vue.createVNode(i,{config:t.textConfig},null,8,["config"])])])}const V=d(m,[["render",g],["__file","/parisma/github/tmagic-editor/packages/ui/src/button/src/index.vue"]]);export{V as default};
|
||||
//# sourceMappingURL=index.d9593460.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index.d9593460.js","sources":["../../../../../../packages/ui/src/button/src/index.vue"],"sourcesContent":["<template>\n <button class=\"magic-ui-button\" @click=\"clickHandler\">\n <slot>\n <magic-ui-text :config=\"textConfig\"></magic-ui-text>\n </slot>\n </button>\n</template>\n<script lang=\"ts\">\nimport { computed, defineComponent, getCurrentInstance, PropType, reactive } from 'vue';\n\nimport { MButton, MButtonInstance, MText } from '../../../src/types';\nimport useApp from '../../useApp';\n\nexport default defineComponent({\n props: {\n config: {\n type: Object as PropType<MButton>,\n default: () => ({}),\n },\n\n model: {\n type: Object,\n default: () => ({}),\n },\n },\n setup(props) {\n useApp(props);\n const vm: MButtonInstance = getCurrentInstance()?.proxy as MButtonInstance;\n const actions = reactive<Function[]>([]);\n const actualActions = computed(() => [\n typeof props.config.preAction === 'function' ? props.config.preAction : () => true,\n ...actions,\n typeof props.config.postAction === 'function' ? props.config.postAction : () => true,\n ]);\n function pushAction(action: Function): void {\n actions.push(action);\n }\n async function clickHandler(): Promise<void> {\n for (const fn of actualActions.value) {\n if (typeof fn === 'function') {\n const ret = await fn(vm, { model: props.model });\n if (ret === false) {\n break;\n }\n }\n }\n }\n\n const textConfig = computed<MText>(() => ({\n type: 'text',\n text: props.config?.text || '',\n disabledText: props.config?.disabledText || '',\n html: props.config?.html || '',\n }));\n\n return {\n pushAction,\n clickHandler,\n textConfig,\n };\n },\n});\n</script>\n"],"names":["_sfc_main","props","useApp","vm","_a","actions","actualActions","pushAction","action","clickHandler","fn","textConfig","_b","_c"],"mappings":"iFAaA,MAAeA,EAAA,IAAgB,gBAAA,CAAA,MAAA,CACtB,OAAA,CACG,KAAA,OACA,QAAA,KAAA,CAAA,EACW,EACnB,MAAA,CAEO,KAAA,OACC,QAAA,KAAA,CAAA,EACW,CACnB,EACF,MAAAC,EAAA,OAEEC,EAAAD,CAAA,EACA,MAAAE,GAA4BC,EAAA,2BAAA,YAAAA,QAC5BC,EAAgB,IAAqB,SAAA,CAAA,CAAA,EACrCC,EAAsB,kBAAe,OAAAL,EAAA,OAAA,WAAA,WAAAA,EAAA,OAAA,UAAA,IAAA,GAC2C,GAAAI,EAC3E,OAAAJ,EAAA,OAAA,YAAA,WAAAA,EAAA,OAAA,WAAA,IAAA,EAC6E,CAAA,EAElF,SAAAM,EAAAC,EAAA,CACEH,EAAA,KAAAG,CAAA,CAAmB,CAErB,eAAAC,GAAA,CACE,UAAAC,KAAAJ,EAAA,MACE,GAAA,OAAAI,GAAA,YACE,MAAAA,EAAAP,EAAA,CAAA,MAAAF,EAAA,KAAA,CAAA,IACA,GACE,KAGN,CAGF,MAAAU,EAAmB,mCAAuB,KAAA,OAClC,OAAAP,EAAAH,EAAA,SAAA,YAAAG,EAAA,OAAA,GACsB,eAAAQ,EAAAX,EAAA,SAAA,YAAAW,EAAA,eAAA,GACgB,OAAAC,EAAAZ,EAAA,SAAA,YAAAY,EAAA,OAAA,EAChB,EAAA,EAG9B,MAAA,CAAO,WAAAN,EACL,aAAAE,EACA,WAAAE,CACA,CACF,CAEJ,CAAA,kCA5DE,IAIS,iBAAA,eAAA,EAJwB,OAAA,IAAA,UAAA,EAAO,IAAA,mBAAA,SAAA,CAAA,MAAA,mFAG/B,EAAA"}
|
@ -1,2 +0,0 @@
|
||||
import{C as i}from"./Component.aeed2f79.js";import{u as c}from"./useApp.8bd6dbd4.js";import{_ as s}from"./index.2a9a64ca.js";const a=Vue.defineComponent({components:{"magic-ui-component":i},props:{config:{type:Object,default:()=>({})}},setup(e){const o=c(e);return{style:Vue.computed(()=>o==null?void 0:o.transformStyle(e.config.style||{})),refresh(){window.location.reload()}}}}),r=["id"];function u(e,o,l,m,p,f){const t=Vue.resolveComponent("magic-ui-component");return Vue.openBlock(),Vue.createElementBlock("div",{id:`${e.config.id||""}`,class:Vue.normalizeClass(`magic-ui-page magic-ui-container magic-layout-${e.config.layout}${e.config.className?` ${e.config.className}`:""}`),style:Vue.normalizeStyle(e.style)},[Vue.renderSlot(e.$slots,"default"),(Vue.openBlock(!0),Vue.createElementBlock(Vue.Fragment,null,Vue.renderList(e.config.items,n=>(Vue.openBlock(),Vue.createBlock(t,{key:n.id,config:n},null,8,["config"]))),128))],14,r)}const y=s(a,[["render",u],["__file","/parisma/github/tmagic-editor/packages/ui/src/page/src/index.vue"]]);export{y as default};
|
||||
//# sourceMappingURL=index.ee5b9415.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index.ee5b9415.js","sources":["../../../../../../packages/ui/src/page/src/index.vue"],"sourcesContent":["<template>\n <div\n :id=\"`${config.id || ''}`\"\n :class=\"`magic-ui-page magic-ui-container magic-layout-${config.layout}${\n config.className ? ` ${config.className}` : ''\n }`\"\n :style=\"style\"\n >\n <slot></slot>\n <magic-ui-component v-for=\"item in config.items\" :key=\"item.id\" :config=\"item\"></magic-ui-component>\n </div>\n</template>\n\n<script lang=\"ts\">\nimport { computed, defineComponent, PropType } from 'vue';\n\nimport type { MPage } from '@tmagic/schema';\n\nimport Component from '../../Component.vue';\nimport useApp from '../../useApp';\n\nexport default defineComponent({\n components: {\n 'magic-ui-component': Component,\n },\n\n props: {\n config: {\n type: Object as PropType<MPage>,\n default: () => ({}),\n },\n },\n\n setup(props) {\n const app = useApp(props);\n\n return {\n style: computed(() => app?.transformStyle(props.config.style || {})),\n\n refresh() {\n window.location.reload();\n },\n };\n },\n});\n</script>\n"],"names":["_sfc_main","Component","props","app","useApp","_ctx","item","_component_magic_ui_component"],"mappings":"6HAqBA,MAAeA,EAAA,IAAgB,gBAAA,CAAA,WAAA,CACjB,qBAAAC,CACY,EACxB,MAAA,CAEO,OAAA,CACG,KAAA,OACA,QAAA,KAAA,CAAA,EACW,CACnB,EACF,MAAAC,EAAA,CAGE,MAAAC,EAAAC,EAAAF,CAAA,EAEA,MAAA,CAAO,MACE,IAAS,SAAA,IAAAC,GAAA,YAAAA,EAAA,eAAAD,EAAA,OAAA,OAAA,CAAA,EAAA,EAAmD,SAAA,CAGjE,OAAA,SAAA,QAAuB,CACzB,CACF,CAEJ,CAAA,2CA3CE,IASM,iBAAA,oBAAA,EAPE,OAAA,IAAA,UAAA,EAAA,IAAA,mBAAA,MAAA,CAAA,GAAA,GAAAG,EAAA,OAAA,IAAA,KAA0F,MAG/F,IAAA,eAAA,iDAAAA,EAAA,OAAA,SAAAA,EAAA,OAAA,UAAA,IAAAA,EAAA,OAAA,YAAA,IAAA,QAEY,IAAA,eAAAA,EAAA,KAAA,CAAA,EAAA,CACb,IAAA,WAAAA,EAAA,OAAA,SAAA,GAAA,cAA8D,EAAA,EAAA,IAAA,mBAAA,IAAA,cAAA,IAAA,WAAAA,EAAA,OAAA,MAAAC,IAAS,IAAA,UAAM,EAAA,IAAA,YAAAC,EAAA,CAAA,IAAAD,EAAA"}
|
@ -1,2 +0,0 @@
|
||||
import{u as n}from"./useApp.8bd6dbd4.js";import{_ as o}from"./index.2a9a64ca.js";const r=Vue.defineComponent({props:{config:{type:Object,default:()=>({})},model:{type:Object,default:()=>({})}},setup(e){return n(e),{clickHandler(){e.config.url&&(window.location.href=e.config.url)}}}}),t=["src"];function s(e,i,l,a,u,f){return Vue.openBlock(),Vue.createElementBlock("img",{class:"magic-ui-img",src:e.config.src,onClick:i[0]||(i[0]=(...c)=>e.clickHandler&&e.clickHandler(...c))},null,8,t)}const m=o(r,[["render",s],["__file","/parisma/github/tmagic-editor/packages/ui/src/img/src/index.vue"]]);export{m as default};
|
||||
//# sourceMappingURL=index.fb366993.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index.fb366993.js","sources":["../../../../../../packages/ui/src/img/src/index.vue"],"sourcesContent":["<template>\n <img class=\"magic-ui-img\" :src=\"config.src\" @click=\"clickHandler\" />\n</template>\n<script lang=\"ts\">\nimport { defineComponent, PropType } from 'vue';\n\nimport { MImg } from '../../types';\nimport useApp from '../../useApp';\n\nexport default defineComponent({\n props: {\n config: {\n type: Object as PropType<MImg>,\n default: () => ({}),\n },\n\n model: {\n type: Object,\n default: () => ({}),\n },\n },\n setup(props) {\n useApp(props);\n\n return {\n clickHandler() {\n if (props.config.url) window.location.href = props.config.url;\n },\n };\n },\n});\n</script>\n"],"names":["_sfc_main","props","useApp","_sfc_render","_ctx","_cache","$props","$setup","$data","$options"],"mappings":"iFASA,MAAeA,EAAA,IAAgB,gBAAA,CAAA,MAAA,CACtB,OAAA,CACG,KAAA,OACA,QAAA,KAAA,CAAA,EACW,EACnB,MAAA,CAEO,KAAA,OACC,QAAA,KAAA,CAAA,EACW,CACnB,EACF,MAAAC,EAAA,CAEE,OAAAC,EAAAD,CAAA,EAEA,CAAO,cAAA,CAEHA,EAAA,OAAA,MAAsB,OAAA,SAAA,KAAAA,EAAA,OAAA,IAAoC,CAC5D,CACF,CAEJ,CAAA,YA7BY,SAAAE,EAAAC,EAAAC,EAAAC,EAAAC,EAAAC,EAAAC,EAAA,CAAsB,OAAA,IAAA,UAAA,EAAU,IAAA,mBAAA,MAAA,CAAA,MAAA,eAAQ,IAAAL,EAAA,OAAA"}
|
@ -1,2 +0,0 @@
|
||||
import{u}from"./useApp.8bd6dbd4.js";import{_ as l}from"./index.2a9a64ca.js";const f=Vue.defineComponent({props:{config:{type:Object,default:()=>({})},model:{type:Object,default:()=>({})}},setup(o){var i;const t=Vue.ref(!1),e=u(o),n=(i=e==null?void 0:e.page)==null?void 0:i.getNode(o.config.id),c=()=>{t.value=!0,e&&e.emit("overlay:open",n)},s=()=>{t.value=!1,e&&e.emit("overlay:close",n)};return e==null||e.on("editor:select",(m,r)=>{r.find(a=>a.id===o.config.id)?c():s()}),{visible:t,openOverlay:c,closeOverlay:s}}});function d(o,t,e,n,c,s){const i=Vue.resolveComponent("magic-ui-container");return o.visible?(Vue.openBlock(),Vue.createBlock(i,{key:0,class:"magic-ui-overlay",config:{items:o.config.items}},{default:Vue.withCtx(()=>[Vue.renderSlot(o.$slots,"default")]),_:3},8,["config"])):Vue.createCommentVNode("v-if",!0)}const v=l(f,[["render",d],["__file","/parisma/github/tmagic-editor/packages/ui/src/overlay/src/index.vue"]]);export{v as default};
|
||||
//# sourceMappingURL=index.fed7c0dd.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"index.fed7c0dd.js","sources":["../../../../../../packages/ui/src/overlay/src/index.vue"],"sourcesContent":["<template>\n <magic-ui-container v-if=\"visible\" class=\"magic-ui-overlay\" :config=\"{ items: config.items }\">\n <slot></slot>\n </magic-ui-container>\n</template>\n<script lang=\"ts\">\nimport { defineComponent, ref } from 'vue';\n\nimport Core from '@tmagic/core';\nimport type { MNode } from '@tmagic/schema';\n\nimport useApp from '../../useApp';\n\nexport default defineComponent({\n props: {\n config: {\n type: Object,\n default: () => ({}),\n },\n\n model: {\n type: Object,\n default: () => ({}),\n },\n },\n\n setup(props) {\n const visible = ref(false);\n const app: Core | undefined = useApp(props);\n const node = app?.page?.getNode(props.config.id);\n\n const openOverlay = () => {\n visible.value = true;\n if (app) {\n app.emit('overlay:open', node);\n }\n };\n\n const closeOverlay = () => {\n visible.value = false;\n if (app) {\n app.emit('overlay:close', node);\n }\n };\n\n app?.on('editor:select', (info, path) => {\n if (path.find((node: MNode) => node.id === props.config.id)) {\n openOverlay();\n } else {\n closeOverlay();\n }\n });\n\n return {\n visible,\n\n openOverlay,\n closeOverlay,\n };\n },\n});\n</script>\n"],"names":["_sfc_main","props","visible","app","useApp","node","_a","openOverlay","closeOverlay","info","path","node2","_component_magic_ui_container","_ctx"],"mappings":"4EAaA,MAAeA,EAAA,IAAgB,gBAAA,CAAA,MAAA,CACtB,OAAA,CACG,KAAA,OACA,QAAA,KAAA,CAAA,EACW,EACnB,MAAA,CAEO,KAAA,OACC,QAAA,KAAA,CAAA,EACW,CACnB,EACF,MAAAC,EAAA,OAGE,MAAAC,EAAgB,YAChBC,EAAAC,EAAAH,CAAA,EACAI,GAAAC,EAAAH,GAAA,YAAAA,EAAA,OAAA,YAAAG,EAAA,QAAAL,EAAA,OAAA,IAEAM,EAAA,IAAA,CACEL,EAAA,MAAA,GACAC,GACEA,EAAA,KAAA,eAAAE,CAAA,CACF,EAGFG,EAAA,IAAA,CACEN,EAAA,MAAA,GACAC,GACEA,EAAA,KAAA,gBAAAE,CAAA,CACF,EAGF,OAAAF,GAAA,MAAAA,EAAA,GAAA,gBAAA,CAAAM,EAAAC,IAAA,CACEA,EAAA,KAAAC,GAAAA,EAAA,KAAAV,EAAA,OAAA,EAAA,EACEM,IAEAC,GACF,GAGF,CAAO,QAAAN,EACL,YAAAK,EAEA,aAAAC,CACA,CACF,CAEJ,CAAA,0BA3DmC,MAAAI,EAAA,IAAA,iBAAA,oBAAA,oCAA0B,IAAA,YAAAA,EAAA,CAAA,IAAA,EAAE,MAAA,mBAC9C,OAAA,CAAA,MAAAC,EAAA,OAAA,KAAA,CAAA,EAAA"}
|
File diff suppressed because one or more lines are too long
@ -1,2 +0,0 @@
|
||||
System.register([],(function(e,t){"use strict";return{execute:function(){e("u",(e=>{const t=Vue.inject("app"),n=t?.page?.getNode(e.config.id),u=Vue.getCurrentInstance()?.proxy;return n?.emit("created",u),Vue.onMounted((()=>{n?.emit("mounted",u)})),Vue.onUnmounted((()=>{n?.emit("destroy",u)})),t}))}}}));
|
||||
//# sourceMappingURL=useApp-legacy.d06aeb34.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"useApp-legacy.d06aeb34.js","sources":["../../../../../../packages/ui/src/useApp.ts"],"sourcesContent":["/*\n * Tencent is pleased to support the open source community by making TMagicEditor available.\n *\n * Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { getCurrentInstance, inject, onMounted, onUnmounted } from 'vue';\n\nimport Core from '@tmagic/core';\n\nexport default (props: any) => {\n const app: Core | undefined = inject('app');\n const node = app?.page?.getNode(props.config.id);\n\n const vm = getCurrentInstance()?.proxy;\n\n node?.emit('created', vm);\n\n onMounted(() => {\n node?.emit('mounted', vm);\n });\n\n onUnmounted(() => {\n node?.emit('destroy', vm);\n });\n\n return app;\n};\n"],"names":["exports","props","app","Vue","node","page","getNode","config","id","vm","emit","onMounted","onUnmounted"],"mappings":"yEAsBAA,EAAA,KAAAC,IACE,MAAAC,EAA8BC,kBAC9BC,EAAAF,GAAAG,MAAAC,QAAAL,EAAAM,OAAAC,IAEAC,EAAWN,gCAYX,OAVAC,GAAAM,KAAA,UAAAD,GAEAN,IAAAQ,WAAA,KACEP,GAAAM,KAAA,UAAAD,EAAA,IAGFN,IAAAS,aAAA,KACER,GAAAM,KAAA,UAAAD,MAGFP,CAAA"}
|
@ -1,2 +0,0 @@
|
||||
const r=c=>{var u,o;const e=Vue.inject("app"),t=(u=e==null?void 0:e.page)==null?void 0:u.getNode(c.config.id),n=(o=Vue.getCurrentInstance())==null?void 0:o.proxy;return t==null||t.emit("created",n),Vue.onMounted(()=>{t==null||t.emit("mounted",n)}),Vue.onUnmounted(()=>{t==null||t.emit("destroy",n)}),e};export{r as u};
|
||||
//# sourceMappingURL=useApp.8bd6dbd4.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"useApp.8bd6dbd4.js","sources":["../../../../../../packages/ui/src/useApp.ts"],"sourcesContent":["/*\n * Tencent is pleased to support the open source community by making TMagicEditor available.\n *\n * Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { getCurrentInstance, inject, onMounted, onUnmounted } from 'vue';\n\nimport Core from '@tmagic/core';\n\nexport default (props: any) => {\n const app: Core | undefined = inject('app');\n const node = app?.page?.getNode(props.config.id);\n\n const vm = getCurrentInstance()?.proxy;\n\n node?.emit('created', vm);\n\n onMounted(() => {\n node?.emit('mounted', vm);\n });\n\n onUnmounted(() => {\n node?.emit('destroy', vm);\n });\n\n return app;\n};\n"],"names":["useApp","props","_a","_b","app","node","vm"],"mappings":"AAsBA,MAAAA,EAAAC,GAAA,CAAA,IAAAC,EAAAC,EACE,MAAAC,EAA8B,kBAC9BC,GAAAH,EAAAE,GAAA,YAAAA,EAAA,OAAA,YAAAF,EAAA,QAAAD,EAAA,OAAA,IAEAK,GAAWH,EAAA,2BAAA,YAAAA,QAEX,OAAAE,GAAA,MAAAA,EAAA,KAAA,UAAAC,GAEA,IAAA,UAAA,IAAA,CACED,GAAA,MAAAA,EAAA,KAAA,UAAAC,EAAwB,CAAA,EAG1B,IAAA,YAAA,IAAA,CACED,GAAA,MAAAA,EAAA,KAAA,UAAAC,EAAwB,CAAA,EAG1BF,CACF"}
|
Binary file not shown.
Before Width: | Height: | Size: 8.6 KiB |
@ -1,38 +0,0 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<link rel="icon" href="/static/vue3/runtime/page/favicon.png" type="image/png">
|
||||
<title>Vue3 Page</title>
|
||||
<style>
|
||||
html,
|
||||
body,
|
||||
#app {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
}
|
||||
|
||||
#app {
|
||||
position: relative;
|
||||
overflow: auto;
|
||||
}
|
||||
|
||||
#app::-webkit-scrollbar {
|
||||
width: 0 !important;
|
||||
display: none;
|
||||
}
|
||||
</style>
|
||||
<script src="https://unpkg.com/vue@next/dist/vue.runtime.global.js"></script>
|
||||
<script type="module" crossorigin src="/static/vue3/runtime/page/assets/index.2a9a64ca.js"></script>
|
||||
<script type="module">try{import.meta.url;import("_").catch(()=>1);}catch(e){}window.__vite_is_modern_browser=true;</script>
|
||||
<script type="module">!function(){if(window.__vite_is_modern_browser)return;console.warn("vite: loading legacy build because dynamic import or import.meta.url is unsupported, syntax error above should be ignored");var e=document.getElementById("vite-legacy-polyfill"),n=document.createElement("script");n.src=e.src,n.onload=function(){System.import(document.getElementById('vite-legacy-entry').getAttribute('data-src'))},document.body.appendChild(n)}();</script>
|
||||
</head>
|
||||
<body>
|
||||
<div id="app"></div>
|
||||
|
||||
<script nomodule>!function(){var e=document,t=e.createElement("script");if(!("noModule"in t)&&"onbeforeload"in t){var n=!1;e.addEventListener("beforeload",(function(e){if(e.target===t)n=!0;else if(!e.target.hasAttribute("nomodule")||!n)return;e.preventDefault()}),!0),t.type="module",t.src=".",e.head.appendChild(t),t.remove()}}();</script>
|
||||
<script nomodule crossorigin id="vite-legacy-polyfill" src="/static/vue3/runtime/page/assets/polyfills-legacy.a62c3647.js"></script>
|
||||
<script nomodule crossorigin id="vite-legacy-entry" data-src="/static/vue3/runtime/page/assets/index-legacy.1dc0be94.js">System.import(document.getElementById('vite-legacy-entry').getAttribute('data-src'))</script>
|
||||
</body>
|
||||
</html>
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@ -1,2 +0,0 @@
|
||||
System.register([],(function(e,t){"use strict";return{execute:function(){e("default",{})}}}));
|
||||
//# sourceMappingURL=plugin-entry-legacy.7d9da6d6.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"plugin-entry-legacy.7d9da6d6.js","sources":["../../../../.tmagic/plugin-entry.ts"],"sourcesContent":["const plugins: Record<string, any> = {};\nexport default plugins;"],"names":["exports"],"mappings":"yEAAqCA,EAAA,UAAA,CAAA"}
|
@ -1,2 +0,0 @@
|
||||
const s={};export{s as default};
|
||||
//# sourceMappingURL=plugin-entry.b765d2c2.js.map
|
@ -1 +0,0 @@
|
||||
{"version":3,"file":"plugin-entry.b765d2c2.js","sources":["../../../../.tmagic/plugin-entry.ts"],"sourcesContent":["const plugins: Record<string, any> = {};\nexport default plugins;"],"names":["plugins"],"mappings":"AAAA,MAAMA,EAA+B,CAAA"}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user