()
+ if (instance.rawProps.length)
+ for (const props of Array.from(instance.rawProps).reverse()) {
+ if (isFunction(props)) {
+ const resolved = props()
+ for (const rawKey in resolved) {
+ registerAttr(rawKey, () => resolved[rawKey])
+ }
+ } else {
+ for (const rawKey in props) {
+ registerAttr(rawKey, props[rawKey])
+ }
+ }
+ }
+
+ for (const key in attrs) {
+ if (!keys.has(key)) {
+ delete attrs[key]
+ }
+ }
+
+ function registerAttr(key: string, getter: () => unknown) {
+ if (
+ (!options || !(camelize(key) in options)) &&
+ !isEmitListener(instance.emitsOptions, key)
+ ) {
+ keys.add(key)
+ if (key in attrs) return
+ Object.defineProperty(attrs, key, {
+ get: getter,
+ enumerable: true,
+ configurable: true,
+ })
+ }
+ }
+}
diff --git a/packages/runtime-vapor/src/componentEmits.ts b/packages/runtime-vapor/src/componentEmits.ts
index 138097c81..bbca2044c 100644
--- a/packages/runtime-vapor/src/componentEmits.ts
+++ b/packages/runtime-vapor/src/componentEmits.ts
@@ -1,5 +1,8 @@
// NOTE: runtime-core/src/componentEmits.ts
+// TODO WIP
+// @ts-nocheck
+
import {
EMPTY_OBJ,
type UnionToIntersection,
@@ -45,6 +48,8 @@ export function emit(
...rawArgs: any[]
) {
if (instance.isUnmounted) return
+ // TODO
+ // @ts-expect-error
const { rawProps } = instance
let args = rawArgs
diff --git a/packages/runtime-vapor/src/componentLifecycle.ts b/packages/runtime-vapor/src/componentLifecycle.ts
new file mode 100644
index 000000000..2dd94b1d5
--- /dev/null
+++ b/packages/runtime-vapor/src/componentLifecycle.ts
@@ -0,0 +1,34 @@
+import { invokeArrayFns } from '@vue/shared'
+import type { VaporLifecycleHooks } from './apiLifecycle'
+import { type ComponentInternalInstance, setCurrentInstance } from './component'
+import { queuePostRenderEffect } from './scheduler'
+import { type DirectiveHookName, invokeDirectiveHook } from './directives'
+
+export function invokeLifecycle(
+ instance: ComponentInternalInstance,
+ lifecycle: VaporLifecycleHooks,
+ directive: DirectiveHookName,
+ post?: boolean,
+) {
+ invokeArrayFns(post ? [invokeSub, invokeCurrent] : [invokeCurrent, invokeSub])
+
+ function invokeCurrent() {
+ const hooks = instance[lifecycle]
+ if (hooks) {
+ const fn = () => {
+ const reset = setCurrentInstance(instance)
+ invokeArrayFns(hooks)
+ reset()
+ }
+ post ? queuePostRenderEffect(fn) : fn()
+ }
+
+ invokeDirectiveHook(instance, directive)
+ }
+
+ function invokeSub() {
+ instance.comps.forEach(comp =>
+ invokeLifecycle(comp, lifecycle, directive, post),
+ )
+ }
+}
diff --git a/packages/runtime-vapor/src/componentProps.ts b/packages/runtime-vapor/src/componentProps.ts
index 5950a57a1..c1ebdf9ef 100644
--- a/packages/runtime-vapor/src/componentProps.ts
+++ b/packages/runtime-vapor/src/componentProps.ts
@@ -1,5 +1,3 @@
-// NOTE: runtime-core/src/componentProps.ts
-
import {
type Data,
EMPTY_ARR,
@@ -11,14 +9,15 @@ import {
isArray,
isFunction,
} from '@vue/shared'
-import { shallowReactive, shallowReadonly, toRaw } from '@vue/reactivity'
+import { baseWatch, shallowReactive } from '@vue/reactivity'
import { warn } from './warning'
import {
type Component,
type ComponentInternalInstance,
setCurrentInstance,
} from './component'
-import { isEmitListener } from './componentEmits'
+import { patchAttrs } from './componentAttrs'
+import { createVaporPreScheduler } from './scheduler'
export type ComponentPropsOptions =
| ComponentObjectPropsOptions
@@ -69,108 +68,129 @@ type NormalizedProp =
})
export type NormalizedProps = Record
-export type NormalizedPropsOptions = [NormalizedProps, string[]] | []
+export type NormalizedPropsOptions =
+ | [props: NormalizedProps, needCastKeys: string[]]
+ | []
+
+type StaticProps = Record unknown>
+type DynamicProps = () => Data
+export type NormalizedRawProps = Array
+export type RawProps = NormalizedRawProps | StaticProps | null
export function initProps(
instance: ComponentInternalInstance,
- rawProps: Data | null,
+ rawProps: RawProps,
isStateful: boolean,
) {
const props: Data = {}
- const attrs: Data = {}
+ const attrs = (instance.attrs = shallowReactive({}))
- const [options, needCastKeys] = instance.propsOptions
- let hasAttrsChanged = false
- let rawCastValues: Data | undefined
- if (rawProps) {
- for (let key in rawProps) {
- const valueGetter = () => rawProps[key]
- let camelKey
- if (options && hasOwn(options, (camelKey = camelize(key)))) {
- if (!needCastKeys || !needCastKeys.includes(camelKey)) {
- // NOTE: must getter
- // props[camelKey] = value
- Object.defineProperty(props, camelKey, {
- get() {
- return valueGetter()
- },
- enumerable: true,
- })
- } else {
- // NOTE: must getter
- // ;(rawCastValues || (rawCastValues = {}))[camelKey] = value
- rawCastValues || (rawCastValues = {})
- Object.defineProperty(rawCastValues, camelKey, {
- get() {
- return valueGetter()
- },
- enumerable: true,
- })
- }
- } else if (!isEmitListener(instance.emitsOptions, key)) {
- // if (!(key in attrs) || value !== attrs[key]) {
- if (!(key in attrs)) {
- // NOTE: must getter
- // attrs[key] = value
- Object.defineProperty(attrs, key, {
- get() {
- return valueGetter()
- },
- enumerable: true,
- })
- hasAttrsChanged = true
- }
- }
- }
- }
+ if (!rawProps) rawProps = []
+ else if (!isArray(rawProps)) rawProps = [rawProps]
+ instance.rawProps = rawProps
- if (needCastKeys) {
- const rawCurrentProps = toRaw(props)
- const castValues = rawCastValues || EMPTY_OBJ
- for (let i = 0; i < needCastKeys.length; i++) {
- const key = needCastKeys[i]
-
- // NOTE: must getter
- // props[key] = resolvePropValue(
- // options!,
- // rawCurrentProps,
- // key,
- // castValues[key],
- // instance,
- // !hasOwn(castValues, key),
- // )
- Object.defineProperty(props, key, {
- get() {
- return resolvePropValue(
- options!,
- rawCurrentProps,
- key,
- castValues[key],
+ const [options] = instance.propsOptions
+
+ const hasDynamicProps = rawProps.some(isFunction)
+ if (options) {
+ if (hasDynamicProps) {
+ for (const key in options) {
+ const getter = () =>
+ getDynamicPropValue(rawProps as NormalizedRawProps, key)
+ registerProp(instance, props, key, getter, true)
+ }
+ } else {
+ for (const key in options) {
+ const rawKey = rawProps[0] && getRawKey(rawProps[0] as StaticProps, key)
+ if (rawKey) {
+ registerProp(
instance,
- !hasOwn(castValues, key),
+ props,
+ key,
+ (rawProps[0] as StaticProps)[rawKey],
)
- },
- })
+ } else {
+ registerProp(instance, props, key, undefined, false, true)
+ }
+ }
}
}
// validation
if (__DEV__) {
- validateProps(rawProps || {}, props, instance)
+ validateProps(rawProps, props, options || {})
+ }
+
+ if (hasDynamicProps) {
+ baseWatch(() => patchAttrs(instance), undefined, {
+ scheduler: createVaporPreScheduler(instance),
+ })
+ } else {
+ patchAttrs(instance)
}
if (isStateful) {
- instance.props = shallowReactive(props)
+ instance.props = /* isSSR ? props : */ shallowReactive(props)
} else {
- if (instance.propsOptions === EMPTY_ARR) {
- instance.props = attrs
+ // functional w/ optional props, props === attrs
+ instance.props = instance.propsOptions === EMPTY_ARR ? attrs : props
+ }
+}
+
+function registerProp(
+ instance: ComponentInternalInstance,
+ props: Data,
+ rawKey: string,
+ getter?: (() => unknown) | (() => DynamicPropResult),
+ isDynamic?: boolean,
+ isAbsent?: boolean,
+) {
+ const key = camelize(rawKey)
+ if (key in props) return
+
+ const [options, needCastKeys] = instance.propsOptions
+ const needCast = needCastKeys && needCastKeys.includes(key)
+ const withCast = (value: unknown, absent?: boolean) =>
+ resolvePropValue(options!, props, key, value, instance, absent)
+
+ if (isAbsent) {
+ props[key] = needCast ? withCast(undefined, true) : undefined
+ } else {
+ const get: () => unknown = isDynamic
+ ? needCast
+ ? () => withCast(...(getter!() as DynamicPropResult))
+ : () => (getter!() as DynamicPropResult)[0]
+ : needCast
+ ? () => withCast(getter!())
+ : getter!
+
+ Object.defineProperty(props, key, {
+ get,
+ enumerable: true,
+ })
+ }
+}
+
+function getRawKey(obj: Data, key: string) {
+ return Object.keys(obj).find(k => camelize(k) === key)
+}
+
+type DynamicPropResult = [value: unknown, absent: boolean]
+function getDynamicPropValue(
+ rawProps: NormalizedRawProps,
+ key: string,
+): DynamicPropResult {
+ for (const props of Array.from(rawProps).reverse()) {
+ if (isFunction(props)) {
+ const resolved = props()
+ const rawKey = getRawKey(resolved, key)
+ if (rawKey) return [resolved[rawKey], false]
} else {
- instance.props = props
+ const rawKey = getRawKey(props, key)
+ if (rawKey) return [props[rawKey](), false]
}
}
- instance.attrs = attrs
-
- return hasAttrsChanged
+ return [undefined, true]
}
function resolvePropValue(
@@ -179,7 +199,7 @@ function resolvePropValue(
key: string,
value: unknown,
instance: ComponentInternalInstance,
- isAbsent: boolean,
+ isAbsent?: boolean,
) {
const opt = options[key]
if (opt != null) {
@@ -223,12 +243,12 @@ function resolvePropValue(
export function normalizePropsOptions(comp: Component): NormalizedPropsOptions {
// TODO: cahching?
- const raw = comp.props as any
- const normalized: NormalizedPropsOptions[0] = {}
+ const raw = comp.props
+ const normalized: NormalizedProps | undefined = {}
const needCastKeys: NormalizedPropsOptions[1] = []
if (!raw) {
- return EMPTY_ARR as any
+ return EMPTY_ARR as []
}
if (isArray(raw)) {
@@ -238,7 +258,7 @@ export function normalizePropsOptions(comp: Component): NormalizedPropsOptions {
normalized[normalizedKey] = EMPTY_OBJ
}
}
- } else if (raw) {
+ } else {
for (const key in raw) {
const normalizedKey = camelize(key)
if (validatePropName(normalizedKey)) {
@@ -267,6 +287,8 @@ export function normalizePropsOptions(comp: Component): NormalizedPropsOptions {
function validatePropName(key: string) {
if (key[0] !== '$') {
return true
+ } else if (__DEV__) {
+ warn(`Invalid prop name: "${key}" is a reserved property.`)
}
return false
}
@@ -296,22 +318,25 @@ function getTypeIndex(
* dev only
*/
function validateProps(
- rawProps: Data,
+ rawProps: NormalizedRawProps,
props: Data,
- instance: ComponentInternalInstance,
+ options: NormalizedProps,
) {
- const resolvedValues = toRaw(props)
- const options = instance.propsOptions[0]
+ const presentKeys: string[] = []
+ for (const props of rawProps) {
+ presentKeys.push(...Object.keys(isFunction(props) ? props() : props))
+ }
+
for (const key in options) {
- let opt = options[key]
- if (opt == null) continue
- validateProp(
- key,
- resolvedValues[key],
- opt,
- __DEV__ ? shallowReadonly(resolvedValues) : resolvedValues,
- !hasOwn(rawProps, key) && !hasOwn(rawProps, hyphenate(key)),
- )
+ const opt = options[key]
+ if (opt != null)
+ validateProp(
+ key,
+ props[key],
+ opt,
+ props,
+ !presentKeys.some(k => camelize(k) === key),
+ )
}
}
@@ -321,11 +346,11 @@ function validateProps(
function validateProp(
name: string,
value: unknown,
- prop: PropOptions,
+ option: PropOptions,
props: Data,
isAbsent: boolean,
) {
- const { required, validator } = prop
+ const { required, validator } = option
// required!
if (required && isAbsent) {
warn('Missing required prop: "' + name + '"')
diff --git a/packages/runtime-vapor/src/directives.ts b/packages/runtime-vapor/src/directives.ts
index 9bff0a8eb..15d2119c4 100644
--- a/packages/runtime-vapor/src/directives.ts
+++ b/packages/runtime-vapor/src/directives.ts
@@ -145,7 +145,3 @@ function callDirectiveHook(
])
resetTracking()
}
-
-export function resolveDirective() {
- // TODO
-}
diff --git a/packages/runtime-vapor/src/dom/element.ts b/packages/runtime-vapor/src/dom/element.ts
index d6d0dfa58..63c515d4a 100644
--- a/packages/runtime-vapor/src/dom/element.ts
+++ b/packages/runtime-vapor/src/dom/element.ts
@@ -1,5 +1,5 @@
import { isArray, toDisplayString } from '@vue/shared'
-import type { Block } from '../render'
+import type { Block } from '../apiRender'
/*! #__NO_SIDE_EFFECTS__ */
export function normalizeBlock(block: Block): Node[] {
diff --git a/packages/runtime-vapor/src/helpers/resolveAssets.ts b/packages/runtime-vapor/src/helpers/resolveAssets.ts
new file mode 100644
index 000000000..35ae0bb9a
--- /dev/null
+++ b/packages/runtime-vapor/src/helpers/resolveAssets.ts
@@ -0,0 +1,7 @@
+export function resolveComponent() {
+ // TODO
+}
+
+export function resolveDirective() {
+ // TODO
+}
diff --git a/packages/runtime-vapor/src/index.ts b/packages/runtime-vapor/src/index.ts
index f3891c4a4..2ce9a8360 100644
--- a/packages/runtime-vapor/src/index.ts
+++ b/packages/runtime-vapor/src/index.ts
@@ -46,7 +46,6 @@ export {
type FunctionalComponent,
type SetupFn,
} from './component'
-export { render, unmountComponent } from './render'
export { renderEffect, renderWatch } from './renderWatch'
export {
watch,
@@ -62,7 +61,6 @@ export {
} from './apiWatch'
export {
withDirectives,
- resolveDirective,
type Directive,
type DirectiveBinding,
type DirectiveHook,
@@ -88,7 +86,6 @@ export { on, delegate, delegateEvents, setDynamicEvents } from './dom/event'
export { setRef } from './dom/templateRef'
export { defineComponent } from './apiDefineComponent'
-export { createComponentInstance } from './component'
export {
onBeforeMount,
onMounted,
@@ -103,8 +100,17 @@ export {
onErrorCaptured,
// onServerPrefetch,
} from './apiLifecycle'
+export {
+ createVaporApp,
+ type App,
+ type AppConfig,
+ type AppContext,
+} from './apiCreateVaporApp'
export { createIf } from './apiCreateIf'
export { createFor } from './apiCreateFor'
+export { createComponent } from './apiCreateComponent'
+
+export { resolveComponent, resolveDirective } from './helpers/resolveAssets'
// **Internal** DOM-only runtime directive helpers
export {
diff --git a/packages/sfc-playground/src/App.vue b/packages/sfc-playground/src/App.vue
index a3525036c..1f0f3e62e 100644
--- a/packages/sfc-playground/src/App.vue
+++ b/packages/sfc-playground/src/App.vue
@@ -120,9 +120,9 @@ watch(
files.value['src/index.html'] = new File(
'src/index.html',
`
+
+
+
+ root comp
+ update
+ update2
+
+
+
diff --git a/playground/src/main.ts b/playground/src/main.ts
index e8f26d40f..58677200c 100644
--- a/playground/src/main.ts
+++ b/playground/src/main.ts
@@ -1,4 +1,4 @@
-import { createComponentInstance, render, unmountComponent } from 'vue/vapor'
+import { createVaporApp } from 'vue/vapor'
import { createApp } from 'vue'
import './style.css'
@@ -6,14 +6,11 @@ const modules = import.meta.glob('./**/*.(vue|js)')
const mod = (modules['.' + location.pathname] || modules['./App.vue'])()
mod.then(({ default: mod }) => {
- if (mod.vapor) {
- const instance = createComponentInstance(mod, {})
- render(instance, '#app')
- // @ts-expect-error
- globalThis.unmount = () => {
- unmountComponent(instance)
- }
- } else {
- createApp(mod).mount('#app')
+ const app = (mod.vapor ? createVaporApp : createApp)(mod)
+ app.mount('#app')
+
+ // @ts-expect-error
+ globalThis.unmount = () => {
+ app.unmount()
}
})
diff --git a/playground/src/props.js b/playground/src/props.js
index 8da476cc8..8127cc3a0 100644
--- a/playground/src/props.js
+++ b/playground/src/props.js
@@ -1,69 +1,53 @@
// @ts-check
import {
- children,
+ createComponent,
defineComponent,
on,
+ reactive,
ref,
- render as renderComponent,
+ renderEffect,
setText,
template,
- watch,
- watchEffect,
} from '@vue/vapor'
const t0 = template(' ')
export default defineComponent({
vapor: true,
- props: undefined,
-
- setup(_, {}) {
+ setup() {
const count = ref(1)
+ const props = reactive({
+ a: 'b',
+ 'foo-bar': 100,
+ })
const handleClick = () => {
count.value++
+ props['foo-bar']++
+ // @ts-expect-error
+ props.boolean = true
+ console.log(count)
}
- const __returned__ = { count, handleClick }
-
- Object.defineProperty(__returned__, '__isScriptSetup', {
- enumerable: false,
- value: true,
- })
-
- return __returned__
- },
-
- render(_ctx) {
- const n0 = t0()
- const n1 = /** @type {HTMLButtonElement} */ (children(n0, 0))
- on(n1, 'click', () => _ctx.handleClick)
- watchEffect(() => {
- setText(n1, _ctx.count)
- })
-
- // TODO: create component fn?
- // const c0 = createComponent(...)
- // insert(n0, c0)
- renderComponent(
- /** @type {any} */ (child),
-
- // TODO: proxy??
- {
- /* */
- get count() {
- return _ctx.count
- },
-
- /* */
- get inlineDouble() {
- return _ctx.count * 2
+ return (() => {
+ const n0 = /** @type {HTMLButtonElement} */ (t0())
+ on(n0, 'click', () => handleClick)
+ renderEffect(() => setText(n0, count.value))
+ /** @type {any} */
+ const n1 = createComponent(child, [
+ {
+ /* */
+ count: () => {
+ // console.trace('access')
+ return count.value
+ },
+ /* */
+ inlineDouble: () => count.value * 2,
+ id: () => 'hello',
},
- },
- // @ts-expect-error TODO
- n0[0],
- )
-
- return n0
+ () => props,
+ ])
+ return [n0, n1]
+ })()
},
})
@@ -74,25 +58,22 @@ const child = defineComponent({
props: {
count: { type: Number, default: 1 },
inlineDouble: { type: Number, default: 2 },
+ fooBar: { type: Number, required: true },
+ boolean: { type: Boolean },
},
- setup(props) {
- watch(
- () => props.count,
- v => console.log('count changed', v),
- )
- watch(
- () => props.inlineDouble,
- v => console.log('inlineDouble changed', v),
- )
- },
+ setup(props, { attrs }) {
+ console.log(props, { ...props })
+ console.log(attrs, { ...attrs })
- render(_ctx) {
- const n0 = t1()
- const n1 = children(n0, 0)
- watchEffect(() => {
- setText(n1, void 0, _ctx.count + ' * 2 = ' + _ctx.inlineDouble)
- })
- return n0
+ return (() => {
+ const n0 = /** @type {HTMLParagraphElement} */ (t1())
+ renderEffect(() =>
+ setText(n0, props.count + ' * 2 = ' + props.inlineDouble),
+ )
+ const n1 = /** @type {HTMLParagraphElement} */ (t1())
+ renderEffect(() => setText(n1, props.fooBar, ', ', props.boolean))
+ return [n0, n1]
+ })()
},
})
diff --git a/playground/src/sub-comp.vue b/playground/src/sub-comp.vue
new file mode 100644
index 000000000..33e7ebd64
--- /dev/null
+++ b/playground/src/sub-comp.vue
@@ -0,0 +1,36 @@
+
+
+
+ sub-comp
+ {{ props }}
+ {{ attrs }}
+ {{ keys(attrs) }}
+