feat(sfc): add defineEmits and deprecate defineEmit (#3725)
This commit is contained in:
committed by
GitHub
parent
6b6d566861
commit
a137da8a9f
@@ -54,7 +54,7 @@ return { a }
|
||||
}"
|
||||
`;
|
||||
|
||||
exports[`SFC compile <script setup> defineEmit() 1`] = `
|
||||
exports[`SFC compile <script setup> defineEmits() 1`] = `
|
||||
"export default {
|
||||
expose: [],
|
||||
emits: ['foo', 'bar'],
|
||||
@@ -720,7 +720,7 @@ return { a, b, c, d, x }
|
||||
}"
|
||||
`;
|
||||
|
||||
exports[`SFC compile <script setup> with TypeScript defineEmit w/ type (type literal w/ call signatures) 1`] = `
|
||||
exports[`SFC compile <script setup> with TypeScript defineEmits w/ type (type literal w/ call signatures) 1`] = `
|
||||
"import { defineComponent as _defineComponent } from 'vue'
|
||||
|
||||
|
||||
@@ -741,7 +741,7 @@ return { emit }
|
||||
})"
|
||||
`;
|
||||
|
||||
exports[`SFC compile <script setup> with TypeScript defineEmit w/ type 1`] = `
|
||||
exports[`SFC compile <script setup> with TypeScript defineEmits w/ type 1`] = `
|
||||
"import { defineComponent as _defineComponent } from 'vue'
|
||||
|
||||
|
||||
|
||||
@@ -49,7 +49,7 @@ const bar = 1
|
||||
},`)
|
||||
})
|
||||
|
||||
test('defineEmit()', () => {
|
||||
test('defineEmit() (deprecated)', () => {
|
||||
const { content, bindings } = compile(`
|
||||
<script setup>
|
||||
import { defineEmit } from 'vue'
|
||||
@@ -61,7 +61,28 @@ const myEmit = defineEmit(['foo', 'bar'])
|
||||
myEmit: BindingTypes.SETUP_CONST
|
||||
})
|
||||
// should remove defineOptions import and call
|
||||
expect(content).not.toMatch('defineEmit')
|
||||
expect(content).not.toMatch(/defineEmits?/)
|
||||
// should generate correct setup signature
|
||||
expect(content).toMatch(`setup(__props, { emit: myEmit }) {`)
|
||||
// should include context options in default export
|
||||
expect(content).toMatch(`export default {
|
||||
expose: [],
|
||||
emits: ['foo', 'bar'],`)
|
||||
})
|
||||
|
||||
test('defineEmits()', () => {
|
||||
const { content, bindings } = compile(`
|
||||
<script setup>
|
||||
import { defineEmits } from 'vue'
|
||||
const myEmit = defineEmits(['foo', 'bar'])
|
||||
</script>
|
||||
`)
|
||||
assertCode(content)
|
||||
expect(bindings).toStrictEqual({
|
||||
myEmit: BindingTypes.SETUP_CONST
|
||||
})
|
||||
// should remove defineOptions import and call
|
||||
expect(content).not.toMatch('defineEmits')
|
||||
// should generate correct setup signature
|
||||
expect(content).toMatch(`setup(__props, { emit: myEmit }) {`)
|
||||
// should include context options in default export
|
||||
@@ -145,9 +166,9 @@ const myEmit = defineEmit(['foo', 'bar'])
|
||||
test('should allow defineProps/Emit at the start of imports', () => {
|
||||
assertCode(
|
||||
compile(`<script setup>
|
||||
import { defineProps, defineEmit, ref } from 'vue'
|
||||
import { defineProps, defineEmits, ref } from 'vue'
|
||||
defineProps(['foo'])
|
||||
defineEmit(['bar'])
|
||||
defineEmits(['bar'])
|
||||
const r = ref(0)
|
||||
</script>`).content
|
||||
)
|
||||
@@ -450,9 +471,9 @@ const myEmit = defineEmit(['foo', 'bar'])
|
||||
test('defineProps/Emit w/ runtime options', () => {
|
||||
const { content } = compile(`
|
||||
<script setup lang="ts">
|
||||
import { defineProps, defineEmit } from 'vue'
|
||||
import { defineProps, defineEmits } from 'vue'
|
||||
const props = defineProps({ foo: String })
|
||||
const emit = defineEmit(['a', 'b'])
|
||||
const emit = defineEmits(['a', 'b'])
|
||||
</script>
|
||||
`)
|
||||
assertCode(content)
|
||||
@@ -549,11 +570,11 @@ const emit = defineEmit(['a', 'b'])
|
||||
})
|
||||
})
|
||||
|
||||
test('defineEmit w/ type', () => {
|
||||
test('defineEmits w/ type', () => {
|
||||
const { content } = compile(`
|
||||
<script setup lang="ts">
|
||||
import { defineEmit } from 'vue'
|
||||
const emit = defineEmit<(e: 'foo' | 'bar') => void>()
|
||||
import { defineEmits } from 'vue'
|
||||
const emit = defineEmits<(e: 'foo' | 'bar') => void>()
|
||||
</script>
|
||||
`)
|
||||
assertCode(content)
|
||||
@@ -561,24 +582,24 @@ const emit = defineEmit(['a', 'b'])
|
||||
expect(content).toMatch(`emits: ["foo", "bar"] as unknown as undefined`)
|
||||
})
|
||||
|
||||
test('defineEmit w/ type (union)', () => {
|
||||
test('defineEmits w/ type (union)', () => {
|
||||
const type = `((e: 'foo' | 'bar') => void) | ((e: 'baz', id: number) => void)`
|
||||
expect(() =>
|
||||
compile(`
|
||||
<script setup lang="ts">
|
||||
import { defineEmit } from 'vue'
|
||||
const emit = defineEmit<${type}>()
|
||||
import { defineEmits } from 'vue'
|
||||
const emit = defineEmits<${type}>()
|
||||
</script>
|
||||
`)
|
||||
).toThrow()
|
||||
})
|
||||
|
||||
test('defineEmit w/ type (type literal w/ call signatures)', () => {
|
||||
test('defineEmits w/ type (type literal w/ call signatures)', () => {
|
||||
const type = `{(e: 'foo' | 'bar'): void; (e: 'baz', id: number): void;}`
|
||||
const { content } = compile(`
|
||||
<script setup lang="ts">
|
||||
import { defineEmit } from 'vue'
|
||||
const emit = defineEmit<${type}>()
|
||||
import { defineEmits } from 'vue'
|
||||
const emit = defineEmits<${type}>()
|
||||
</script>
|
||||
`)
|
||||
assertCode(content)
|
||||
@@ -906,8 +927,8 @@ const emit = defineEmit(['a', 'b'])
|
||||
|
||||
expect(() => {
|
||||
compile(`<script setup lang="ts">
|
||||
import { defineEmit } from 'vue'
|
||||
defineEmit<{}>({})
|
||||
import { defineEmits } from 'vue'
|
||||
defineEmits<{}>({})
|
||||
</script>`)
|
||||
}).toThrow(`cannot accept both type and non-type arguments`)
|
||||
})
|
||||
@@ -927,9 +948,9 @@ const emit = defineEmit(['a', 'b'])
|
||||
|
||||
expect(() =>
|
||||
compile(`<script setup>
|
||||
import { defineEmit } from 'vue'
|
||||
import { defineEmits } from 'vue'
|
||||
const bar = 'hello'
|
||||
defineEmit([bar])
|
||||
defineEmits([bar])
|
||||
</script>`)
|
||||
).toThrow(`cannot reference locally declared variables`)
|
||||
})
|
||||
@@ -947,9 +968,9 @@ const emit = defineEmit(['a', 'b'])
|
||||
|
||||
expect(() =>
|
||||
compile(`<script setup>
|
||||
import { defineEmit } from 'vue'
|
||||
import { defineEmits } from 'vue'
|
||||
ref: bar = 1
|
||||
defineEmit({
|
||||
defineEmits({
|
||||
bar
|
||||
})
|
||||
</script>`)
|
||||
@@ -959,14 +980,14 @@ const emit = defineEmit(['a', 'b'])
|
||||
test('should allow defineProps/Emit() referencing scope var', () => {
|
||||
assertCode(
|
||||
compile(`<script setup>
|
||||
import { defineProps, defineEmit } from 'vue'
|
||||
import { defineProps, defineEmits } from 'vue'
|
||||
const bar = 1
|
||||
defineProps({
|
||||
foo: {
|
||||
default: bar => bar + 1
|
||||
}
|
||||
})
|
||||
defineEmit({
|
||||
defineEmits({
|
||||
foo: bar => bar > 1
|
||||
})
|
||||
</script>`).content
|
||||
@@ -976,14 +997,14 @@ const emit = defineEmit(['a', 'b'])
|
||||
test('should allow defineProps/Emit() referencing imported binding', () => {
|
||||
assertCode(
|
||||
compile(`<script setup>
|
||||
import { defineProps, defineEmit } from 'vue'
|
||||
import { defineProps, defineEmits } from 'vue'
|
||||
import { bar } from './bar'
|
||||
defineProps({
|
||||
foo: {
|
||||
default: () => bar
|
||||
}
|
||||
})
|
||||
defineEmit({
|
||||
defineEmits({
|
||||
foo: () => bar > 1
|
||||
})
|
||||
</script>`).content
|
||||
|
||||
@@ -36,6 +36,7 @@ import { rewriteDefault } from './rewriteDefault'
|
||||
|
||||
const DEFINE_PROPS = 'defineProps'
|
||||
const DEFINE_EMIT = 'defineEmit'
|
||||
const DEFINE_EMITS = 'defineEmits'
|
||||
|
||||
export interface SFCScriptCompileOptions {
|
||||
/**
|
||||
@@ -286,10 +287,10 @@ export function compileScript(
|
||||
return false
|
||||
}
|
||||
|
||||
function processDefineEmit(node: Node): boolean {
|
||||
if (isCallOf(node, DEFINE_EMIT)) {
|
||||
function processDefineEmits(node: Node): boolean {
|
||||
if (isCallOf(node, DEFINE_EMIT) || isCallOf(node, DEFINE_EMITS)) {
|
||||
if (hasDefineEmitCall) {
|
||||
error(`duplicate ${DEFINE_EMIT}() call`, node)
|
||||
error(`duplicate ${DEFINE_EMITS}() call`, node)
|
||||
}
|
||||
hasDefineEmitCall = true
|
||||
emitRuntimeDecl = node.arguments[0]
|
||||
@@ -309,7 +310,7 @@ export function compileScript(
|
||||
emitTypeDecl = typeArg
|
||||
} else {
|
||||
error(
|
||||
`type argument passed to ${DEFINE_EMIT}() must be a function type ` +
|
||||
`type argument passed to ${DEFINE_EMITS}() must be a function type ` +
|
||||
`or a literal type with call signatures.`,
|
||||
typeArg
|
||||
)
|
||||
@@ -627,7 +628,9 @@ export function compileScript(
|
||||
const existing = userImports[local]
|
||||
if (
|
||||
source === 'vue' &&
|
||||
(imported === DEFINE_PROPS || imported === DEFINE_EMIT)
|
||||
(imported === DEFINE_PROPS ||
|
||||
imported === DEFINE_EMIT ||
|
||||
imported === DEFINE_EMITS)
|
||||
) {
|
||||
removeSpecifier(i)
|
||||
} else if (existing) {
|
||||
@@ -651,11 +654,11 @@ export function compileScript(
|
||||
}
|
||||
}
|
||||
|
||||
// process `defineProps` and `defineEmit` calls
|
||||
// process `defineProps` and `defineEmit(s)` calls
|
||||
if (
|
||||
node.type === 'ExpressionStatement' &&
|
||||
(processDefineProps(node.expression) ||
|
||||
processDefineEmit(node.expression))
|
||||
processDefineEmits(node.expression))
|
||||
) {
|
||||
s.remove(node.start! + startOffset, node.end! + startOffset)
|
||||
}
|
||||
@@ -669,14 +672,14 @@ export function compileScript(
|
||||
decl.id.end!
|
||||
)
|
||||
}
|
||||
const isDefineEmit = processDefineEmit(decl.init)
|
||||
if (isDefineEmit) {
|
||||
const isDefineEmits = processDefineEmits(decl.init)
|
||||
if (isDefineEmits) {
|
||||
emitIdentifier = scriptSetup.content.slice(
|
||||
decl.id.start!,
|
||||
decl.id.end!
|
||||
)
|
||||
}
|
||||
if (isDefineProps || isDefineEmit)
|
||||
if (isDefineProps || isDefineEmits)
|
||||
if (node.declarations.length === 1) {
|
||||
s.remove(node.start! + startOffset, node.end! + startOffset)
|
||||
} else {
|
||||
@@ -1040,7 +1043,9 @@ function walkDeclaration(
|
||||
for (const { id, init } of node.declarations) {
|
||||
const isDefineCall = !!(
|
||||
isConst &&
|
||||
(isCallOf(init, DEFINE_PROPS) || isCallOf(init, DEFINE_EMIT))
|
||||
(isCallOf(init, DEFINE_PROPS) ||
|
||||
isCallOf(init, DEFINE_EMIT) ||
|
||||
isCallOf(init, DEFINE_EMITS))
|
||||
)
|
||||
if (id.type === 'Identifier') {
|
||||
let bindingType
|
||||
|
||||
Reference in New Issue
Block a user