diff --git a/packages/runtime-core/__tests__/componentProps.spec.ts b/packages/runtime-core/__tests__/componentProps.spec.ts index b8eb0e47208..eb739b8ef5e 100644 --- a/packages/runtime-core/__tests__/componentProps.spec.ts +++ b/packages/runtime-core/__tests__/componentProps.spec.ts @@ -333,6 +333,41 @@ describe('component props', () => { }) }) + test('extendValidator custom warn message', async () => { + let warnMsg = '' + vi.spyOn(console, 'warn').mockImplementation(msg => { + warnMsg = msg + }) + const Comp = defineComponent({ + props: { + foo: { + type: Number, + extendValidator: (name, value, props, warn) => { + if (typeof value !== 'number') { + warn( + 'Invalid prop: custom validator check failed for prop "' + + name + + '".', + ) + } else if (!Number.isInteger(value)) { + warn(`Invalid prop: ${name}. Expected an integer.`) + } + }, + }, + bar: { + type: Number, + }, + }, + template: `
`, + }) + + // Note this one is using the main Vue render so it can compile template + // on the fly + const root = document.createElement('div') + domRender(h(Comp, { foo: 1.1, bar: 2 }), root) + expect(warnMsg).toMatch(`Invalid prop: foo. Expected an integer.`) + }) + //#12011 test('replace camelize with hyphenate to handle props key', () => { const Comp = { diff --git a/packages/runtime-core/src/componentProps.ts b/packages/runtime-core/src/componentProps.ts index 8baa7808665..8a6b082b766 100644 --- a/packages/runtime-core/src/componentProps.ts +++ b/packages/runtime-core/src/componentProps.ts @@ -25,6 +25,7 @@ import { toRawType, } from '@vue/shared' import { warn } from './warning' +import type { WarnFunction } from './warning' import { type ComponentInternalInstance, type ComponentOptions, @@ -57,6 +58,12 @@ export interface PropOptions