Common Mistakes in
(And, more importantly... how to avoid them)
data:image/s3,"s3://crabby-images/2c8c3/2c8c32a8bb44ff4256dc230e34efdea9573c1618" alt=""
ue.js
data:image/s3,"s3://crabby-images/719a4/719a4e3185076848f779b0a9eef8313b6ec08b89" alt=""
Daniel Kelly
Lead Instructor @ Vue School
Full Stack developer (10 + years)
Husband and Father
presentation Icons from fontawesome.com
Why this topic
?
- We're all human and make mistakes
- Simple mistakes can cause big headaches
- Learn from the mistakes of others
- It's a very practical topic
Why this topic?
?
Mistake # 1
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
index as :key
<li
v-for="(item, index) in items"
:key="index"
>...</li>
What's wrong with this? 🤔
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
index as :key
<li
v-for="(item, index) in items"
:key="index"
>...</li>
What's wrong with this? 🤔
❌
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
index as :key
Bug Manifests Under 2 Conditions
- When the elements in the array change order
- When a nested element or component has it's own state
It's tricky!
data:image/s3,"s3://crabby-images/239a1/239a1072f3e5df2082312962de8fbaa040fb68a6" alt=""
Let's look at an example!
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
index as :key
It's not just for loops!
Let's look at an example!
Mistake # 2
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Non-reactive value as dep
const cookiesAccepted = computed(()=>
localStorage.getItem("cookiesAccepted")
)
What's wrong with this? 🤔
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Non-reactive value as dep
const cookiesAccepted = computed(()=>
localStorage.getItem("cookiesAccepted")
)
What's wrong with this? 🤔
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Non-reactive value as dep
const cookiesAccepted = computed(()=>
localStorage.getItem("cookiesAccepted")
) // 0
localStorage.setItem("cookiesAccepted", 1)
cookiesAccepted.value // 0
Changing the value of local storage doesn't update the value of the computed prop.
const accepted = 0;
const cookiesAccepted = computed(()=> accepted)
This is more obvious
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Non-reactive value as dep
Often see this with browser API's:
- localStorage
- Date()
- DOM element properties
(like clientWidth, etc) - etc
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Non-reactive value as dep
And, of course, usually these hide in more complex computed props
(nothing as obvious as our simple example)
What's the solution?
🤔
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Reactive Deps
VueUse
VueUse
handy reactive wrapper around browser APIs
Let's look at an example!
Mistake # 3
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Replacing Reactive State
let posts = reactive([
'Post 1',
'Post 2'
]);
function loadMore(){
// fetch more posts, etc
posts = newPosts;
}
What's wrong
with this? 🤔
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Replacing Reactive State
let posts = reactive([
'Post 1',
'Post 2'
]);
function loadMore(){
// fetch more posts, etc
posts = newPosts;
}
What's wrong
with this? 🤔
👈 replacing reactive
kills reactivity
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Replacing Reactive State
data:image/s3,"s3://crabby-images/441fa/441faf702b5978dd0faaaa73ba10a1aa0bc3dbd7" alt=""
data:image/s3,"s3://crabby-images/441fa/441faf702b5978dd0faaaa73ba10a1aa0bc3dbd7" alt=""
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Replacing Reactive State
data:image/s3,"s3://crabby-images/441fa/441faf702b5978dd0faaaa73ba10a1aa0bc3dbd7" alt=""
data:image/s3,"s3://crabby-images/441fa/441faf702b5978dd0faaaa73ba10a1aa0bc3dbd7" alt=""
Let's look at some examples!
Moral of the story:
just use ref() everywhere!
I'm not the only one,
it's the recommendation of the official Vue.js docs
data:image/s3,"s3://crabby-images/0fd23/0fd237040854e9b1f28191eb2c255b9e6e4453a8" alt=""
data:image/s3,"s3://crabby-images/dc0a1/dc0a10f22ba97aefa9db3b30f012bab0166258db" alt=""
I'm not the only one,
it's the recommendation of the official Vue.js docs
data:image/s3,"s3://crabby-images/0fd23/0fd237040854e9b1f28191eb2c255b9e6e4453a8" alt=""
data:image/s3,"s3://crabby-images/dc0a1/dc0a10f22ba97aefa9db3b30f012bab0166258db" alt=""
Oh and one more reason to use ref() everywhere
default watcher depth is different between ref() and reactive()
Let's see an example of that
Mistake # 4
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Mutating Props
<script setup lang="ts">
defineProps<{
modelValue: string;
}>();
</script>
<template>
<input id="username" v-model="modelValue" />
</template>
What's wrong with this? 🤔
data:image/s3,"s3://crabby-images/bf5db/bf5db9c7bfb1dfd8a3c9de458873ffd0183d2a09" alt=""
Mutating Props
<script setup lang="ts">
defineProps<{
modelValue: string;
}>();
</script>
<template>
<input id="username" v-model="modelValue" />
</template>
What's wrong with this? 🤔
Mutating a prop
Let's look at some examples!
Common Vue.js Mistakes and How to Avoid Them
data:image/s3,"s3://crabby-images/04012/04012e71465861c590ae65616107a03aea8e4db4" alt=""
🔥 Free
https://vueschool.io/courses/common-vue-js-mistakes-and-how-to-avoid-them
- Omitting the Key Directive on v-for
- Prop Drilling
- Watching Arrays the Wrong Way
- Replacing Reactive State the Wrong Way
- Unintentionally Mutating Props
- Forgetting to Clean Up Your Manual Event Listeners
- Expecting Changes to Non-Reactive Dependencies to Trigger Updates
- Not Considering TypeScript
- Destructuring Reactive Data
- Calling Composables in the Wrong Place
- Using v-html with User Provided Data
- Unnecessary Manual DOM Manipulation
👈
👈
👈
👈
data:image/s3,"s3://crabby-images/57b72/57b729bad7d38e4b80fe18de3bd04731bd44f4a6" alt=""
But before I go!
certificates.dev
data:image/s3,"s3://crabby-images/a39d6/a39d6bebbce8478eea78715996eca622b263cb47" alt=""
83% say the exam is moderate to difficult
74% say it actually helped improve their Vue.js skills
data:image/s3,"s3://crabby-images/a1f7f/a1f7f670cc289340cfea37b41c729293188bd369" alt=""
data:image/s3,"s3://crabby-images/d6d53/d6d531383c0f4b76e2f452b8c725d179e5dfd51a" alt=""
data:image/s3,"s3://crabby-images/59104/5910496504b1e674ff468ddcd5d3090b7322d9fb" alt=""
data:image/s3,"s3://crabby-images/fa88f/fa88ffb1517e1c6e03cb6cbe347f4fe47cea1bf4" alt=""
data:image/s3,"s3://crabby-images/3b2ca/3b2ca3bb669f82bbf602e7d0fd629fca98d905ff" alt=""
data:image/s3,"s3://crabby-images/e5078/e5078a16638f8f9733e2c7d563434c60a650df14" alt=""
Developer Experience
Your Path to a Dynamic and Engaging
certificates.dev
data:image/s3,"s3://crabby-images/76733/76733f20b1c07ada7ec4fb1c83b8256c9ff3ab46" alt=""
data:image/s3,"s3://crabby-images/97908/97908c701557d52e9796b90dee1a1cb87b6608d2" alt=""
frontendnation.com
60+
Speakers
vue.school/masterclass
data:image/s3,"s3://crabby-images/0d427/0d4277f57a8619bc95cbf3df92607ae670a546d2" alt=""
data:image/s3,"s3://crabby-images/b86ba/b86baac0efa582ea32bf7d3d3aa2f17fda664140" alt=""
data:image/s3,"s3://crabby-images/36f7b/36f7b3c65d70062e48a9eff725f2199dac0b8de3" alt=""
Not just crappy multiple choice questions
👉 you gotta code to pass
data:image/s3,"s3://crabby-images/9b0a1/9b0a1c86e0d3dc17d91b4e71537c98fd2760abac" alt=""
data:image/s3,"s3://crabby-images/9b0a1/9b0a1c86e0d3dc17d91b4e71537c98fd2760abac" alt=""
data:image/s3,"s3://crabby-images/9b0a1/9b0a1c86e0d3dc17d91b4e71537c98fd2760abac" alt=""
data:image/s3,"s3://crabby-images/9b0a1/9b0a1c86e0d3dc17d91b4e71537c98fd2760abac" alt=""
data:image/s3,"s3://crabby-images/aea6e/aea6e2f51dd4bde7d559cfe18dc8c26d260b3353" alt=""
data:image/s3,"s3://crabby-images/9b0a1/9b0a1c86e0d3dc17d91b4e71537c98fd2760abac" alt=""
data:image/s3,"s3://crabby-images/f0242/f0242b86babb728f9e487fa2867334ab83e1c0ca" alt=""
data:image/s3,"s3://crabby-images/6e0ea/6e0ea987d42779b55b344931818a626adc3a59c1" alt=""
data:image/s3,"s3://crabby-images/aea6e/aea6e2f51dd4bde7d559cfe18dc8c26d260b3353" alt=""
data:image/s3,"s3://crabby-images/9b0a1/9b0a1c86e0d3dc17d91b4e71537c98fd2760abac" alt=""
data:image/s3,"s3://crabby-images/f0242/f0242b86babb728f9e487fa2867334ab83e1c0ca" alt=""
data:image/s3,"s3://crabby-images/aea6e/aea6e2f51dd4bde7d559cfe18dc8c26d260b3353" alt=""
Grab Some Swag 🤗
data:image/s3,"s3://crabby-images/91fc7/91fc7c9da1766f2a8fe13bee0d59d47d8e6c08fc" alt=""
data:image/s3,"s3://crabby-images/84a67/84a67719ac0acfb2d1304754074b522e1985f1bf" alt=""
data:image/s3,"s3://crabby-images/620bc/620bc3fe9564ac84d0416ddd86dd5df2fcf916a6" alt=""
data:image/s3,"s3://crabby-images/ab42d/ab42debb40fffe2bfc8832d74146ff0b843efa2f" alt=""
data:image/s3,"s3://crabby-images/d1876/d1876ebcde5d1192424a6d0fd4081ae20aa549df" alt=""
danielkelly_io
I'll tweet out the slides and the codebase after this
data:image/s3,"s3://crabby-images/3eff9/3eff97f601c37ea55cd2e03451448803713416f7" alt=""
Thank you 🙏
Go code awesome things!
(with Vue 😉)
Common Mistakes in Vue.js and How to Avoid Them
By Daniel Kelly
Common Mistakes in Vue.js and How to Avoid Them
- 1,513