If you are starting using Vue, learning Vue in plain old JavaScript first is recommanded, then come back here.
Also, don't use it:
- for small or prototype applications.
- for teams that don't want to use it.
You will need to install:
And you need to disable TypeScript and JavaScript Language Features
for the workspace only by searching the extension for @builtin typescript
. Reload the VSC window to finish.
It is as simple as telling the script tag the code is TypeScript with the lang
attribut equal to ts
:
<script setup lang="ts">
// your code
</script>
As I explained in Event definition, using defineEmits
allows to get autocomplete in the IDE.
On the parent component, as you type @
, you will see the event name as you defined it in the child component.
From there, to type the event requires a little extra syntax:
defineEmits<{
(
//name of event
//adding the "@" helps to identify native event (one "@") and custom events (two "@").
event: "@add-entry",
//payload definition
entry: { entryMessage: string; emoji: Emoji | null }
): void; //return type is always void
}>();
IMPORTANT: In the naming of the event, always use kebab-case as above if you want autocomplete on the parent component. I've asked the VueSchool community why in a comment of the lesson.
OK, I tried this one alone, but it wasn't easy.
But, it is actually as easy as the defineEmits
. Using the Type-based decoration is recommended in TypeScript project instead of the runtime decoration.
const props = defineProps<{
entry: JournalEntry;
}>();
You could also do it this way:
interface Props {
entry: JournalEntry;
}
const props = defineProps<Props>();
IMPORTANT: you cannot use the different style decoration in a same component.
If you need defaults, you will need to wrap defineProps
in withDefaults
.
import type PropsAppLoginForm from "@/types/PropsAppLoginForm";
const props = withDefaults(defineProps<PropsAppLoginForm>(), {
enableRegister: true,
});
Template ref is a way to access, when necessary, a DOM Element to work on in the script setup.
For example: let's say you have a textarea input where you want to focus on as a page loads.
You can achieve this using the following technique:
<script lang="ts" setup>
import { computed, onMounted, ref } from "vue";
//type must be null as default as the onMounted hook is called when the DOM is loaded.
const textareaElement = ref<HTMLTextAreaElement | null>(null);
//hooks
onMounted(() => textareaElement.value?.focus());
</script>
<template>
<form class="entry-form" @submit.prevent="handleSubmit">
<textarea
v-model="body"
:maxlength="MAX_CHARS"
ref="textareaElement"
placeholder="New Journal Entry for danielkelly_io"
></textarea>
<div class="entry-form-footer">
<button>Save</button>
</div>
</form>
</template>
This is also a method I used in the implementation of a native dialog modal.
Read more in the docs about typing the template refs.
Read more in the docs about the Template refs, what they are and are used for.
Similar to the JavaScript version, the difference is found in the use of the type InjectionKey
from vue
:
Injection keys need to be store in a seperate file that is imported when needed.
For example, a user object to pass on to childs of App.vue
:
- we create the file containing the inject keys:
import type { InjectionKey } from "vue";
import User from "./types/User";
//create a unique InjectionKey since provide requires that.
// Using a Symbol guaranteed the unicity. See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Symbol
export const userInjectionKey = Symbol() as InjectionKey<User>;
- we provide the value in
App.vue
:
import { provide } from "vue";
import { userInjectionKey } from "@/injectKeys";
provide(userInjectionKey, user);
- we use it in any component of the application:
import { inject } from "vue";
import { userInjectionKey } from "@/injectKeys";
const user = inject(userInjectionKey);
NB: by default, TypeScript knows the injected object could be null. So when you use in the template of the component using the value, think about it.
<span>{{ user?.username || "Anonymous" }}</span>
You will need to user defineComponent
and it must be imported, it is not a macro!
It is used to infer types for the component's options.
Also, as noted previously, we cannot use type interface on a runtime decoration style.
For example, in the props definition, if a prop is an object, you will need to:
- define the prop as
Object
, - use the
PropType
utility to infer the type.
import { defineComponent } from "vue";
import type { PropType } from "vue";
export default defineComponent({
props: {
date: { type: Object as PropType<MyType>, required: true },
},
});
For computed, using the function type return can add type-safety.
Using the ComponentCustomProperties
interface, we can extend vue, if we needed.
It requires to create a file at the root. For example:
// File: vue-global-props.d.ts
import axios from "axios";
//This extends the vue api
declare module "vue" {
interface ComponentCustomProperties {
$axios: typeof axios;
}
}
Then, you need to tell TypeScript the file exists:
{
"include": [
"env.d.ts",
"src/**/*",
"src/**/*.vue",
"src/**/*.ts",
"vue-global-props.d.ts"
]
}
Then, in a component, the usage would be:
export default defineComponent({
mounted() {
this.$axios("https://jsonplaceholder.typicode.com/todos/1");
},
});
This works only in the Options API. In the Composition API, using composables is the way to go.
See this article or that one.