}"
`;
-exports[`SFC compile <script setup> defineEmit() 1`] = `
+exports[`SFC compile <script setup> defineEmits() 1`] = `
"export default {
expose: [],
emits: ['foo', 'bar'],
}"
`;
-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'
})"
`;
-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'
},`)
})
- test('defineEmit()', () => {
+ test('defineEmit() (deprecated)', () => {
const { content, bindings } = compile(`
<script setup>
import { defineEmit } from 'vue'
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
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
)
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)
})
})
- 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)
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)
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`)
})
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`)
})
expect(() =>
compile(`<script setup>
- import { defineEmit } from 'vue'
+ import { defineEmits } from 'vue'
ref: bar = 1
- defineEmit({
+ defineEmits({
bar
})
</script>`)
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
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
const DEFINE_PROPS = 'defineProps'
const DEFINE_EMIT = 'defineEmit'
+const DEFINE_EMITS = 'defineEmits'
export interface SFCScriptCompileOptions {
/**
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]
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
)
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) {
}
}
- // 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)
}
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 {
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
render,
SetupContext
} from '@vue/runtime-test'
-import { defineEmit, defineProps, useContext } from '../src/apiSetupHelpers'
+import { defineEmits, defineProps, useContext } from '../src/apiSetupHelpers'
describe('SFC <script setup> helpers', () => {
test('should warn runtime usage', () => {
defineProps()
expect(`defineProps() is a compiler-hint`).toHaveBeenWarned()
- defineEmit()
- expect(`defineEmit() is a compiler-hint`).toHaveBeenWarned()
+ defineEmits()
+ expect(`defineEmits() is a compiler-hint`).toHaveBeenWarned()
})
test('useContext (no args)', () => {
return null as any
}
-export function defineEmit<
+export function defineEmits<
TypeEmit = undefined,
E extends EmitsOptions = EmitsOptions,
EE extends string = string,
InferredEmit = EmitFn<E>
>(emitOptions?: E | EE[]): TypeEmit extends undefined ? InferredEmit : TypeEmit
// implementation
-export function defineEmit() {
+export function defineEmits() {
if (__DEV__) {
warn(
- `defineEmit() is a compiler-hint helper that is only usable inside ` +
+ `defineEmits() is a compiler-hint helper that is only usable inside ` +
`<script setup> of a single file component. Its arguments should be ` +
`compiled away and passing it at runtime has no effect.`
)
return null as any
}
+/**
+ * @deprecated use `defineEmits` instead.
+ */
+export const defineEmit = defineEmits
+
export function useContext(): SetupContext {
const i = getCurrentInstance()!
if (__DEV__ && !i) {
export { nextTick } from './scheduler'
export { defineComponent } from './apiDefineComponent'
export { defineAsyncComponent } from './apiAsyncComponent'
-export { defineProps, defineEmit, useContext } from './apiSetupHelpers'
+export {
+ defineProps,
+ defineEmits,
+ defineEmit,
+ useContext
+} from './apiSetupHelpers'
// Advanced API ----------------------------------------------------------------
</template>
<script setup lang="ts">
-import { ref, onMounted, defineProps, defineEmit, watchEffect } from 'vue'
+import { ref, onMounted, defineProps, defineEmits, watchEffect } from 'vue'
import { debounce } from '../utils'
import CodeMirror from './codemirror'
}
})
-const emit = defineEmit<(e: 'change', value: string) => void>()
+const emit = defineEmits<(e: 'change', value: string) => void>()
onMounted(() => {
const addonOptions = {
expectType,
defineProps,
defineEmit,
+ defineEmits,
useContext,
Slots,
describe
props2.baz
})
-describe('defineEmit w/ type declaration', () => {
- const emit = defineEmit<(e: 'change') => void>()
+describe('defineEmits w/ type declaration', () => {
+ const emit = defineEmits<(e: 'change') => void>()
emit('change')
// @ts-expect-error
emit()
emit('bar')
type Emits = { (e: 'foo' | 'bar'): void; (e: 'baz', id: number): void }
- const emit2 = defineEmit<Emits>()
+ const emit2 = defineEmits<Emits>()
emit2('foo')
emit2('bar')
emit2('baz')
})
-describe('defineEmit w/ runtime declaration', () => {
- const emit = defineEmit({
+describe('defineEmits w/ runtime declaration', () => {
+ const emit = defineEmits({
foo: () => {},
bar: null
})
// @ts-expect-error
emit('baz')
- const emit2 = defineEmit(['foo', 'bar'])
+ const emit2 = defineEmits(['foo', 'bar'])
emit2('foo')
emit2('bar', 123)
// @ts-expect-error
emit2('baz')
})
+describe('deprecated defineEmit', () => {
+ const emit = defineEmit({
+ foo: () => {},
+ bar: null
+ })
+ emit('foo')
+ emit('bar', 123)
+ // @ts-expect-error
+ emit('baz')
+})
+
describe('useContext', () => {
const { attrs, emit, slots } = useContext()
expectType<Record<string, unknown>>(attrs)