DEV Community

Dany the spacecowboy
Dany the spacecowboy

Posted on

From Vue2 Master to Vue3 Noob

you can check the original post on my blog in both english and spanish

so I used to tell myself I was good at building webapps with vue, but then the vue3 nation attacked (yes, an avatar reference).
with this introduction to vue3, a lot of things changed, I like to think about it as the react hooks change but for vue, the main thing I consider that is a major and it's worth getting used to it and learn is the Composition API which I'll be talking more in detail while creating a simple webapp to display some pokemon data

Setup()

Vue3 introduces a new component option called setup, which is the entrypoint for the composition API, it's important to know that the code written here is executed before the creation of the component therefore the data, computed, and method options aren't available inside of it.
The setup component options contains two parameters

  • props
  • context (which contains attrs, slots, emit)

In this example I'll be getting the pokemonId prop or if it doesn't exist I'll be assigning one randomly,
then I'll call the API to get the pokemon sprite, the return of the setup options allows you to use those values on the template

setup(props) {
    const pokemonId = props.pokemonId || Math.ceil(Math.random() * 500)
    const pokemonSprite = getPokemonData(pokemonId); //magic function that returns a sprite
    return {
        pokemonId,
        pokemonSprite
    }
}
Enter fullscreen mode Exit fullscreen mode

after this we can just do this inside the template to show the sprite image:

<template>
    <img :src="pokemonSprite" alt="pokemon sprite" />
</template>
Enter fullscreen mode Exit fullscreen mode

Ref() and Reactive()

Vue3 by default doesn't make every variable reactive, you have to define it yourself using the new
ref(), reactive() methods

these methods are really similar, the main difference is that is recommended to use ref() on data types (ints, bools, strings, etc..) and to use reactive() on data structures (arrays, sets, json, etc...)

for example, if we want to add a button to randomize a pokemon sprite on click, we need to make the pokemonId prop reactive:

import { ref } from 'vue'
setup(props) {
    let pokemon = ref({})
    const getPokemonData = async (pokemonId) => {
        const response = await fetch(`https://pokeapi.co/api/v2/pokemon/${pokemonId}`)
        pokemon.value = await response.json();
    }
    getPokemonData(props.pokemonId);
    return {
        pokemon,
        getPokemonData
    }
},
methods: {
    randomize() {
        this.getPokemonData(Math.ceil(Math.random() * 500));
    }
}
Enter fullscreen mode Exit fullscreen mode

and then in our template we can just do this:

<template>
    <div v-if="pokemon.sprites">
        <img :src="pokemon.sprites.front_default" alt="pokemon sprite" />
        <button @click="randomize">Randomize</button>
    </div>
</template>
Enter fullscreen mode Exit fullscreen mode

Lifecycle hooks

Another new addition to vue3, is the ability to call lifecycle hooks only inside the setup component option, replacing all the other components options we used to have(mounted {}, created{}, etc..) with the new variants (onMounted(), onCreated(), etc..)
this makes the code more readable since in the setup, you'll be declaring all the initialization of your component in a single place

we can change our pokemon randomizer code to make use of the onMounted hook inside the setup

import { ref, onMounted } from 'vue'
setup(props) {
    let pokemon = ref({})
    const getPokemonData = async (pokemonId) => {
        const response = await fetch(`https://pokeapi.co/api/v2/pokemon/${pokemonId}`)
        pokemon.value = await response.json();
    }
    onMounted(getPokemonData(props.pokemonId));
    return {
        pokemon,
        getPokemonData
    }
},
Enter fullscreen mode Exit fullscreen mode

Script setup

Now that I understand how the setup worked, I wanted to go even further explorating the <script setup> proposal
after playing with it for a couple of minutes, I understood how awesome it is! the time saving and less code written are really good benefits of it, it adds a couple of new different syntax for props, but nothing to extreme,
this is how the pokemon randomizer code looks like with the script setup:

<script setup>
import { ref, onMounted } from 'vue'
const props = defineProps({
  pokemonId: Number,
});
let pokemon = ref({})
const getPokemonData = async (pokemonId) => {
    const response = await fetch(`https://pokeapi.co/api/v2/pokemon/${pokemonId}`)
    pokemon.value = await response.json();
}
const randomize = () =>  {
    getPokemonData(Math.ceil(Math.random() * 500));
}
onMounted(getPokemonData(props.pokemonId));
Enter fullscreen mode Exit fullscreen mode

you can check the source code in here

Sources

Top comments (0)