mirror of
https://github.com/Tencent/tmagic-editor.git
synced 2025-04-06 03:57:56 +08:00
feat(reate-runtime-help,vue-runtime-help): 新增组件状态hook
This commit is contained in:
parent
63fe6ec68b
commit
6f5bb84c04
@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"version": "0.0.4",
|
"version": "0.1.0",
|
||||||
"name": "@tmagic/react-runtime-help",
|
"name": "@tmagic/react-runtime-help",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"sideEffects": false,
|
"sideEffects": false,
|
||||||
|
@ -19,45 +19,44 @@
|
|||||||
import { useContext, useEffect, useState } from 'react';
|
import { useContext, useEffect, useState } from 'react';
|
||||||
|
|
||||||
import type TMagicApp from '@tmagic/core';
|
import type TMagicApp from '@tmagic/core';
|
||||||
import type { Id, MNodeInstance } from '@tmagic/core';
|
import type { Id, MNodeInstance, Node as TMagicNode } from '@tmagic/core';
|
||||||
import { isDslNode } from '@tmagic/core';
|
import { isDslNode } from '@tmagic/core';
|
||||||
|
|
||||||
import AppContent from '../AppContent';
|
import AppContent from '../AppContent';
|
||||||
|
|
||||||
interface UseAppOptions<T extends MNodeInstance = MNodeInstance> {
|
interface Methods {
|
||||||
|
[key: string]: (...args: any[]) => any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface UseAppOptions<T extends MNodeInstance = MNodeInstance> {
|
||||||
config: T;
|
config: T;
|
||||||
iteratorContainerId?: Id[];
|
iteratorContainerId?: Id[];
|
||||||
iteratorIndex?: number[];
|
iteratorIndex?: number[];
|
||||||
methods?: {
|
methods?: {
|
||||||
[key: string]: Function;
|
[key: string]: (...args: any[]) => any;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export const useApp = ({ methods = {}, config, iteratorContainerId, iteratorIndex }: UseAppOptions) => {
|
export const useNode = (
|
||||||
const app: TMagicApp | undefined = useContext(AppContent);
|
props: Pick<UseAppOptions, 'config' | 'iteratorContainerId' | 'iteratorIndex'>,
|
||||||
|
app = useContext(AppContent),
|
||||||
|
) =>
|
||||||
|
isDslNode(props.config) && props.config.id
|
||||||
|
? app?.getNode(props.config.id, props.iteratorContainerId, props.iteratorIndex)
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
export const registerNodeHooks = (node?: TMagicNode, methods: Methods = {}) => {
|
||||||
|
if (!node) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const emitData = {
|
const emitData = {
|
||||||
config,
|
config: node.data,
|
||||||
...methods,
|
...methods,
|
||||||
};
|
};
|
||||||
|
|
||||||
const display = <T extends MNodeInstance>(config: T) => {
|
|
||||||
if (config.visible === false) return false;
|
|
||||||
if (config.condResult === false) return false;
|
|
||||||
|
|
||||||
const displayCfg = config.display;
|
|
||||||
|
|
||||||
if (typeof displayCfg === 'function') {
|
|
||||||
return displayCfg(app);
|
|
||||||
}
|
|
||||||
|
|
||||||
return displayCfg !== false;
|
|
||||||
};
|
|
||||||
|
|
||||||
const node = isDslNode(config) && config.id ? app?.getNode(config.id, iteratorContainerId, iteratorIndex) : undefined;
|
|
||||||
const [created, setCreated] = useState(false);
|
const [created, setCreated] = useState(false);
|
||||||
|
|
||||||
if (node) {
|
|
||||||
if (!created) {
|
if (!created) {
|
||||||
// 只需要触发一次 created
|
// 只需要触发一次 created
|
||||||
setCreated(true);
|
setCreated(true);
|
||||||
@ -71,7 +70,36 @@ export const useApp = ({ methods = {}, config, iteratorContainerId, iteratorInde
|
|||||||
node?.emit('destroy', emitData);
|
node?.emit('destroy', emitData);
|
||||||
};
|
};
|
||||||
}, []);
|
}, []);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const useApp = ({ methods = {}, config, iteratorContainerId, iteratorIndex }: UseAppOptions) => {
|
||||||
|
const app: TMagicApp | undefined = useContext(AppContent);
|
||||||
|
|
||||||
|
const node = useNode(
|
||||||
|
{
|
||||||
|
config,
|
||||||
|
iteratorContainerId,
|
||||||
|
iteratorIndex,
|
||||||
|
},
|
||||||
|
app,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (node) {
|
||||||
|
registerNodeHooks(node, methods);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const display = <T extends MNodeInstance>(config: T) => {
|
||||||
|
if (config.visible === false) return false;
|
||||||
|
if (config.condResult === false) return false;
|
||||||
|
|
||||||
|
const displayCfg = config.display;
|
||||||
|
|
||||||
|
if (typeof displayCfg === 'function') {
|
||||||
|
return displayCfg({ app, node });
|
||||||
|
}
|
||||||
|
|
||||||
|
return displayCfg !== false;
|
||||||
|
};
|
||||||
|
|
||||||
return { app, node, display };
|
return { app, node, display };
|
||||||
};
|
};
|
||||||
|
108
runtime/react-runtime-help/src/hooks/use-component-status.ts
Normal file
108
runtime/react-runtime-help/src/hooks/use-component-status.ts
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
import { useContext, useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
import type TMagicApp from '@tmagic/core';
|
||||||
|
import { type MComponent, type StyleSchema, toLine } from '@tmagic/core';
|
||||||
|
|
||||||
|
import AppContent from '../AppContent';
|
||||||
|
|
||||||
|
export interface StatusData {
|
||||||
|
style?: StyleSchema;
|
||||||
|
className?: string;
|
||||||
|
[key: string]: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useComponentStatus = (props: { config: Omit<MComponent, 'id'> }) => {
|
||||||
|
const app: TMagicApp | undefined = useContext(AppContent);
|
||||||
|
|
||||||
|
const [status, setStatus] = useState('default');
|
||||||
|
const [style, setStyle] = useState({});
|
||||||
|
const [className, setClassName] = useState('');
|
||||||
|
const styleStatusMap = new Map<string, StyleSchema>();
|
||||||
|
const classStatusMap = new Map<string, string>();
|
||||||
|
const statusMap = new Map<string, Omit<StatusData, 'style' | 'className'>>();
|
||||||
|
|
||||||
|
const registerStatus = (type: string, { style, className, ...data }: StatusData) => {
|
||||||
|
if (style) {
|
||||||
|
styleStatusMap.set(type, style);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (className) {
|
||||||
|
classStatusMap.set(type, className);
|
||||||
|
}
|
||||||
|
|
||||||
|
statusMap.set(type, data);
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
registerStatus('default', {
|
||||||
|
style: props.config.style,
|
||||||
|
className: props.config.className,
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
styleStatusMap.clear();
|
||||||
|
classStatusMap.clear();
|
||||||
|
statusMap.clear();
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const type = status || 'default';
|
||||||
|
const defaultStyle = styleStatusMap.get('default') || {};
|
||||||
|
const statusStyle = styleStatusMap.get(type);
|
||||||
|
|
||||||
|
let style = app?.transformStyle(defaultStyle) || {};
|
||||||
|
|
||||||
|
if (type !== 'default' && statusStyle) {
|
||||||
|
style = Object.keys(statusStyle).reduce((obj, key) => {
|
||||||
|
const value = statusStyle[key];
|
||||||
|
if (value === null || typeof value === 'undefined' || isNaN(value) || value === '') {
|
||||||
|
return {
|
||||||
|
...obj,
|
||||||
|
[key]: statusStyle[key],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return { ...obj };
|
||||||
|
}, style);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (props.config.displayHidden) {
|
||||||
|
style.display = 'none';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof props.config.condResult !== 'undefined' && props.config.displayRenderModel === 'mount') {
|
||||||
|
if (props.config.condResult === false) {
|
||||||
|
style.display = 'none';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setStyle(style);
|
||||||
|
|
||||||
|
const className = classStatusMap.get(type) ?? '';
|
||||||
|
|
||||||
|
const list = [];
|
||||||
|
|
||||||
|
if (props.config.type) {
|
||||||
|
list.push(`magic-ui-${toLine(props.config.type)}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (props.config.layout) {
|
||||||
|
list.push(`magic-layout-${props.config.layout}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (className) {
|
||||||
|
list.push(className);
|
||||||
|
}
|
||||||
|
|
||||||
|
setClassName(list.join(' '));
|
||||||
|
}, [status, props.config]);
|
||||||
|
|
||||||
|
return {
|
||||||
|
status,
|
||||||
|
style,
|
||||||
|
className,
|
||||||
|
|
||||||
|
setStatus,
|
||||||
|
registerStatus,
|
||||||
|
};
|
||||||
|
};
|
@ -19,3 +19,4 @@ export { default as AppContent } from './AppContent';
|
|||||||
export * from './hooks/use-editor-dsl';
|
export * from './hooks/use-editor-dsl';
|
||||||
export * from './hooks/use-dsl';
|
export * from './hooks/use-dsl';
|
||||||
export * from './hooks/use-app';
|
export * from './hooks/use-app';
|
||||||
|
export * from './hooks/use-component-status';
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"version": "1.0.1",
|
"version": "1.1.0",
|
||||||
"name": "@tmagic/vue-runtime-help",
|
"name": "@tmagic/vue-runtime-help",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"sideEffects": false,
|
"sideEffects": false,
|
||||||
|
@ -19,42 +19,38 @@
|
|||||||
import { inject, onBeforeUnmount, onMounted } from 'vue-demi';
|
import { inject, onBeforeUnmount, onMounted } from 'vue-demi';
|
||||||
|
|
||||||
import type TMagicApp from '@tmagic/core';
|
import type TMagicApp from '@tmagic/core';
|
||||||
import type { Id, MNodeInstance } from '@tmagic/core';
|
import type { Id, MNodeInstance, Node as TMagicNode } from '@tmagic/core';
|
||||||
import { isDslNode } from '@tmagic/core';
|
import { isDslNode } from '@tmagic/core';
|
||||||
|
|
||||||
|
interface Methods {
|
||||||
|
[key: string]: (...args: any[]) => any;
|
||||||
|
}
|
||||||
|
|
||||||
interface UseAppOptions<T extends MNodeInstance = MNodeInstance> {
|
interface UseAppOptions<T extends MNodeInstance = MNodeInstance> {
|
||||||
config: T;
|
config: T;
|
||||||
iteratorContainerId?: Id[];
|
iteratorContainerId?: Id[];
|
||||||
iteratorIndex?: number[];
|
iteratorIndex?: number[];
|
||||||
methods?: {
|
methods?: Methods;
|
||||||
[key: string]: Function;
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const useApp = ({ methods = {}, config, iteratorContainerId, iteratorIndex }: UseAppOptions) => {
|
export const useNode = (
|
||||||
const app = inject<TMagicApp>('app');
|
props: Pick<UseAppOptions, 'config' | 'iteratorContainerId' | 'iteratorIndex'>,
|
||||||
|
app = inject<TMagicApp>('app'),
|
||||||
|
) =>
|
||||||
|
isDslNode(props.config) && props.config.id
|
||||||
|
? app?.getNode(props.config.id, props.iteratorContainerId, props.iteratorIndex)
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
export const registerNodeHooks = (node?: TMagicNode, methods: Methods = {}) => {
|
||||||
|
if (!node) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const emitData = {
|
const emitData = {
|
||||||
config,
|
config: node.data,
|
||||||
...methods,
|
...methods,
|
||||||
};
|
};
|
||||||
|
|
||||||
const display = <T extends MNodeInstance>(config: T) => {
|
|
||||||
if (config.visible === false) return false;
|
|
||||||
if (config.condResult === false) return false;
|
|
||||||
|
|
||||||
const displayCfg = config.display;
|
|
||||||
|
|
||||||
if (typeof displayCfg === 'function') {
|
|
||||||
return displayCfg(app);
|
|
||||||
}
|
|
||||||
|
|
||||||
return displayCfg !== false;
|
|
||||||
};
|
|
||||||
|
|
||||||
const node = isDslNode(config) && config.id ? app?.getNode(config.id, iteratorContainerId, iteratorIndex) : undefined;
|
|
||||||
|
|
||||||
if (node) {
|
|
||||||
node.emit('created', emitData);
|
node.emit('created', emitData);
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
@ -64,12 +60,31 @@ export const useApp = ({ methods = {}, config, iteratorContainerId, iteratorInde
|
|||||||
onBeforeUnmount(() => {
|
onBeforeUnmount(() => {
|
||||||
node.emit('destroy', emitData);
|
node.emit('destroy', emitData);
|
||||||
});
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const useApp = <T extends TMagicApp = TMagicApp>({
|
||||||
|
methods = {},
|
||||||
|
config,
|
||||||
|
iteratorContainerId,
|
||||||
|
iteratorIndex,
|
||||||
|
}: UseAppOptions) => {
|
||||||
|
const app = inject<T>('app');
|
||||||
|
|
||||||
|
const node = useNode(
|
||||||
|
{
|
||||||
|
config,
|
||||||
|
iteratorContainerId,
|
||||||
|
iteratorIndex,
|
||||||
|
},
|
||||||
|
app,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (node) {
|
||||||
|
registerNodeHooks(node, methods);
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
app,
|
app,
|
||||||
node,
|
node,
|
||||||
|
|
||||||
display,
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
108
runtime/vue-runtime-help/src/hooks/use-component-status.ts
Normal file
108
runtime/vue-runtime-help/src/hooks/use-component-status.ts
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
import { computed, inject, onScopeDispose, ref, shallowReactive, watchEffect } from 'vue-demi';
|
||||||
|
|
||||||
|
import type TMagicCore from '@tmagic/core';
|
||||||
|
import { type MComponent, type StyleSchema, toLine } from '@tmagic/core';
|
||||||
|
|
||||||
|
export interface StatusData {
|
||||||
|
style?: StyleSchema;
|
||||||
|
className?: string;
|
||||||
|
[key: string]: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useComponentStatus = (props: { config: Omit<MComponent, 'id'> }) => {
|
||||||
|
const app = inject<TMagicCore>('app');
|
||||||
|
|
||||||
|
const status = ref('default');
|
||||||
|
const styleStatusMap = new Map<string, StyleSchema>();
|
||||||
|
const classStatusMap = new Map<string, string>();
|
||||||
|
const statusMap = new Map<string, Omit<StatusData, 'style' | 'className'>>();
|
||||||
|
|
||||||
|
const setStatus = (value: string) => {
|
||||||
|
status.value = value;
|
||||||
|
};
|
||||||
|
|
||||||
|
const registerStatus = (type: string, { style, className, ...data }: StatusData) => {
|
||||||
|
if (style) {
|
||||||
|
styleStatusMap.set(type, style);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (className) {
|
||||||
|
classStatusMap.set(type, className);
|
||||||
|
}
|
||||||
|
|
||||||
|
statusMap.set(type, shallowReactive(data));
|
||||||
|
};
|
||||||
|
|
||||||
|
watchEffect(() => {
|
||||||
|
registerStatus('default', {
|
||||||
|
style: props.config.style,
|
||||||
|
className: props.config.className,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
onScopeDispose(() => {
|
||||||
|
statusMap.clear();
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
status: computed(() => status.value),
|
||||||
|
|
||||||
|
style: computed(() => {
|
||||||
|
const type = status.value || 'default';
|
||||||
|
const defaultStyle = styleStatusMap.get('default') || {};
|
||||||
|
const statusStyle = styleStatusMap.get(type);
|
||||||
|
|
||||||
|
let style = app?.transformStyle(defaultStyle) || {};
|
||||||
|
|
||||||
|
if (type !== 'default' && statusStyle) {
|
||||||
|
style = Object.keys(statusStyle).reduce((obj, key) => {
|
||||||
|
const value = statusStyle[key];
|
||||||
|
if (value === null || typeof value === 'undefined' || isNaN(value) || value === '') {
|
||||||
|
return {
|
||||||
|
...obj,
|
||||||
|
[key]: statusStyle[key],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return { ...obj };
|
||||||
|
}, style);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (props.config.displayHidden) {
|
||||||
|
style.display = 'none';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof props.config.condResult !== 'undefined' && props.config.displayRenderModel === 'mount') {
|
||||||
|
if (props.config.condResult === false) {
|
||||||
|
style.display = 'none';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return style;
|
||||||
|
}),
|
||||||
|
|
||||||
|
className: computed(() => {
|
||||||
|
const type = status.value || 'default';
|
||||||
|
const className = classStatusMap.get(type) ?? '';
|
||||||
|
|
||||||
|
const list = [];
|
||||||
|
|
||||||
|
if (props.config.type) {
|
||||||
|
list.push(`magic-ui-${toLine(props.config.type)}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (props.config.layout) {
|
||||||
|
list.push(`magic-layout-${props.config.layout}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (className) {
|
||||||
|
list.push(className);
|
||||||
|
}
|
||||||
|
|
||||||
|
return list.join(' ');
|
||||||
|
}),
|
||||||
|
|
||||||
|
setStatus,
|
||||||
|
|
||||||
|
registerStatus,
|
||||||
|
};
|
||||||
|
};
|
@ -1,20 +1,29 @@
|
|||||||
import { h } from 'vue-demi';
|
import { h } from 'vue-demi';
|
||||||
|
|
||||||
import type { MComponent } from '@tmagic/core';
|
import type { MComponent, StyleSchema } from '@tmagic/core';
|
||||||
|
|
||||||
export * from './hooks/use-editor-dsl';
|
export * from './hooks/use-editor-dsl';
|
||||||
export * from './hooks/use-dsl';
|
export * from './hooks/use-dsl';
|
||||||
export * from './hooks/use-app';
|
export * from './hooks/use-app';
|
||||||
|
export * from './hooks/use-component-status';
|
||||||
export { useComponent } from './hooks/use-component';
|
export { useComponent } from './hooks/use-component';
|
||||||
|
|
||||||
export interface userRenderFunctionOptions {
|
export interface userRenderFunctionOptions {
|
||||||
h: typeof h;
|
h: typeof h;
|
||||||
type: Parameters<typeof h>[0];
|
type: Parameters<typeof h>[0];
|
||||||
props: any;
|
props?: {
|
||||||
attrs: any;
|
[key: string]: any;
|
||||||
className: string | string[];
|
};
|
||||||
style: any;
|
attrs?: {
|
||||||
config: MComponent;
|
[key: string]: any;
|
||||||
|
};
|
||||||
|
className?: string | string[];
|
||||||
|
style?: StyleSchema;
|
||||||
|
config: Omit<MComponent, 'id'>;
|
||||||
|
on?: {
|
||||||
|
[key: string]: (...args: any[]) => any;
|
||||||
|
};
|
||||||
|
directives?: { name: string; value: any; modifiers: any }[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export type UserRenderFunction = (options: userRenderFunctionOptions) => any;
|
export type UserRenderFunction = (options: userRenderFunctionOptions) => any;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user