feat(vue-component, runtime): 使用 app.resolveComponent 获取组件 (#631)

This commit is contained in:
洩氏诹诹子 2024-09-03 21:07:55 +08:00 committed by roymondchen
parent 6e71448158
commit c3bc1035ad
21 changed files with 184 additions and 31 deletions

3
pnpm-lock.yaml generated
View File

@ -1286,6 +1286,9 @@ importers:
'@types/node':
specifier: ^18.19.0
version: 18.19.42
'@vue/test-utils':
specifier: ^2.4.6
version: 2.4.6
rimraf:
specifier: ^3.0.2
version: 3.0.2

View File

@ -57,6 +57,7 @@
},
"devDependencies": {
"@types/node": "^18.19.0",
"@vue/test-utils": "^2.4.6",
"rimraf": "^3.0.2"
}
}

View File

@ -0,0 +1,67 @@
/*
* Tencent is pleased to support the open source community by making TMagicEditor 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 { inject } from 'vue-demi';
import type Core from '@tmagic/core';
import { toLine } from '@tmagic/utils';
interface UseComponentOptions {
/** 组件类型 */
componentType?: string;
/** App 实例 */
app?: Core;
}
/**
* App
* @param options
* @returns magic-ui-
*/
export function useComponent<Component = any>(options: string | UseComponentOptions = '') {
let componentType: string | undefined;
let app: Core | undefined;
let component: Component | undefined;
if (typeof options === 'string') {
componentType = options;
} else {
({ componentType, app } = options);
}
if (!componentType || componentType === '') {
componentType = 'container';
}
if (!app) {
app = inject('app');
}
component = resolveComponent({ componentType, app });
if (!component && !componentType.startsWith('magic-ui-')) {
componentType = `magic-ui-${toLine(componentType)}`;
component = resolveComponent({ componentType, app });
}
return component ?? componentType;
}
type resolveComponentOptions = Required<Pick<UseComponentOptions, 'componentType'>> & UseComponentOptions;
function resolveComponent<Component = any>({ componentType, app }: resolveComponentOptions): Component | undefined {
return app?.resolveComponent(componentType);
}

View File

@ -1,3 +1,4 @@
export * from './hooks/use-editor-dsl';
export * from './hooks/use-dsl';
export * from './hooks/use-app';
export { useComponent } from './hooks/use-component';

View File

@ -0,0 +1,55 @@
import { describe, expect, test } from 'vitest';
import { defineComponent, isVue3, provide } from 'vue-demi';
import { mount } from '@vue/test-utils';
import Core from '@tmagic/core';
import { useComponent } from '../src';
describe('useComponent', () => {
const app = new Core({});
const fooComponent = 'foo-component';
app.registerComponent('foo', fooComponent);
const containerComponent = {};
app.registerComponent('magic-ui-container', containerComponent);
test('string para', () => {
const component = useComponent('foo');
expect(component).toEqual('magic-ui-foo');
});
test('object para and can find component', () => {
const component = useComponent({ componentType: 'foo', app });
expect(component).toEqual(fooComponent);
});
test('without app and can not find component', () => {
const component = useComponent({ componentType: 'foo' });
expect(component).toEqual('magic-ui-foo');
});
test('with magic-ui- componentType and can not find component', () => {
const component = useComponent({ componentType: 'magic-ui-foo', app });
expect(component).toEqual('magic-ui-foo');
});
test.runIf(isVue3)('auto inject and empty para', () => {
const child = defineComponent({
setup() {
const component = useComponent();
expect(component).toEqual(containerComponent);
},
});
const parent = defineComponent({
template: '<child-com></child-com>',
components: { 'child-com': child },
setup() {
provide('app', app);
},
});
const vueApp = mount(parent);
vueApp.unmount();
});
});

View File

@ -37,6 +37,7 @@ export default defineConfig({
{ find: /^@tmagic\/data-source/, replacement: path.join(__dirname, '../../packages/data-source/src/index.ts') },
{ find: /^@tmagic\/dep/, replacement: path.join(__dirname, '../../packages/dep/src/index.ts') },
{ find: /^@tmagic\/schema/, replacement: path.join(__dirname, '../../packages/schema/src/index.ts') },
{ find: /^@tmagic\/vue-runtime-help/, replacement: path.join(__dirname, '../vue-runtime-help/src/index.ts') },
],
},

View File

