diff --git a/packages/compiler-core/__tests__/transforms/vModel.spec.ts b/packages/compiler-core/__tests__/transforms/vModel.spec.ts index 82dd4909fd6..d092e72019e 100644 --- a/packages/compiler-core/__tests__/transforms/vModel.spec.ts +++ b/packages/compiler-core/__tests__/transforms/vModel.spec.ts @@ -449,7 +449,7 @@ describe('compiler: transform v-model', () => { expect(codegen.dynamicProps).toBe(`["modelValue", "onUpdate:modelValue"]`) }) - test('should generate modelModifiers for component v-model', () => { + test('should generate modelValueModifiers for component v-model', () => { const root = parseWithVModel('', { prefixIdentifiers: true, }) @@ -461,7 +461,7 @@ describe('compiler: transform v-model', () => { { key: { content: `modelValue` } }, { key: { content: `onUpdate:modelValue` } }, { - key: { content: 'modelModifiers' }, + key: { content: 'modelValueModifiers' }, value: { content: `{ trim: true, "bar-baz": true }`, isStatic: false, @@ -469,7 +469,7 @@ describe('compiler: transform v-model', () => { }, ], }) - // should NOT include modelModifiers in dynamicPropNames because it's never + // should NOT include modelValueModifiers in dynamicPropNames because it's never // gonna change expect(vnodeCall.dynamicProps).toBe(`["modelValue", "onUpdate:modelValue"]`) }) diff --git a/packages/compiler-core/src/babelUtils.ts b/packages/compiler-core/src/babelUtils.ts index 6ede6bd0386..342050dc496 100644 --- a/packages/compiler-core/src/babelUtils.ts +++ b/packages/compiler-core/src/babelUtils.ts @@ -30,10 +30,6 @@ export function walkIdentifiers( parentStack: Node[] = [], knownIds: Record = Object.create(null), ): void { - if (__BROWSER__) { - return - } - const rootExp = root.type === 'Program' ? root.body[0].type === 'ExpressionStatement' && root.body[0].expression @@ -110,10 +106,6 @@ export function isReferencedIdentifier( parent: Node | null, parentStack: Node[], ): boolean { - if (__BROWSER__) { - return false - } - if (!parent) { return true } diff --git a/packages/compiler-core/src/transforms/vModel.ts b/packages/compiler-core/src/transforms/vModel.ts index 598c1ea4387..c75ef1c3e60 100644 --- a/packages/compiler-core/src/transforms/vModel.ts +++ b/packages/compiler-core/src/transforms/vModel.ts @@ -138,7 +138,7 @@ export const transformModel: DirectiveTransform = (dir, node, context) => { ? isStaticExp(arg) ? `${arg.content}Modifiers` : createCompoundExpression([arg, ' + "Modifiers"']) - : `modelModifiers` + : `modelValueModifiers` props.push( createObjectProperty( modifiersKey, diff --git a/packages/compiler-sfc/__tests__/compileScript/__snapshots__/defineModel.spec.ts.snap b/packages/compiler-sfc/__tests__/compileScript/__snapshots__/defineModel.spec.ts.snap index 12462dcf423..cdf71076819 100644 --- a/packages/compiler-sfc/__tests__/compileScript/__snapshots__/defineModel.spec.ts.snap +++ b/packages/compiler-sfc/__tests__/compileScript/__snapshots__/defineModel.spec.ts.snap @@ -6,7 +6,7 @@ exports[`defineModel() > basic usage 1`] = ` export default { props: { "modelValue": { required: true }, - "modelModifiers": {}, + "modelValueModifiers": {}, "count": {}, "countModifiers": {}, "toString": { type: Function }, @@ -34,7 +34,7 @@ export default /*@__PURE__*/_defineComponent({ "modelValue": { required: true }, - "modelModifiers": {}, + "modelValueModifiers": {}, }, emits: ["update:modelValue"], setup(__props, { expose: __expose }) { @@ -60,7 +60,7 @@ export default /*@__PURE__*/_defineComponent({ default: 0, required: true, }, - "modelModifiers": {}, + "modelValueModifiers": {}, }, emits: ["update:modelValue"], setup(__props, { expose: __expose }) { @@ -86,7 +86,7 @@ export default /*@__PURE__*/_defineComponent({ }, { "modelValue": { }, - "modelModifiers": {}, + "modelValueModifiers": {}, }), emits: ["update:modelValue"], setup(__props: any, { expose: __expose }) { @@ -109,7 +109,7 @@ exports[`defineModel() > w/ Boolean And Function types, production mode 1`] = ` export default /*@__PURE__*/_defineComponent({ props: { "modelValue": { type: [Boolean, String] }, - "modelModifiers": {}, + "modelValueModifiers": {}, }, emits: ["update:modelValue"], setup(__props, { expose: __expose }) { @@ -150,7 +150,7 @@ exports[`defineModel() > w/ defineProps and defineEmits 1`] = ` export default { props: /*@__PURE__*/_mergeModels({ foo: String }, { "modelValue": { default: 0 }, - "modelModifiers": {}, + "modelValueModifiers": {}, }), emits: /*@__PURE__*/_mergeModels(['change'], ["update:modelValue"]), setup(__props, { expose: __expose }) { @@ -172,7 +172,7 @@ exports[`defineModel() > w/ types, basic usage 1`] = ` export default /*@__PURE__*/_defineComponent({ props: { "modelValue": { type: [Boolean, String] }, - "modelModifiers": {}, + "modelValueModifiers": {}, "count": { type: Number }, "countModifiers": {}, "disabled": { type: Number, ...{ required: false } }, @@ -201,7 +201,7 @@ exports[`defineModel() > w/ types, production mode 1`] = ` export default /*@__PURE__*/_defineComponent({ props: { "modelValue": { type: Boolean }, - "modelModifiers": {}, + "modelValueModifiers": {}, "fn": {}, "fnModifiers": {}, "fnWithDefault": { type: Function, ...{ default: () => null } }, @@ -233,7 +233,7 @@ exports[`defineModel() > w/ types, production mode, boolean + multiple types 1`] export default /*@__PURE__*/_defineComponent({ props: { "modelValue": { type: [Boolean, String, Object] }, - "modelModifiers": {}, + "modelValueModifiers": {}, }, emits: ["update:modelValue"], setup(__props, { expose: __expose }) { @@ -253,7 +253,7 @@ exports[`defineModel() > w/ types, production mode, function + runtime opts + mu export default /*@__PURE__*/_defineComponent({ props: { "modelValue": { type: [Number, Function], ...{ default: () => 1 } }, - "modelModifiers": {}, + "modelValueModifiers": {}, }, emits: ["update:modelValue"], setup(__props, { expose: __expose }) { diff --git a/packages/compiler-sfc/__tests__/compileScript/defineModel.spec.ts b/packages/compiler-sfc/__tests__/compileScript/defineModel.spec.ts index 5d696a95d88..210c18ddaec 100644 --- a/packages/compiler-sfc/__tests__/compileScript/defineModel.spec.ts +++ b/packages/compiler-sfc/__tests__/compileScript/defineModel.spec.ts @@ -94,7 +94,7 @@ describe('defineModel()', () => { ) assertCode(content) expect(content).toMatch('"modelValue": { type: [Boolean, String] }') - expect(content).toMatch('"modelModifiers": {}') + expect(content).toMatch('"modelValueModifiers": {}') expect(content).toMatch('"count": { type: Number }') expect(content).toMatch( '"disabled": { type: Number, ...{ required: false } }', diff --git a/packages/compiler-sfc/src/script/defineModel.ts b/packages/compiler-sfc/src/script/defineModel.ts index 05082800284..9c8f254fee4 100644 --- a/packages/compiler-sfc/src/script/defineModel.ts +++ b/packages/compiler-sfc/src/script/defineModel.ts @@ -167,9 +167,7 @@ export function genModelProps(ctx: ScriptCompileContext) { modelPropsDecl += `\n ${JSON.stringify(name)}: ${decl},` // also generate modifiers prop - const modifierPropName = JSON.stringify( - name === 'modelValue' ? `modelModifiers` : `${name}Modifiers`, - ) + const modifierPropName = JSON.stringify(`${name}Modifiers`) modelPropsDecl += `\n ${modifierPropName}: {},` } return `{${modelPropsDecl}\n }` diff --git a/packages/compiler-vapor/__tests__/__snapshots__/compile.spec.ts.snap b/packages/compiler-vapor/__tests__/__snapshots__/compile.spec.ts.snap index e56676d8706..d4a8b6827bb 100644 --- a/packages/compiler-vapor/__tests__/__snapshots__/compile.spec.ts.snap +++ b/packages/compiler-vapor/__tests__/__snapshots__/compile.spec.ts.snap @@ -157,9 +157,9 @@ export function render(_ctx, $props, $emit, $attrs, $slots) { const _component_Comp = _resolveComponent("Comp") const n0 = t0() const n3 = t1() + const n2 = _child(n3) _setInsertionState(n3, 0) const n1 = _createComponentWithFallback(_component_Comp) - const n2 = _child(n3) _renderEffect(() => { _setText(n2, _toDisplayString(_ctx.bar)) _setProp(n3, "id", _ctx.foo) @@ -230,6 +230,30 @@ export function render(_ctx) { }" `; +exports[`compile > setInsertionState > next, child and nthChild should be above the setInsertionState 1`] = ` +"import { resolveComponent as _resolveComponent, child as _child, next as _next, setInsertionState as _setInsertionState, createComponentWithFallback as _createComponentWithFallback, nthChild as _nthChild, createIf as _createIf, setProp as _setProp, renderEffect as _renderEffect, template as _template } from 'vue'; +const t0 = _template("
") +const t1 = _template("
", true) + +export function render(_ctx) { + const _component_Comp = _resolveComponent("Comp") + const n6 = t1() + const n5 = _next(_child(n6)) + const n7 = _nthChild(n6, 3) + const p0 = _next(n7) + const n4 = _child(p0) + _setInsertionState(n6, n5) + const n0 = _createComponentWithFallback(_component_Comp) + _setInsertionState(n6, n7) + const n1 = _createIf(() => (true), () => { + const n3 = t0() + return n3 + }) + _renderEffect(() => _setProp(n4, "disabled", _ctx.foo)) + return n6 +}" +`; + exports[`compile > static + dynamic root 1`] = ` "import { toDisplayString as _toDisplayString, setText as _setText, template as _template } from 'vue'; const t0 = _template(" ") diff --git a/packages/compiler-vapor/__tests__/compile.spec.ts b/packages/compiler-vapor/__tests__/compile.spec.ts index 33f399caa77..3a2ce41f0cd 100644 --- a/packages/compiler-vapor/__tests__/compile.spec.ts +++ b/packages/compiler-vapor/__tests__/compile.spec.ts @@ -220,4 +220,21 @@ describe('compile', () => { expect(code).matchSnapshot() }) }) + + describe('setInsertionState', () => { + test('next, child and nthChild should be above the setInsertionState', () => { + const code = compile(` +
+
+ +
+
+
+
+
+ `) + expect(code).toMatchSnapshot() + }) + }) }) diff --git a/packages/compiler-vapor/__tests__/transforms/__snapshots__/transformElement.spec.ts.snap b/packages/compiler-vapor/__tests__/transforms/__snapshots__/transformElement.spec.ts.snap index 69527c0b100..a3726401180 100644 --- a/packages/compiler-vapor/__tests__/transforms/__snapshots__/transformElement.spec.ts.snap +++ b/packages/compiler-vapor/__tests__/transforms/__snapshots__/transformElement.spec.ts.snap @@ -224,6 +224,36 @@ export function render(_ctx) { }" `; +exports[`compiler: element transform > component event with keys modifier 1`] = ` +"import { resolveComponent as _resolveComponent, withKeys as _withKeys, createComponentWithFallback as _createComponentWithFallback } from 'vue'; + +export function render(_ctx) { + const _component_Foo = _resolveComponent("Foo") + const n0 = _createComponentWithFallback(_component_Foo, { onKeyup: () => _withKeys(_ctx.bar, ["enter"]) }, null, true) + return n0 +}" +`; + +exports[`compiler: element transform > component event with multiple modifiers and event options 1`] = ` +"import { resolveComponent as _resolveComponent, withModifiers as _withModifiers, withKeys as _withKeys, createComponentWithFallback as _createComponentWithFallback } from 'vue'; + +export function render(_ctx) { + const _component_Foo = _resolveComponent("Foo") + const n0 = _createComponentWithFallback(_component_Foo, { onFooCaptureOnce: () => _withKeys(_withModifiers(_ctx.bar, ["stop","prevent"]), ["enter"]) }, null, true) + return n0 +}" +`; + +exports[`compiler: element transform > component event with nonKeys modifier 1`] = ` +"import { resolveComponent as _resolveComponent, withModifiers as _withModifiers, createComponentWithFallback as _createComponentWithFallback } from 'vue'; + +export function render(_ctx) { + const _component_Foo = _resolveComponent("Foo") + const n0 = _createComponentWithFallback(_component_Foo, { onFoo: () => _withModifiers(_ctx.bar, ["stop","prevent"]) }, null, true) + return n0 +}" +`; + exports[`compiler: element transform > component event with once modifier 1`] = ` "import { resolveComponent as _resolveComponent, createComponentWithFallback as _createComponentWithFallback } from 'vue'; diff --git a/packages/compiler-vapor/__tests__/transforms/__snapshots__/vModel.spec.ts.snap b/packages/compiler-vapor/__tests__/transforms/__snapshots__/vModel.spec.ts.snap index 5ef064974c0..14330e57811 100644 --- a/packages/compiler-vapor/__tests__/transforms/__snapshots__/vModel.spec.ts.snap +++ b/packages/compiler-vapor/__tests__/transforms/__snapshots__/vModel.spec.ts.snap @@ -1,13 +1,13 @@ // Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html -exports[`compiler: vModel transform > component > v-model for component should generate modelModifiers 1`] = ` +exports[`compiler: vModel transform > component > v-model for component should generate modelValueModifiers 1`] = ` "import { resolveComponent as _resolveComponent, createComponentWithFallback as _createComponentWithFallback } from 'vue'; export function render(_ctx) { const _component_Comp = _resolveComponent("Comp") const n0 = _createComponentWithFallback(_component_Comp, { modelValue: () => (_ctx.foo), "onUpdate:modelValue": () => _value => (_ctx.foo = _value), - modelModifiers: () => ({ trim: true, "bar-baz": true }) }, null, true) + modelValueModifiers: () => ({ trim: true, "bar-baz": true }) }, null, true) return n0 }" `; diff --git a/packages/compiler-vapor/__tests__/transforms/transformChildren.spec.ts b/packages/compiler-vapor/__tests__/transforms/transformChildren.spec.ts index e656312356c..2d8ae8c960d 100644 --- a/packages/compiler-vapor/__tests__/transforms/transformChildren.spec.ts +++ b/packages/compiler-vapor/__tests__/transforms/transformChildren.spec.ts @@ -69,8 +69,8 @@ describe('compiler: children transform', () => {
`, ) // ensure the insertion anchor is generated before the insertion statement - expect(code).toMatch(`const n3 = _next(_child(n4)) - _setInsertionState(n4, n3)`) + expect(code).toMatch(`const n3 = _next(_child(n4))`) + expect(code).toMatch(`_setInsertionState(n4, n3)`) expect(code).toMatchSnapshot() }) }) diff --git a/packages/compiler-vapor/__tests__/transforms/transformElement.spec.ts b/packages/compiler-vapor/__tests__/transforms/transformElement.spec.ts index adaad182cf3..ca64729e050 100644 --- a/packages/compiler-vapor/__tests__/transforms/transformElement.spec.ts +++ b/packages/compiler-vapor/__tests__/transforms/transformElement.spec.ts @@ -878,6 +878,78 @@ describe('compiler: element transform', () => { }) }) + test('component event with keys modifier', () => { + const { code, ir } = compileWithElementTransform( + ``, + ) + expect(code).toMatchSnapshot() + expect(ir.block.dynamic.children[0].operation).toMatchObject({ + type: IRNodeTypes.CREATE_COMPONENT_NODE, + tag: 'Foo', + props: [ + [ + { + key: { content: 'keyup' }, + handler: true, + handlerModifiers: { + keys: ['enter'], + nonKeys: [], + options: [], + }, + }, + ], + ], + }) + }) + + test('component event with nonKeys modifier', () => { + const { code, ir } = compileWithElementTransform( + ``, + ) + expect(code).toMatchSnapshot() + expect(ir.block.dynamic.children[0].operation).toMatchObject({ + type: IRNodeTypes.CREATE_COMPONENT_NODE, + tag: 'Foo', + props: [ + [ + { + key: { content: 'foo' }, + handler: true, + handlerModifiers: { + keys: [], + nonKeys: ['stop', 'prevent'], + options: [], + }, + }, + ], + ], + }) + }) + + test('component event with multiple modifiers and event options', () => { + const { code, ir } = compileWithElementTransform( + ``, + ) + expect(code).toMatchSnapshot() + expect(ir.block.dynamic.children[0].operation).toMatchObject({ + type: IRNodeTypes.CREATE_COMPONENT_NODE, + tag: 'Foo', + props: [ + [ + { + key: { content: 'foo' }, + handler: true, + handlerModifiers: { + keys: ['enter'], + nonKeys: ['stop', 'prevent'], + options: ['capture', 'once'], + }, + }, + ], + ], + }) + }) + test('component with dynamic event arguments', () => { const { code, ir } = compileWithElementTransform( ``, diff --git a/packages/compiler-vapor/__tests__/transforms/vModel.spec.ts b/packages/compiler-vapor/__tests__/transforms/vModel.spec.ts index 51eaa9e0230..bed60ff63a5 100644 --- a/packages/compiler-vapor/__tests__/transforms/vModel.spec.ts +++ b/packages/compiler-vapor/__tests__/transforms/vModel.spec.ts @@ -266,13 +266,13 @@ describe('compiler: vModel transform', () => { }) }) - test('v-model for component should generate modelModifiers', () => { + test('v-model for component should generate modelValueModifiers', () => { const { code, ir } = compileWithVModel( '', ) expect(code).toMatchSnapshot() expect(code).contain( - `modelModifiers: () => ({ trim: true, "bar-baz": true })`, + `modelValueModifiers: () => ({ trim: true, "bar-baz": true })`, ) expect(ir.block.dynamic.children[0].operation).toMatchObject({ type: IRNodeTypes.CREATE_COMPONENT_NODE, diff --git a/packages/compiler-vapor/src/generate.ts b/packages/compiler-vapor/src/generate.ts index 193a0f5da77..469e55a707f 100644 --- a/packages/compiler-vapor/src/generate.ts +++ b/packages/compiler-vapor/src/generate.ts @@ -19,7 +19,14 @@ import { } from './generators/utils' import { setTemplateRefIdent } from './generators/templateRef' -export type CodegenOptions = Omit +type CustomGenOperation = ( + opers: any, + context: CodegenContext, +) => CodeFragment[] | void + +export type CodegenOptions = Omit & { + customGenOperation?: CustomGenOperation | null +} export class CodegenContext { options: Required @@ -87,6 +94,7 @@ export class CodegenContext { inline: false, bindingMetadata: {}, expressionPlugins: [], + customGenOperation: null, } this.options = extend(defaultOptions, options) this.block = ir.block diff --git a/packages/compiler-vapor/src/generators/block.ts b/packages/compiler-vapor/src/generators/block.ts index b161b8f45d1..48e4b5cb890 100644 --- a/packages/compiler-vapor/src/generators/block.ts +++ b/packages/compiler-vapor/src/generators/block.ts @@ -52,7 +52,7 @@ export function genBlockContent( push(...genSelf(child, context)) } for (const child of dynamic.children) { - push(...genChildren(child, context, `n${child.id!}`)) + push(...genChildren(child, context, push, `n${child.id!}`)) } push(...genOperations(operation, context)) diff --git a/packages/compiler-vapor/src/generators/component.ts b/packages/compiler-vapor/src/generators/component.ts index 7c232db754b..52b95927425 100644 --- a/packages/compiler-vapor/src/generators/component.ts +++ b/packages/compiler-vapor/src/generators/component.ts @@ -211,7 +211,7 @@ function genProp(prop: IRProp, context: CodegenContext, isStatic?: boolean) { ? genEventHandler( context, prop.values[0], - undefined, + prop.handlerModifiers, true /* wrap handlers passed to components */, ) : isStatic @@ -240,9 +240,7 @@ function genModelModifiers( if (!modelModifiers || !modelModifiers.length) return [] const modifiersKey = key.isStatic - ? key.content === 'modelValue' - ? [`modelModifiers`] - : [`${key.content}Modifiers`] + ? [`${key.content}Modifiers`] : ['[', ...genExpression(key, context), ' + "Modifiers"]'] const modifiersVal = genDirectiveModifiers(modelModifiers) diff --git a/packages/compiler-vapor/src/generators/operation.ts b/packages/compiler-vapor/src/generators/operation.ts index 4247bc6feca..2ca8f6a5c7b 100644 --- a/packages/compiler-vapor/src/generators/operation.ts +++ b/packages/compiler-vapor/src/generators/operation.ts @@ -88,6 +88,11 @@ export function genOperation( case IRNodeTypes.GET_TEXT_CHILD: return genGetTextChild(oper, context) default: + if (context.options.customGenOperation) { + const result = context.options.customGenOperation(oper, context) + if (result) return result + } + const exhaustiveCheck: never = oper throw new Error( `Unhandled operation type in genOperation: ${exhaustiveCheck}`, diff --git a/packages/compiler-vapor/src/generators/prop.ts b/packages/compiler-vapor/src/generators/prop.ts index 62fca087ed7..a873f1fede3 100644 --- a/packages/compiler-vapor/src/generators/prop.ts +++ b/packages/compiler-vapor/src/generators/prop.ts @@ -114,9 +114,10 @@ export function genPropKey( ): CodeFragment[] { const { helper } = context - const handlerModifierPostfix = handlerModifiers - ? handlerModifiers.map(capitalize).join('') - : '' + const handlerModifierPostfix = + handlerModifiers && handlerModifiers.options + ? handlerModifiers.options.map(capitalize).join('') + : '' // static arg was transformed by v-bind transformer if (node.isStatic) { // only quote keys if necessary diff --git a/packages/compiler-vapor/src/generators/template.ts b/packages/compiler-vapor/src/generators/template.ts index 356c1ccbe15..5a066b09e9a 100644 --- a/packages/compiler-vapor/src/generators/template.ts +++ b/packages/compiler-vapor/src/generators/template.ts @@ -41,6 +41,7 @@ export function genSelf( export function genChildren( dynamic: IRDynamicInfo, context: CodegenContext, + pushBlock: (...items: CodeFragment[]) => number, from: string = `n${dynamic.id}`, ): CodeFragment[] { const { helper } = context @@ -72,17 +73,17 @@ export function genChildren( // p for "placeholder" variables that are meant for possible reuse by // other access paths const variable = id === undefined ? `p${context.block.tempId++}` : `n${id}` - push(NEWLINE, `const ${variable} = `) + pushBlock(NEWLINE, `const ${variable} = `) if (prev) { if (elementIndex - prev[1] === 1) { - push(...genCall(helper('next'), prev[0])) + pushBlock(...genCall(helper('next'), prev[0])) } else { - push(...genCall(helper('nthChild'), from, String(elementIndex))) + pushBlock(...genCall(helper('nthChild'), from, String(elementIndex))) } } else { if (elementIndex === 0) { - push(...genCall(helper('child'), from)) + pushBlock(...genCall(helper('child'), from)) } else { // check if there's a node that we can reuse from let init = genCall(helper('child'), from) @@ -91,7 +92,7 @@ export function genChildren( } else if (elementIndex > 1) { init = genCall(helper('nthChild'), from, String(elementIndex)) } - push(...init) + pushBlock(...init) } } @@ -109,7 +110,7 @@ export function genChildren( if (childrenToGen.length) { for (const [child, from] of childrenToGen) { - push(...genChildren(child, context, from)) + push(...genChildren(child, context, pushBlock, from)) } } diff --git a/packages/compiler-vapor/src/generators/text.ts b/packages/compiler-vapor/src/generators/text.ts index 89e3167c664..700297b3b4f 100644 --- a/packages/compiler-vapor/src/generators/text.ts +++ b/packages/compiler-vapor/src/generators/text.ts @@ -10,8 +10,8 @@ export function genSetText( context: CodegenContext, ): CodeFragment[] { const { helper } = context - const { element, values, generated, jsx } = oper - const texts = combineValues(values, context, jsx) + const { element, values, generated } = oper + const texts = combineValues(values, context) return [ NEWLINE, ...genCall(helper('setText'), `${generated ? 'x' : 'n'}${element}`, texts), @@ -21,16 +21,15 @@ export function genSetText( function combineValues( values: SimpleExpressionNode[], context: CodegenContext, - jsx?: boolean, ): CodeFragment[] { return values.flatMap((value, i) => { let exp = genExpression(value, context) - if (!jsx && getLiteralExpressionValue(value) == null) { + if (getLiteralExpressionValue(value) == null) { // dynamic, wrap with toDisplayString exp = genCall(context.helper('toDisplayString'), exp) } if (i > 0) { - exp.unshift(jsx ? ', ' : ' + ') + exp.unshift(' + ') } return exp }) diff --git a/packages/compiler-vapor/src/generators/utils.ts b/packages/compiler-vapor/src/generators/utils.ts index 904b3dc87ca..46d8e52509a 100644 --- a/packages/compiler-vapor/src/generators/utils.ts +++ b/packages/compiler-vapor/src/generators/utils.ts @@ -10,6 +10,8 @@ import { import { isArray, isString } from '@vue/shared' import type { CodegenContext } from '../generate' +export { genExpression } from './expression' + export const NEWLINE: unique symbol = Symbol(__DEV__ ? `newline` : ``) /** increase offset but don't push actual code */ export const LF: unique symbol = Symbol(__DEV__ ? `line feed` : ``) diff --git a/packages/compiler-vapor/src/index.ts b/packages/compiler-vapor/src/index.ts index 6eda1021060..7594f835581 100644 --- a/packages/compiler-vapor/src/index.ts +++ b/packages/compiler-vapor/src/index.ts @@ -13,13 +13,7 @@ export { type CodegenOptions, type VaporCodegenResult, } from './generate' -export { - genCall, - genMulti, - buildCodeFragment, - codeFragmentToString, - type CodeFragment, -} from './generators/utils' +export * from './generators/utils' export { wrapTemplate, compile, diff --git a/packages/compiler-vapor/src/ir/index.ts b/packages/compiler-vapor/src/ir/index.ts index da636113224..88c235a0f49 100644 --- a/packages/compiler-vapor/src/ir/index.ts +++ b/packages/compiler-vapor/src/ir/index.ts @@ -124,7 +124,6 @@ export interface SetTextIRNode extends BaseIRNode { element: number values: SimpleExpressionNode[] generated?: boolean // whether this is a generated empty text node by `processTextLikeContainer` - jsx?: boolean } export type KeyOverride = [find: string, replacement: string] diff --git a/packages/compiler-vapor/src/transform.ts b/packages/compiler-vapor/src/transform.ts index 76563899d2b..b3963124e54 100644 --- a/packages/compiler-vapor/src/transform.ts +++ b/packages/compiler-vapor/src/transform.ts @@ -23,6 +23,7 @@ import { type IRSlots, type OperationNode, type RootIRNode, + type SetEventIRNode, type VaporDirectiveNode, } from './ir' import { isConstantExpression, isStaticExpression } from './utils' @@ -45,7 +46,7 @@ export interface DirectiveTransformResult { modifier?: '.' | '^' runtimeCamelize?: boolean handler?: boolean - handlerModifiers?: string[] + handlerModifiers?: SetEventIRNode['modifiers'] model?: boolean modelModifiers?: string[] } diff --git a/packages/compiler-vapor/src/transforms/vOn.ts b/packages/compiler-vapor/src/transforms/vOn.ts index fcbfc265d43..fe63ece0a88 100644 --- a/packages/compiler-vapor/src/transforms/vOn.ts +++ b/packages/compiler-vapor/src/transforms/vOn.ts @@ -65,7 +65,11 @@ export const transformVOn: DirectiveTransform = (dir, node, context) => { key: arg, value: handler, handler: true, - handlerModifiers: eventOptionModifiers, + handlerModifiers: { + keys: keyModifiers, + nonKeys: nonKeyModifiers, + options: eventOptionModifiers, + }, } } diff --git a/packages/runtime-core/__tests__/componentEmits.spec.ts b/packages/runtime-core/__tests__/componentEmits.spec.ts index dc82c04919f..c793433921a 100644 --- a/packages/runtime-core/__tests__/componentEmits.spec.ts +++ b/packages/runtime-core/__tests__/componentEmits.spec.ts @@ -325,7 +325,7 @@ describe('component: emit', () => { const Comp = () => h(Foo, { modelValue: null, - modelModifiers: { number: true }, + modelValueModifiers: { number: true }, 'onUpdate:modelValue': fn1, foo: null, @@ -356,7 +356,7 @@ describe('component: emit', () => { const Comp = () => h(Foo, { modelValue: null, - modelModifiers: { trim: true }, + modelValueModifiers: { trim: true }, 'onUpdate:modelValue': fn1, foo: null, @@ -410,7 +410,7 @@ describe('component: emit', () => { const Comp = () => h(Foo, { modelValue: null, - modelModifiers: { trim: true }, + modelValueModifiers: { trim: true }, 'onUpdate:modelValue': fn1, firstName: null, @@ -464,7 +464,7 @@ describe('component: emit', () => { const Comp = () => h(Foo, { modelValue: null, - modelModifiers: { trim: true, number: true }, + modelValueModifiers: { trim: true, number: true }, 'onUpdate:modelValue': fn1, foo: null, @@ -492,7 +492,7 @@ describe('component: emit', () => { const Comp = () => h(Foo, { modelValue: null, - modelModifiers: { trim: true }, + modelValueModifiers: { trim: true }, 'onUpdate:modelValue': fn, }) diff --git a/packages/runtime-core/src/helpers/useModel.ts b/packages/runtime-core/src/helpers/useModel.ts index e85edc6e9a7..603a7bf6ea8 100644 --- a/packages/runtime-core/src/helpers/useModel.ts +++ b/packages/runtime-core/src/helpers/useModel.ts @@ -145,9 +145,9 @@ export const getModelModifiers = ( modelName: string, getter: (props: Record, key: string) => any, ): Record | undefined => { - return modelName === 'modelValue' || modelName === 'model-value' - ? getter(props, 'modelModifiers') - : getter(props, `${modelName}Modifiers`) || - getter(props, `${camelize(modelName)}Modifiers`) || - getter(props, `${hyphenate(modelName)}Modifiers`) + return ( + getter(props, `${modelName}Modifiers`) || + getter(props, `${camelize(modelName)}Modifiers`) || + getter(props, `${hyphenate(modelName)}Modifiers`) + ) } diff --git a/packages/runtime-core/src/index.ts b/packages/runtime-core/src/index.ts index c7150e38e80..7daec103062 100644 --- a/packages/runtime-core/src/index.ts +++ b/packages/runtime-core/src/index.ts @@ -30,6 +30,8 @@ export { stop, getCurrentWatcher, onWatcherCleanup, + pauseTracking, + resetTracking, ReactiveEffect, // effect scope effectScope, diff --git a/packages/runtime-vapor/__tests__/componentAttrs.spec.ts b/packages/runtime-vapor/__tests__/componentAttrs.spec.ts index e4076855cb4..fc6fae5f021 100644 --- a/packages/runtime-vapor/__tests__/componentAttrs.spec.ts +++ b/packages/runtime-vapor/__tests__/componentAttrs.spec.ts @@ -322,4 +322,43 @@ describe('attribute fallthrough', () => { expect(el.getAttribute('aria-x')).toBe(parentVal.value) expect(el.getAttribute('aria-y')).toBe(parentVal.value) }) + + it('empty string should not be passed to classList.add', async () => { + const t0 = template('
', true /* root */) + const Child = defineVaporComponent({ + setup() { + const n = t0() as Element + renderEffect(() => { + setClass(n, { + foo: false, + }) + }) + return n + }, + }) + + const Parent = defineVaporComponent({ + setup() { + return createComponent( + Child, + { + class: () => ({ + bar: false, + }), + }, + null, + true, + ) + }, + }) + + const { host } = define({ + setup() { + return createComponent(Parent) + }, + }).render() + + const el = host.children[0] + expect(el.classList.length).toBe(0) + }) }) diff --git a/packages/runtime-vapor/__tests__/componentEmits.spec.ts b/packages/runtime-vapor/__tests__/componentEmits.spec.ts index 8c8a56085ba..6b542bbf6cc 100644 --- a/packages/runtime-vapor/__tests__/componentEmits.spec.ts +++ b/packages/runtime-vapor/__tests__/componentEmits.spec.ts @@ -265,7 +265,7 @@ describe('component: emit', () => { const fn2 = vi.fn() render({ modelValue: () => null, - modelModifiers: () => ({ number: true }), + modelValueModifiers: () => ({ number: true }), ['onUpdate:modelValue']: () => fn1, foo: () => null, fooModifiers: () => ({ number: true }), @@ -291,7 +291,7 @@ describe('component: emit', () => { modelValue() { return null }, - modelModifiers() { + modelValueModifiers() { return { trim: true } }, ['onUpdate:modelValue']() { @@ -327,7 +327,7 @@ describe('component: emit', () => { modelValue() { return null }, - modelModifiers() { + modelValueModifiers() { return { trim: true, number: true } }, ['onUpdate:modelValue']() { @@ -361,7 +361,7 @@ describe('component: emit', () => { modelValue() { return null }, - modelModifiers() { + modelValueModifiers() { return { trim: true } }, ['onUpdate:modelValue']() { diff --git a/packages/runtime-vapor/src/component.ts b/packages/runtime-vapor/src/component.ts index 548babebf8b..9d1beb7fb81 100644 --- a/packages/runtime-vapor/src/component.ts +++ b/packages/runtime-vapor/src/component.ts @@ -182,6 +182,14 @@ export function createComponent( appContext, ) + // HMR + if (__DEV__ && component.__hmrId) { + registerHMR(instance) + instance.isSingleRoot = isSingleRoot + instance.hmrRerender = hmrRerender.bind(null, instance) + instance.hmrReload = hmrReload.bind(null, instance) + } + if (__DEV__) { pushWarningContext(instance) startMeasure(instance, `init`) @@ -221,14 +229,6 @@ export function createComponent( // TODO make the proxy warn non-existent property access during dev instance.setupState = proxyRefs(setupResult) devRender(instance) - - // HMR - if (component.__hmrId) { - registerHMR(instance) - instance.isSingleRoot = isSingleRoot - instance.hmrRerender = hmrRerender.bind(null, instance) - instance.hmrReload = hmrReload.bind(null, instance) - } } } else { // component has a render function but no setup function @@ -283,18 +283,33 @@ export let isApplyingFallthroughProps = false */ export function devRender(instance: VaporComponentInstance): void { instance.block = - callWithErrorHandling( - instance.type.render!, - instance, - ErrorCodes.RENDER_FUNCTION, - [ - instance.setupState, - instance.props, - instance.emit, - instance.attrs, - instance.slots, - ], - ) || [] + (instance.type.render + ? callWithErrorHandling( + instance.type.render, + instance, + ErrorCodes.RENDER_FUNCTION, + [ + instance.setupState, + instance.props, + instance.emit, + instance.attrs, + instance.slots, + ], + ) + : callWithErrorHandling( + isFunction(instance.type) ? instance.type : instance.type.setup!, + instance, + ErrorCodes.SETUP_FUNCTION, + [ + instance.props, + { + slots: instance.slots, + attrs: instance.attrs, + emit: instance.emit, + expose: instance.expose, + }, + ], + )) || [] } const emptyContext: GenericAppContext = { diff --git a/packages/runtime-vapor/src/dom/prop.ts b/packages/runtime-vapor/src/dom/prop.ts index f464a2f6299..8c42ad766a5 100644 --- a/packages/runtime-vapor/src/dom/prop.ts +++ b/packages/runtime-vapor/src/dom/prop.ts @@ -122,7 +122,9 @@ function setClassIncremental(el: any, value: any): void { const prev = el[cacheKey] if ((value = el[cacheKey] = normalizeClass(value)) !== prev) { const nextList = value.split(/\s+/) - el.classList.add(...nextList) + if (value) { + el.classList.add(...nextList) + } if (prev) { for (const cls of prev.split(/\s+/)) { if (!nextList.includes(cls)) el.classList.remove(cls)