<script setup>
is a compile-time syntactic sugar for using Composition API inside Single-File Components (SFCs). It is the recommended syntax if you are using both SFCs and Composition API. It provides a number of advantages over the normal <script>
syntax:
To opt-in to the syntax, add the setup
attribute to the <script>
block:
<script setup> console.log('hello script setup') </script>
The code inside is compiled as the content of the component's setup()
function. This means that unlike normal <script>
, which only executes once when the component is first imported, code inside <script setup>
will execute every time an instance of the component is created.
When using <script setup>
, any top-level bindings (including variables, function declarations, and imports) declared inside <script setup>
are directly usable in the template:
<script setup> // variable const msg = 'Hello!' // functions function log() { console.log(msg) } </script> <template> <button @click="log">{{ msg }}</button> </template>
Imports are exposed in the same fashion. This means you can directly use an imported helper function in template expressions without having to expose it via the methods
option:
<script setup> import { capitalize } from './helpers' </script> <template> <div>{{ capitalize('hello') }}</div> </template>
Reactive state needs to be explicitly created using Reactivity APIs. Similar to values returned from a setup()
function, refs are automatically unwrapped when referenced in templates:
<script setup> import { ref } from 'vue' const count = ref(0) </script> <template> <button @click="count++">{{ count }}</button> </template>
Values in the scope of <script setup>
can also be used directly as custom component tag names:
<script setup> import MyComponent from './MyComponent.vue' </script> <template> <MyComponent /> </template>
Think of MyComponent
as being referenced as a variable. If you have used JSX, the mental model is similar here. The kebab-case equivalent <my-component>
also works in the template - however PascalCase component tags are strongly recommended for consistency. It also helps differentiating from native custom elements.
Since components are referenced as variables instead of registered under string keys, we should use dynamic :is
binding when using dynamic components inside <script setup>
:
<script setup> import Foo from './Foo.vue' import Bar from './Bar.vue' </script> <template> <component :is="Foo" /> <component :is="someCondition ? Foo : Bar" /> </template>
Note how the components can be used as variables in a ternary expression.
An SFC can implicitly refer to itself via its filename. E.g. a file named FooBar.vue
can refer to itself as <FooBar/>
in its template.
Note this has lower priority than imported components. If you have a named import that conflicts with the component's inferred name, you can alias the import:
import { FooBar as FooBarChild } from './components'
You can use component tags with dots like <Foo.Bar>
to refer to components nested under object properties. This is useful when you import multiple components from a single file:
<script setup> import * as Form from './form-components' </script> <template> <Form.Input> <Form.Label>label</Form.Label> </Form.Input> </template>
Globally registered custom directives just work as normal. Local custom directives don't need to be explicitly registered with <script setup>
, but they must follow the naming scheme vNameOfDirective
:
<script setup> const vMyDirective = { beforeMount: (el) => { // do something with the element } } </script> <template> <h1 v-my-directive>This is a Heading</h1> </template>
If you're importing a directive from elsewhere, it can be renamed to fit the required naming scheme:
<script setup> import { myDirective as vMyDirective } from './MyDirective.js' </script>
To declare options like props
and emits
with full type inference support, we can use the defineProps
and defineEmits
APIs, which are automatically available inside <script setup>
:
<script setup> const props = defineProps({ foo: String }) const emit = defineEmits(['change', 'delete']) // setup code </script>
defineProps
and defineEmits
are compiler macros only usable inside <script setup>
. They do not need to be imported, and are compiled away when <script setup>
is processed.
defineProps
accepts the same value as the props
option, while defineEmits
accepts the same value as the emits
option.
defineProps
and defineEmits
provide proper type inference based on the options passed.
The options passed to defineProps
and defineEmits
will be hoisted out of setup into module scope. Therefore, the options cannot reference local variables declared in setup scope. Doing so will result in a compile error. However, it can reference imported bindings since they are in the module scope as well.
Props and emits can also be declared using pure-type syntax by passing a literal type argument to defineProps
or defineEmits
:
const props = defineProps<{ foo: string bar?: number }>() const emit = defineEmits<{ (e: 'change', id: number): void (e: 'update', value: string): void }>() // 3.3+: alternative, more succinct syntax const emit = defineEmits<{ change: [id: number] // named tuple syntax update: [value: string] }>()
defineProps
or defineEmits
can only use either runtime declaration OR type declaration. Using both at the same time will result in a compile error.
When using type declaration, the equivalent runtime declaration is automatically generated from static analysis to remove the need for double declaration and still ensure correct runtime behavior.
In dev mode, the compiler will try to infer corresponding runtime validation from the types. For example here foo: String
is inferred from the foo: string
type. If the type is a reference to an imported type, the inferred result will be foo: null
(equal to any
type) since the compiler does not have information of external files.
In prod mode, the compiler will generate the array format declaration to reduce bundle size (the props here will be compiled into ['foo', 'bar']
)
In version 3.2 and below, the generic type parameter for defineProps()
were limited to a type literal or a reference to a local interface.
This limitation has been resolved in 3.3. The latest version of Vue supports referencing imported and a limited set of complex types in the type parameter position. However, because the type to runtime conversion is still AST-based, some complex types that require actual type analysis, e.g. conditional types, are not supported. You can use conditional types for the type of a single prop, but not the entire props object.
One drawback of the type-only defineProps
declaration is that it doesn't have a way to provide default values for the props. To resolve this problem, a withDefaults
compiler macro is also provided:
export interface Props { msg?: string labels?: string[] } const props = withDefaults(defineProps<Props>(), { msg: 'hello', labels: () => ['one', 'two'] })
This will be compiled to equivalent runtime props default
options. In addition, the withDefaults
helper provides type checks for the default values, and ensures the returned props
type has the optional flags removed for properties that do have default values declared.
This macro can be used to declare a two-way binding prop that can be consumed via v-model
from the parent component. Example usage is also discussed in the Component v-model
guide.
Under the hood, this macro declares a model prop and a corresponding value update event. If the first argument is a literal string, it will be used as the prop name; Otherwise the prop name will default to "modelValue"
. In both cases, you can also pass an additional object which can include the prop's options and the model ref's value transform options.
// declares "modelValue" prop, consumed by parent via v-model const model = defineModel() // OR: declares "modelValue" prop with options const model = defineModel({ type: String }) // emits "update:modelValue" when mutated model.value = 'hello' // declares "count" prop, consumed by parent via v-model:count const count = defineModel('count') // OR: declares "count" prop with options const count = defineModel('count', { type: Number, default: 0 }) function inc() { // emits "update:count" when mutated count.value++ }
To access modifiers used with the v-model
directive, we can destructure the return value of defineModel()
like this:
const [modelValue, modelModifiers] = defineModel() // corresponds to v-model.trim if (modelModifiers.trim) { // ... }
When a modifier is present, we likely need to transform the value when reading or syncing it back to the parent. We can achieve this by using the get
and set
transformer options:
const [modelValue, modelModifiers] = defineModel({ // get() omitted as it is not needed here set(value) { // if the .trim modifier is used, return trimmed value if (modelModifiers.trim) { return value.trim() } // otherwise, return the value as-is return value } })
Like defineProps
and defineEmits
, defineModel
can also receive type arguments to specify the types of the model value and the modifiers:
const modelValue = defineModel<string>() // ^? Ref<string | undefined> // default model with options, required removes possible undefined values const modelValue = defineModel<string>({ required: true }) // ^? Ref<string> const [modelValue, modifiers] = defineModel<string, 'trim' | 'uppercase'>() // ^? Record<'trim' | 'uppercase', true | undefined>
Components using <script setup>
are closed by default - i.e. the public instance of the component, which is retrieved via template refs or $parent
chains, will not expose any of the bindings declared inside <script setup>
.
To explicitly expose properties in a <script setup>
component, use the defineExpose
compiler macro:
<script setup> import { ref } from 'vue' const a = 1 const b = ref(2) defineExpose({ a, b }) </script>
When a parent gets an instance of this component via template refs, the retrieved instance will be of the shape { a: number, b: number }
(refs are automatically unwrapped just like on normal instances).
This macro can be used to declare component options directly inside <script setup>
without having to use a separate <script>
block:
<script setup> defineOptions({ inheritAttrs: false, customOptions: { /* ... */ } }) </script>
<script setup>
that are not literal constants.This macro can be used to provide type hints to IDEs for slot name and props type checking.
defineSlots()
only accepts a type parameter and no runtime arguments. The type parameter should be a type literal where the property key is the slot name, and the value type is the slot function. The first argument of the function is the props the slot expects to receive, and its type will be used for slot props in the template. The return type is currently ignored and can be any
, but we may leverage it for slot content checking in the future.
It also returns the slots
object, which is equivalent to the slots
object exposed on the setup context or returned by useSlots()
.
<script setup lang="ts"> const slots = defineSlots<{ default(props: { msg: string }): any }>() </script>
useSlots()
& useAttrs()
Usage of slots
and attrs
inside <script setup>
should be relatively rare, since you can access them directly as $slots
and $attrs
in the template. In the rare case where you do need them, use the useSlots
and useAttrs
helpers respectively:
<script setup> import { useSlots, useAttrs } from 'vue' const slots = useSlots() const attrs = useAttrs() </script>
useSlots
and useAttrs
are actual runtime functions that return the equivalent of setupContext.slots
and setupContext.attrs
. They can be used in normal composition API functions as well.
<script>
<script setup>
can be used alongside normal <script>
. A normal <script>
may be needed in cases where we need to:
<script setup>
, for example inheritAttrs
or custom options enabled via plugins (Can be replaced by defineOptions
in 3.3+).<script> // normal <script>, executed in module scope (only once) runSideEffectOnce() // declare additional options export default { inheritAttrs: false, customOptions: {} } </script> <script setup> // executed in setup() scope (for each instance) </script>
Support for combining <script setup>
and <script>
in the same component is limited to the scenarios described above. Specifically:
<script>
section for options that can already be defined using <script setup>
, such as props
and emits
.<script setup>
are not added as properties to the component instance, making them inaccessible from the Options API. Mixing APIs in this way is strongly discouraged.If you find yourself in one of the scenarios that is not supported then you should consider switching to an explicit setup()
function, instead of using <script setup>
.
await
Top-level await
can be used inside <script setup>
. The resulting code will be compiled as async setup()
:
<script setup> const post = await fetch(`/api/post/1`).then((r) => r.json()) </script>
In addition, the awaited expression will be automatically compiled in a format that preserves the current component instance context after the await
.
async setup()
must be used in combination with Suspense
, which is currently still an experimental feature. We plan to finalize and document it in a future release - but if you are curious now, you can refer to its tests to see how it works.
Generic type parameters can be declared using the generic
attribute on the <script>
tag:
<script setup lang="ts" generic="T"> defineProps<{ items: T[] selected: T }>() </script>
The value of generic
works exactly the same as the parameter list between <...>
in TypeScript. For example, you can use multiple parameters, extends
constraints, default types, and reference imported types:
<script setup lang="ts" generic="T extends string | number, U extends Item" > import type { Item } from './types' defineProps<{ id: T list: U[] }>() </script>
<script setup>
relies on the context of an SFC. When moved into external .js
or .ts
files, it may lead to confusion for both developers and tools. Therefore, <script setup>
cannot be used with the src
attribute.<script setup>
does not support In-DOM Root Component Template.(Related Discussion)
© 2013–present Yuxi Evan You
Licensed under the MIT License.
https://vuejs.org/api/sfc-script-setup