@ -1,5 +1,5 @@
<template>
<magic-ui-page :config="pageConfig"></magic-ui-page>
<component :is="pageComponent" :config="pageConfig"></component>
</template>
<script lang="ts">
@ -8,7 +8,7 @@ import { defineComponent, inject } from 'vue';
import type { Page } from '@tmagic/core';
import type Core from '@tmagic/core';
import { addParamToUrl } from '@tmagic/utils';
import { useDsl } from '@tmagic/vue-runtime-help';
import { useComponent, useDsl } from '@tmagic/vue-runtime-help';
export default defineComponent({
name: 'App',
@ -16,6 +16,7 @@ export default defineComponent({
setup() {
const app = inject<Core | undefined>('app');
const { pageConfig } = useDsl(app);
const pageComponent = useComponent('page');
app?.on('page-change', (page?: Page) => {
if (!page) {
@ -25,6 +26,7 @@ export default defineComponent({
});
return {
pageComponent,
pageConfig,
};
},

View File

@ -48,8 +48,8 @@ const app = new Core({
app.setDesignWidth(app.env.isWeb ? window.document.documentElement.getBoundingClientRect().width : 375);
Object.keys(components).forEach((type: string) => {
Vue.component(`magic-ui-${type}`, components[type]);
Object.entries(components).forEach(([type, component]: [string, any]) => {
app.registerComponent(type, component);
});
Object.values(plugins).forEach((plugin: any) => {

View File

@ -1,19 +1,21 @@
<template>
<magic-ui-page v-if="pageConfig" :key="pageConfig.id" :config="pageConfig"></magic-ui-page>
<component v-if="pageConfig" :is="pageComponent" :key="pageConfig.id" :config="pageConfig"></component>
</template>
<script lang="ts">
import { defineComponent, inject } from 'vue';
import type Core from '@tmagic/core';
import { useEditorDsl } from '@tmagic/vue-runtime-help';
import { useComponent, useEditorDsl } from '@tmagic/vue-runtime-help';
export default defineComponent({
setup() {
const app = inject<Core | undefined>('app');
const { pageConfig } = useEditorDsl(app);
const pageComponent = useComponent('page');
return {
pageComponent,
pageConfig,
};
},

View File

@ -42,7 +42,7 @@ Promise.all([
}
Object.entries(components.default).forEach(([type, component]: [string, any]) => {
Vue.component(`magic-ui-${type}`, component);
app.registerComponent(type, component);
});
Object.entries(dataSources).forEach(([type, ds]: [string, any]) => {

View File

@ -1,17 +1,19 @@
<template>
<magic-ui-page :config="pageConfig"></magic-ui-page>
<component :is="pageComponent" :config="(pageConfig as MPage)"></component>
</template>
<script lang="ts" setup>
import { inject } from 'vue';
import { MPage } from 'packages/schema/types';
import type { Page } from '@tmagic/core';
import type Core from '@tmagic/core';
import { addParamToUrl } from '@tmagic/utils';
import { useDsl } from '@tmagic/vue-runtime-help';
import { useComponent, useDsl } from '@tmagic/vue-runtime-help';
const app = inject<Core | undefined>('app');
const { pageConfig } = useDsl(app);
const pageComponent = useComponent('page');
app?.on('page-change', (page?: Page) => {
if (!page) {

View File

@ -51,7 +51,7 @@ const app = new Core({
app.setDesignWidth(app.env.isWeb ? window.document.documentElement.getBoundingClientRect().width : 375);
Object.entries(components).forEach(([type, component]: [string, any]) => {
vueApp.component(`magic-ui-${type}`, defineAsyncComponent(component));
app.registerComponent(type, defineAsyncComponent(component));
});
Object.values(plugins).forEach((plugin: any) => {

View File

@ -1,13 +1,14 @@
<template>
<magic-ui-page v-if="pageConfig" :key="pageConfig.id" :config="pageConfig"></magic-ui-page>
<component v-if="pageConfig" :is="pageComponent" :key="pageConfig.id" :config="pageConfig"></component>
</template>
<script lang="ts" setup>
import { inject } from 'vue';
import type Core from '@tmagic/core';
import { useEditorDsl } from '@tmagic/vue-runtime-help';
import { useComponent, useEditorDsl } from '@tmagic/vue-runtime-help';
const app = inject<Core | undefined>('app');
const { pageConfig } = useEditorDsl(app);
const pageComponent = useComponent('page');
</script>

View File

@ -44,7 +44,7 @@ Promise.all([
}
Object.entries(components.default).forEach(([type, component]: [string, any]) => {
vueApp.component(`magic-ui-${type}`, component);
app.registerComponent(type, component);
});
Object.entries(dataSources.default).forEach(([type, ds]: [string, any]) => {

View File

@ -9,7 +9,7 @@ export default defineConfig({
plugins: [Vue()],
test: {
include: ['./packages/*/tests/**'],
include: ['./packages/*/tests/**', './runtime/*/tests/**'],
environment: 'jsdom',
environmentMatchGlobs: [['packages/cli/**', 'node']],
coverage: {

View File

@ -5,7 +5,7 @@
<component
v-if="display(item)"
:key="item.id"
:is="`magic-ui-${toLine(item.type)}`"
:is="useComponent({ componentType: item.type, app })"
:data-tmagic-id="item.id"
:data-tmagic-iterator-index="iteratorIndex"
:data-tmagic-iterator-container-id="iteratorContainerId"
@ -27,7 +27,7 @@ import { computed, defineComponent, type PropType } from 'vue-demi';
import type { Id, MContainer } from '@tmagic/schema';
import { IS_DSL_NODE_KEY, toLine } from '@tmagic/utils';
import { useApp } from '@tmagic/vue-runtime-help';
import { useApp, useComponent } from '@tmagic/vue-runtime-help';
interface ContainerSchema extends Omit<MContainer, 'id'> {
id?: Id;
@ -82,6 +82,7 @@ export default defineComponent({
display,
toLine,
useComponent,
};
},
});

View File

@ -1,12 +1,13 @@
<template>
<div>
<magic-ui-container
<component
:is="containerComponent"
v-for="(item, index) in configs"
:iterator-index="[...(iteratorIndex || []), index]"
:iterator-container-id="[...(iteratorContainerId || []), config.id]"
:key="index"
:config="item"
></magic-ui-container>
></component>
</div>
</template>
@ -15,7 +16,7 @@ import { computed, defineComponent, type PropType, watch } from 'vue-demi';
import type { IteratorContainer as TMagicIteratorContainer } from '@tmagic/core';
import type { Id, MIteratorContainer, MNode } from '@tmagic/schema';
import { useApp } from '@tmagic/vue-runtime-help';
import { useApp, useComponent } from '@tmagic/vue-runtime-help';
interface IteratorContainerSchema extends Omit<MIteratorContainer, 'id'> {
id?: Id;
@ -52,6 +53,8 @@ export default defineComponent({
methods: {},
});
const containerComponent = useComponent({ componentType: 'container', app });
const configs = computed<IteratorItemSchema[]>(() => {
let { iteratorData = [] } = props.config;
const { itemConfig, dsField, items } = props.config;
@ -115,6 +118,7 @@ export default defineComponent({
return {
configs,
containerComponent,
};
},
});

View File

@ -1,14 +1,14 @@
<template>
<magic-ui-container v-if="visible" :config="{ items: config.items }">
<component v-if="visible" :is="containerComponent" :config="{ items: config.items }">
<slot></slot>
</magic-ui-container>
</component>
</template>
<script lang="ts">
import { defineComponent, onBeforeUnmount, type PropType, ref } from 'vue-demi';
import type { Id, MContainer, MNode, MPage } from '@tmagic/schema';
import { useApp } from '@tmagic/vue-runtime-help';
import { useApp, useComponent } from '@tmagic/vue-runtime-help';
interface OverlaySchema extends Omit<MContainer, 'id'> {
id?: Id;
@ -48,6 +48,8 @@ export default defineComponent({
},
});
const containerComponent = useComponent({ componentType: 'container', app });
const editorSelectHandler = (
info: {
node: MNode;
@ -70,6 +72,7 @@ export default defineComponent({
});
return {
containerComponent,
visible,
};
},

View File

@ -1,11 +1,12 @@
<template>
<div>
<magic-ui-container
<component
:is="containerComponent"
:iterator-index="iteratorIndex"
:iterator-container-id="iteratorContainerId"
:config="containerConfig"
:model="model"
></magic-ui-container>
></component>
</div>
</template>
@ -13,7 +14,7 @@
import { computed, defineComponent, type PropType } from 'vue-demi';
import { type Id, type MComponent, type MNode, NodeType } from '@tmagic/schema';
import { useApp } from '@tmagic/vue-runtime-help';
import { useApp, useComponent } from '@tmagic/vue-runtime-help';
export default defineComponent({
props: {
@ -37,6 +38,8 @@ export default defineComponent({
iteratorIndex: props.iteratorIndex,
});
const containerComponent = useComponent({ componentType: 'container', app });
const fragment = computed(() => app?.dsl?.items?.find((page) => page.id === props.config.pageFragmentId));
const containerConfig = computed(() => {
@ -64,6 +67,7 @@ export default defineComponent({
});
return {
containerComponent,
containerConfig,
};
},

View File

@ -1,17 +1,18 @@
<template>
<magic-ui-container
<component
:is="containerComponent"
class="magic-ui-page-fragment"
:data-tmagic-id="config.id"
:config="config"
:style="app?.transformStyle(config.style || {})"
></magic-ui-container>
></component>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue-demi';
import type { MPageFragment } from '@tmagic/schema';
import { useApp } from '@tmagic/vue-runtime-help';
import { useApp, useComponent } from '@tmagic/vue-runtime-help';
export default defineComponent({
props: {
@ -31,8 +32,11 @@ export default defineComponent({
methods: {},
});
const containerComponent = useComponent({ componentType: 'container', app });
return {
app,
containerComponent,
};
},
});

View File

@ -1,12 +1,12 @@
<template>
<magic-ui-container class="magic-ui-page" :data-tmagic-id="config.id" :config="config"></magic-ui-container>
<component :is="containerComponent" class="magic-ui-page" :data-tmagic-id="config.id" :config="config"></component>
</template>
<script lang="ts">
import { defineComponent, type PropType } from 'vue-demi';
import type { MPage } from '@tmagic/schema';
import { useApp } from '@tmagic/vue-runtime-help';
import { useApp, useComponent } from '@tmagic/vue-runtime-help';
export default defineComponent({
props: {
@ -30,8 +30,10 @@ export default defineComponent({
methods: { refresh },
});
const containerComponent = useComponent({ componentType: 'container', app });
return {
app,
containerComponent,
};
},
});