diff --git a/packages/compiler-vapor/src/generators/component.ts b/packages/compiler-vapor/src/generators/component.ts index 80550bf6f..af7482db2 100644 --- a/packages/compiler-vapor/src/generators/component.ts +++ b/packages/compiler-vapor/src/generators/component.ts @@ -12,6 +12,7 @@ import { import { genExpression } from './expression' import { genPropKey } from './prop' +// TODO: generate component slots export function genCreateComponent( oper: CreateComponentIRNode, context: CodegenContext, diff --git a/packages/runtime-vapor/__tests__/apiInject.spec.ts b/packages/runtime-vapor/__tests__/apiInject.spec.ts index 1b0b35cd3..3414f136c 100644 --- a/packages/runtime-vapor/__tests__/apiInject.spec.ts +++ b/packages/runtime-vapor/__tests__/apiInject.spec.ts @@ -6,7 +6,6 @@ import { createComponent, createTextNode, createVaporApp, - getCurrentInstance, hasInjectionContext, inject, nextTick, diff --git a/packages/runtime-vapor/__tests__/componentAttrs.spec.ts b/packages/runtime-vapor/__tests__/componentAttrs.spec.ts index 8a1970eec..46a472b03 100644 --- a/packages/runtime-vapor/__tests__/componentAttrs.spec.ts +++ b/packages/runtime-vapor/__tests__/componentAttrs.spec.ts @@ -40,6 +40,8 @@ describe('attribute fallthrough', () => { id: () => _ctx.id, }, ], + null, + null, true, ) }, @@ -85,6 +87,8 @@ describe('attribute fallthrough', () => { id: () => _ctx.id, }, ], + null, + null, true, ) }, @@ -123,6 +127,8 @@ describe('attribute fallthrough', () => { 'custom-attr': () => 'custom-attr', }, ], + null, + null, true, ) return n0 @@ -144,6 +150,8 @@ describe('attribute fallthrough', () => { id: () => _ctx.id, }, ], + null, + null, true, ) }, diff --git a/packages/runtime-vapor/__tests__/componentProps.spec.ts b/packages/runtime-vapor/__tests__/componentProps.spec.ts index ba85905c6..4487c65b1 100644 --- a/packages/runtime-vapor/__tests__/componentProps.spec.ts +++ b/packages/runtime-vapor/__tests__/componentProps.spec.ts @@ -244,6 +244,8 @@ describe('component: props', () => { foo: () => _ctx.foo, id: () => _ctx.id, }, + null, + null, true, ) }, diff --git a/packages/runtime-vapor/__tests__/componentSlots.spec.ts b/packages/runtime-vapor/__tests__/componentSlots.spec.ts new file mode 100644 index 000000000..ec8788060 --- /dev/null +++ b/packages/runtime-vapor/__tests__/componentSlots.spec.ts @@ -0,0 +1,191 @@ +// NOTE: This test is implemented based on the case of `runtime-core/__test__/componentSlots.spec.ts`. + +import { + createComponent, + createVaporApp, + defineComponent, + getCurrentInstance, + nextTick, + ref, + template, +} from '../src' +import { makeRender } from './_utils' + +const define = makeRender() +function renderWithSlots(slots: any): any { + let instance: any + const Comp = defineComponent({ + render() { + const t0 = template('
') + const n0 = t0() + instance = getCurrentInstance() + return n0 + }, + }) + + const { render } = define({ + render() { + return createComponent(Comp, {}, slots) + }, + }) + + render() + return instance +} + +describe('component: slots', () => { + test('initSlots: instance.slots should be set correctly', () => { + const { slots } = renderWithSlots({ _: 1 }) + expect(slots).toMatchObject({ _: 1 }) + }) + + // NOTE: slot normalization is not supported + test.todo( + 'initSlots: should normalize object slots (when value is null, string, array)', + () => {}, + ) + test.todo( + 'initSlots: should normalize object slots (when value is function)', + () => {}, + ) + + test('initSlots: instance.slots should be set correctly', () => { + let instance: any + const Comp = defineComponent({ + render() { + const t0 = template('
') + const n0 = t0() + instance = getCurrentInstance() + return n0 + }, + }) + + const { render } = define({ + render() { + return createComponent(Comp, {}, { header: () => template('header')() }) + }, + }) + + render() + + expect(instance.slots.header()).toMatchObject( + document.createTextNode('header'), + ) + }) + + // runtime-core's "initSlots: instance.slots should be set correctly (when vnode.shapeFlag is not SLOTS_CHILDREN)" + test('initSlots: instance.slots should be set correctly', () => { + const { slots } = renderWithSlots({ + default: () => template('')(), + }) + + // expect( + // '[Vue warn]: Non-function value encountered for default slot. Prefer function slots for better performance.', + // ).toHaveBeenWarned() + + expect(slots.default()).toMatchObject(document.createElement('span')) + }) + + test('updateSlots: instance.slots should be updated correctly', async () => { + const flag1 = ref(true) + + let instance: any + const Child = () => { + instance = getCurrentInstance() + return template('child')() + } + + const { render } = define({ + render() { + return createComponent(Child, {}, { _: 2 as any }, () => [ + flag1.value + ? { name: 'one', fn: () => template('')() } + : { name: 'two', fn: () => template('
')() }, + ]) + }, + }) + + render() + + expect(instance.slots).toHaveProperty('one') + expect(instance.slots).not.toHaveProperty('two') + + flag1.value = false + await nextTick() + + expect(instance.slots).not.toHaveProperty('one') + expect(instance.slots).toHaveProperty('two') + }) + + // NOTE: it is not supported + // test('updateSlots: instance.slots should be updated correctly (when slotType is null)', () => {}) + + // runtime-core's "updateSlots: instance.slots should be update correctly (when vnode.shapeFlag is not SLOTS_CHILDREN)" + test('updateSlots: instance.slots should be update correctly', async () => { + const flag1 = ref(true) + + let instance: any + const Child = () => { + instance = getCurrentInstance() + return template('child')() + } + + const { render } = define({ + setup() { + return createComponent(Child, {}, {}, () => [ + flag1.value + ? [{ name: 'header', fn: () => template('header')() }] + : [{ name: 'footer', fn: () => template('footer')() }], + ]) + }, + }) + render() + + expect(instance.slots).toHaveProperty('header') + flag1.value = false + await nextTick() + + // expect( + // '[Vue warn]: Non-function value encountered for default slot. Prefer function slots for better performance.', + // ).toHaveBeenWarned() + + expect(instance.slots).toHaveProperty('footer') + }) + + test.todo('should respect $stable flag', async () => { + // TODO: $stable flag? + }) + + test.todo('should not warn when mounting another app in setup', () => { + // TODO: warning + const Comp = defineComponent({ + render() { + const i = getCurrentInstance() + return i!.slots.default!() + }, + }) + const mountComp = () => { + createVaporApp({ + render() { + return createComponent( + Comp, + {}, + { default: () => template('msg')() }, + )! + }, + }) + } + const App = { + setup() { + mountComp() + }, + render() { + return null! + }, + } + createVaporApp(App).mount(document.createElement('div')) + expect( + 'Slot "default" invoked outside of the render function', + ).not.toHaveBeenWarned() + }) +}) diff --git a/packages/runtime-vapor/src/apiCreateComponent.ts b/packages/runtime-vapor/src/apiCreateComponent.ts index 133b40fc5..cf282706b 100644 --- a/packages/runtime-vapor/src/apiCreateComponent.ts +++ b/packages/runtime-vapor/src/apiCreateComponent.ts @@ -5,17 +5,22 @@ import { } from './component' import { setupComponent } from './apiRender' import type { RawProps } from './componentProps' +import type { DynamicSlots, Slots } from './componentSlots' import { withAttrs } from './componentAttrs' export function createComponent( comp: Component, rawProps: RawProps | null = null, + slots: Slots | null = null, + dynamicSlots: DynamicSlots | null = null, singleRoot: boolean = false, ) { const current = currentInstance! const instance = createComponentInstance( comp, singleRoot ? withAttrs(rawProps) : rawProps, + slots, + dynamicSlots, ) setupComponent(instance, singleRoot) diff --git a/packages/runtime-vapor/src/apiCreateVaporApp.ts b/packages/runtime-vapor/src/apiCreateVaporApp.ts index 10f98b472..2d07bbba3 100644 --- a/packages/runtime-vapor/src/apiCreateVaporApp.ts +++ b/packages/runtime-vapor/src/apiCreateVaporApp.ts @@ -41,7 +41,13 @@ export function createVaporApp( mount(rootContainer): any { if (!instance) { - instance = createComponentInstance(rootComponent, rootProps, context) + instance = createComponentInstance( + rootComponent, + rootProps, + null, + null, + context, + ) setupComponent(instance) render(instance, rootContainer) return instance diff --git a/packages/runtime-vapor/src/component.ts b/packages/runtime-vapor/src/component.ts index 883c36d94..482be5661 100644 --- a/packages/runtime-vapor/src/component.ts +++ b/packages/runtime-vapor/src/component.ts @@ -17,6 +17,12 @@ import { emit, normalizeEmitsOptions, } from './componentEmits' +import { + type DynamicSlots, + type InternalSlots, + type Slots, + initSlots, +} from './componentSlots' import { VaporLifecycleHooks } from './apiLifecycle' import { warn } from './warning' import { type AppContext, createAppContext } from './apiCreateVaporApp' @@ -32,7 +38,7 @@ export type SetupContext = E extends any attrs: Data emit: EmitFn expose: (exposed?: Record) => void - // TODO slots + slots: Readonly } : never @@ -46,6 +52,9 @@ export function createSetupContext( get attrs() { return getAttrsProxy(instance) }, + get slots() { + return getSlotsProxy(instance) + }, get emit() { return (event: string, ...args: any[]) => instance.emit(event, ...args) }, @@ -57,6 +66,7 @@ export function createSetupContext( return getAttrsProxy(instance) }, emit: instance.emit, + slots: instance.slots, expose: NOOP, } } @@ -102,9 +112,11 @@ export interface ComponentInternalInstance { emit: EmitFn emitted: Record | null attrs: Data + slots: InternalSlots refs: Data - attrsProxy: Data | null + attrsProxy?: Data + slotsProxy?: Slots // lifecycle isMounted: boolean @@ -188,6 +200,8 @@ let uid = 0 export function createComponentInstance( component: ObjectComponent | FunctionalComponent, rawProps: RawProps | null, + slots: Slots | null = null, + dynamicSlots: DynamicSlots | null = null, // application root node only appContext: AppContext | null = null, ): ComponentInternalInstance { @@ -224,10 +238,9 @@ export function createComponentInstance( emit: null!, emitted: null, attrs: EMPTY_OBJ, + slots: EMPTY_OBJ, refs: EMPTY_OBJ, - attrsProxy: null, - // lifecycle isMounted: false, isUnmounted: false, @@ -283,6 +296,7 @@ export function createComponentInstance( // [VaporLifecycleHooks.SERVER_PREFETCH]: null, } initProps(instance, rawProps, !isFunction(component)) + initSlots(instance, slots, dynamicSlots) instance.emit = emit.bind(null, instance) return instance @@ -315,3 +329,17 @@ function getAttrsProxy(instance: ComponentInternalInstance): Data { )) ) } + +/** + * Dev-only + */ +function getSlotsProxy(instance: ComponentInternalInstance): Slots { + return ( + instance.slotsProxy || + (instance.slotsProxy = new Proxy(instance.slots, { + get(target, key: string) { + return target[key] + }, + })) + ) +} diff --git a/packages/runtime-vapor/src/componentSlots.ts b/packages/runtime-vapor/src/componentSlots.ts new file mode 100644 index 000000000..4eba7abf7 --- /dev/null +++ b/packages/runtime-vapor/src/componentSlots.ts @@ -0,0 +1,80 @@ +import { type IfAny, extend, isArray } from '@vue/shared' +import { baseWatch } from '@vue/reactivity' +import type { ComponentInternalInstance } from './component' +import type { Block } from './apiRender' +import { createVaporPreScheduler } from './scheduler' + +// TODO: SSR + +export type Slot = ( + ...args: IfAny +) => Block + +export type InternalSlots = { + [name: string]: Slot | undefined +} + +export type Slots = Readonly + +export interface DynamicSlot { + name: string + fn: Slot + key?: string +} + +export type DynamicSlots = () => (DynamicSlot | DynamicSlot[])[] + +export const initSlots = ( + instance: ComponentInternalInstance, + rawSlots: InternalSlots | null = null, + dynamicSlots: DynamicSlots | null = null, +) => { + const slots: InternalSlots = extend({}, rawSlots) + + if (dynamicSlots) { + const dynamicSlotKeys: Record = {} + baseWatch( + () => { + const _dynamicSlots = dynamicSlots() + for (let i = 0; i < _dynamicSlots.length; i++) { + const slot = _dynamicSlots[i] + // array of dynamic slot generated by