Skip to content

Commit 81e1e47

Browse files
committed
fix: respect type order when boolean casting multi-typed props
fix #7485
1 parent a2cd412 commit 81e1e47

File tree

2 files changed

+38
-12
lines changed

2 files changed

+38
-12
lines changed

src/core/util/props.js

+21-12
Original file line numberDiff line numberDiff line change
@@ -27,12 +27,18 @@ export function validateProp (
2727
const prop = propOptions[key]
2828
const absent = !hasOwn(propsData, key)
2929
let value = propsData[key]
30-
// handle boolean props
31-
if (isType(Boolean, prop.type)) {
30+
// boolean casting
31+
const booleanIndex = getTypeIndex(Boolean, prop.type)
32+
if (booleanIndex > -1) {
3233
if (absent && !hasOwn(prop, 'default')) {
3334
value = false
34-
} else if (!isType(String, prop.type) && (value === '' || value === hyphenate(key))) {
35-
value = true
35+
} else if (value === '' || value === hyphenate(key)) {
36+
// only cast empty string / same name to boolean if
37+
// boolean has higher priority
38+
const stringIndex = getTypeIndex(String, prop.type)
39+
if (stringIndex < 0 || booleanIndex < stringIndex) {
40+
value = true
41+
}
3642
}
3743
}
3844
// check default value
@@ -179,15 +185,18 @@ function getType (fn) {
179185
return match ? match[1] : ''
180186
}
181187

182-
function isType (type, fn) {
183-
if (!Array.isArray(fn)) {
184-
return getType(fn) === getType(type)
188+
function isSameType (a, b) {
189+
return getType(a) === getType(b)
190+
}
191+
192+
function getTypeIndex (type, expectedTypes): number {
193+
if (!Array.isArray(expectedTypes)) {
194+
return isSameType(expectedTypes, type) ? 0 : -1
185195
}
186-
for (let i = 0, len = fn.length; i < len; i++) {
187-
if (getType(fn[i]) === getType(type)) {
188-
return true
196+
for (let i = 0, len = expectedTypes.length; i < len; i++) {
197+
if (isSameType(expectedTypes[i], type)) {
198+
return i
189199
}
190200
}
191-
/* istanbul ignore next */
192-
return false
201+
return -1
193202
}

test/unit/features/options/props.spec.js

+17
Original file line numberDiff line numberDiff line change
@@ -512,4 +512,21 @@ describe('Options props', () => {
512512
expect(`"${attr}" is a reserved attribute`).toHaveBeenWarned()
513513
})
514514
})
515+
516+
it('should consider order when casting [Boolean, String] multi-type props', () => {
517+
const vm = new Vue({
518+
template: '<test ref="test" booleanOrString stringOrBoolean />',
519+
components: {
520+
test: {
521+
template: '<div></div>',
522+
props: {
523+
booleanOrString: [Boolean, String],
524+
stringOrBoolean: [String, Boolean]
525+
}
526+
}
527+
}
528+
}).$mount()
529+
expect(vm.$refs.test.$props.booleanOrString).toBe(true)
530+
expect(vm.$refs.test.$props.stringOrBoolean).toBe('')
531+
})
515532
})

0 commit comments

Comments
 (0)