Skip to content

Commit 910fa76

Browse files
authoredNov 9, 2022
fix(compiler-sfc): resolve computed object key (#6963)
1 parent 7663a79 commit 910fa76

File tree

5 files changed

+79
-28
lines changed

5 files changed

+79
-28
lines changed
 

‎packages/compiler-sfc/__tests__/__snapshots__/compileScript.spec.ts.snap

+3-2
Original file line numberDiff line numberDiff line change
@@ -1743,12 +1743,13 @@ export default /*#__PURE__*/_defineComponent({
17431743
foo: { type: String, required: false, default: 'hi' },
17441744
bar: { type: Number, required: false },
17451745
baz: { type: Boolean, required: true },
1746-
qux: { type: Function, required: false, default() { return 1 } }
1746+
qux: { type: Function, required: false, default() { return 1 } },
1747+
quux: { type: Function, required: false, default() { } }
17471748
},
17481749
setup(__props: any, { expose }) {
17491750
expose();
17501751

1751-
const props = __props as { foo: string, bar?: number, baz: boolean, qux(): number };
1752+
const props = __props as { foo: string, bar?: number, baz: boolean, qux(): number, quux(): void };
17521753

17531754

17541755

‎packages/compiler-sfc/__tests__/__snapshots__/compileScriptPropsTransform.spec.ts.snap

+19
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,25 @@ return (_ctx, _cache) => {
5959
}"
6060
`;
6161

62+
exports[`sfc props transform computed static key 1`] = `
63+
"import { toDisplayString as _toDisplayString } from \\"vue\\"
64+
65+
66+
export default {
67+
props: ['foo'],
68+
setup(__props) {
69+
70+
71+
console.log(__props.foo)
72+
73+
return (_ctx, _cache) => {
74+
return _toDisplayString(__props.foo)
75+
}
76+
}
77+
78+
}"
79+
`;
80+
6281
exports[`sfc props transform default values w/ runtime declaration 1`] = `
6382
"import { mergeDefaults as _mergeDefaults } from 'vue'
6483

‎packages/compiler-sfc/__tests__/compileScript.spec.ts

+9-3
Original file line numberDiff line numberDiff line change
@@ -1039,10 +1039,12 @@ const emit = defineEmits(['a', 'b'])
10391039
foo?: string
10401040
bar?: number;
10411041
baz: boolean;
1042-
qux?(): number
1042+
qux?(): number;
1043+
quux?(): void
10431044
}>(), {
10441045
foo: 'hi',
1045-
qux() { return 1 }
1046+
qux() { return 1 },
1047+
['quux']() { }
10461048
})
10471049
</script>
10481050
`)
@@ -1056,14 +1058,18 @@ const emit = defineEmits(['a', 'b'])
10561058
`qux: { type: Function, required: false, default() { return 1 } }`
10571059
)
10581060
expect(content).toMatch(
1059-
`{ foo: string, bar?: number, baz: boolean, qux(): number }`
1061+
`quux: { type: Function, required: false, default() { } }`
1062+
)
1063+
expect(content).toMatch(
1064+
`{ foo: string, bar?: number, baz: boolean, qux(): number, quux(): void }`
10601065
)
10611066
expect(content).toMatch(`const props = __props`)
10621067
expect(bindings).toStrictEqual({
10631068
foo: BindingTypes.PROPS,
10641069
bar: BindingTypes.PROPS,
10651070
baz: BindingTypes.PROPS,
10661071
qux: BindingTypes.PROPS,
1072+
quux: BindingTypes.PROPS,
10671073
props: BindingTypes.SETUP_CONST
10681074
})
10691075
})

‎packages/compiler-sfc/__tests__/compileScriptPropsTransform.spec.ts

+18
Original file line numberDiff line numberDiff line change
@@ -184,6 +184,24 @@ describe('sfc props transform', () => {
184184
assertCode(content)
185185
})
186186

187+
// #6960
188+
test('computed static key', () => {
189+
const { content, bindings } = compile(`
190+
<script setup>
191+
const { ['foo']: foo } = defineProps(['foo'])
192+
console.log(foo)
193+
</script>
194+
<template>{{ foo }}</template>
195+
`)
196+
expect(content).not.toMatch(`const { foo } =`)
197+
expect(content).toMatch(`console.log(__props.foo)`)
198+
expect(content).toMatch(`_toDisplayString(__props.foo)`)
199+
assertCode(content)
200+
expect(bindings).toStrictEqual({
201+
foo: BindingTypes.PROPS
202+
})
203+
})
204+
187205
describe('errors', () => {
188206
test('should error on deep destructure', () => {
189207
expect(() =>

‎packages/compiler-sfc/src/compileScript.ts

+30-23
Original file line numberDiff line numberDiff line change
@@ -447,18 +447,15 @@ export function compileScript(
447447
// props destructure - handle compilation sugar
448448
for (const prop of declId.properties) {
449449
if (prop.type === 'ObjectProperty') {
450-
if (prop.computed) {
450+
const propKey = resolveObjectKey(prop.key, prop.computed)
451+
452+
if (!propKey) {
451453
error(
452454
`${DEFINE_PROPS}() destructure cannot use computed key.`,
453455
prop.key
454456
)
455457
}
456458

457-
const propKey =
458-
prop.key.type === 'StringLiteral'
459-
? prop.key.value
460-
: (prop.key as Identifier).name
461-
462459
if (prop.value.type === 'AssignmentPattern') {
463460
// default value { foo = 123 }
464461
const { left, right } = prop.value
@@ -774,7 +771,8 @@ export function compileScript(
774771
propsRuntimeDefaults.type === 'ObjectExpression' &&
775772
propsRuntimeDefaults.properties.every(
776773
node =>
777-
(node.type === 'ObjectProperty' && !node.computed) ||
774+
(node.type === 'ObjectProperty' &&
775+
(!node.computed || node.key.type.endsWith('Literal'))) ||
778776
node.type === 'ObjectMethod'
779777
)
780778
)
@@ -795,9 +793,10 @@ export function compileScript(
795793
if (destructured) {
796794
defaultString = `default: ${destructured}`
797795
} else if (hasStaticDefaults) {
798-
const prop = propsRuntimeDefaults!.properties.find(
799-
(node: any) => node.key.name === key
800-
) as ObjectProperty | ObjectMethod
796+
const prop = propsRuntimeDefaults!.properties.find(node => {
797+
if (node.type === 'SpreadElement') return false
798+
return resolveObjectKey(node.key, node.computed) === key
799+
}) as ObjectProperty | ObjectMethod
801800
if (prop) {
802801
if (prop.type === 'ObjectProperty') {
803802
// prop has corresponding static default value
@@ -874,9 +873,13 @@ export function compileScript(
874873
m.key.type === 'Identifier'
875874
) {
876875
if (
877-
propsRuntimeDefaults!.properties.some(
878-
(p: any) => p.key.name === (m.key as Identifier).name
879-
)
876+
propsRuntimeDefaults!.properties.some(p => {
877+
if (p.type === 'SpreadElement') return false
878+
return (
879+
resolveObjectKey(p.key, p.computed) ===
880+
(m.key as Identifier).name
881+
)
882+
})
880883
) {
881884
res +=
882885
m.key.name +
@@ -2139,16 +2142,9 @@ function analyzeBindingsFromOptions(node: ObjectExpression): BindingMetadata {
21392142
function getObjectExpressionKeys(node: ObjectExpression): string[] {
21402143
const keys = []
21412144
for (const prop of node.properties) {
2142-
if (
2143-
(prop.type === 'ObjectProperty' || prop.type === 'ObjectMethod') &&
2144-
!prop.computed
2145-
) {
2146-
if (prop.key.type === 'Identifier') {
2147-
keys.push(prop.key.name)
2148-
} else if (prop.key.type === 'StringLiteral') {
2149-
keys.push(prop.key.value)
2150-
}
2151-
}
2145+
if (prop.type === 'SpreadElement') continue
2146+
const key = resolveObjectKey(prop.key, prop.computed)
2147+
if (key) keys.push(String(key))
21522148
}
21532149
return keys
21542150
}
@@ -2297,3 +2293,14 @@ export function hmrShouldReload(
22972293

22982294
return false
22992295
}
2296+
2297+
export function resolveObjectKey(node: Node, computed: boolean) {
2298+
switch (node.type) {
2299+
case 'StringLiteral':
2300+
case 'NumericLiteral':
2301+
return node.value
2302+
case 'Identifier':
2303+
if (!computed) return node.name
2304+
}
2305+
return undefined
2306+
}

0 commit comments

Comments
 (0)
Please sign in to comment.