chore: 将magic-admin迁移出

https://github.com/vft-magic/tmagic-admin
This commit is contained in:
roymondchen 2023-03-22 15:47:46 +08:00
parent b9248498eb
commit cb99304993
171 changed files with 4 additions and 69035 deletions

View File

@ -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。

View File

@ -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
│   ├── apiweb 端接口文件)
│   ├── 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数据库实例初始化文件
│   ├── serviceservice 文件)
│   ├── 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

View File

@ -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": ""
}

View File

@ -1,10 +0,0 @@
{
"presets": [
[
"@babel/preset-env"
],
[
"@babel/preset-typescript"
]
]
}

View File

@ -1,3 +0,0 @@
dist
node_modules
pm2.config.js

View File

@ -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$'],
],
},
],
},
};

View File

@ -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

View File

@ -1,7 +0,0 @@
export default {
collectCoverage: true,
coverageProvider: 'v8',
moduleNameMapper: {
'^@src/(.*)$': '<rootDir>/src/$1',
},
};

File diff suppressed because it is too large Load Diff

View File

@ -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"
}
}

View File

@ -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,
},
],
},
};

View File

@ -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>';

View File

@ -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',
};

View File

@ -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',
};

View File

@ -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();

View File

@ -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();

View File

@ -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();

View File

@ -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表';

View File

@ -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}`);

View File

@ -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;
}

View File

@ -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];

View File

@ -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;
}

View File

@ -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;

View File

@ -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;

View File

@ -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;

View File

@ -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;

View File

@ -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;

View File

@ -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;
}
}

View File

@ -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);
};

View File

@ -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;
}

View File

@ -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);
/**
* codesrcCode和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('保存新添加的页面失败');
}
};
}

View File

@ -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);
});
};

View File

@ -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',
},
],
},
];

View File

@ -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: '会员登录认证',
},
],
};

View File

@ -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;
}

View File

@ -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);
},
};

View File

@ -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 };

View File

@ -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();

View File

@ -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 编译失败');
}
}

View File

@ -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

View File

@ -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

View File

@ -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;;;;;;;;"}

View File

@ -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

View File

@ -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

View File

@ -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"}

View File

@ -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

View File

@ -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"}

View File

@ -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

View File

@ -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"}

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -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"}

View File

@ -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

View File

@ -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"}

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -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"}

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -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"}

View File

@ -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

View File

@ -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"}

View File

@ -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

View File

@ -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"}

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -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"}

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -1,2 +0,0 @@
System.register([],(function(e,t){"use strict";return{execute:function(){e("default",{})}}}));
//# sourceMappingURL=plugin-entry-legacy.7d9da6d6.js.map

View File

@ -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"}

View File

@ -1,2 +0,0 @@
const s={};export{s as default};
//# sourceMappingURL=plugin-entry.b765d2c2.js.map

View File

@ -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