Nuxt 3 já está disponível! Descobre mais sobre isso https://nuxt.com/v3
Let’s build a blazing fast articles and tutorials app using Nuxt and the DEV API, with lazy loading, placeholders, caching and trendy neumorphic design UI.
Let’s build a blazing fast articles and tutorials app using Nuxt and the DEV API, with lazy loading, placeholders, caching and trendy neumorphic design UI.
This article is intended to demonstrate use cases and awesomeness of new Nuxt fetch
functionality introduced in release v2.12 , and show you how to apply its power in your own projects. For in-depth technical analysis and details of the new fetch
you can check Krutie Patel’s article .
Here’s the high-level outline of how we will build our dev.to clone using fetch
hook. We will:
$fetchState
for showing nice placeholders while data is fetching on the client side
keep-alive
and activated
hook to efficiently cache API requests on pages that have already been visited
fetch
hook with this.$fetch()
fetchOnServer
value to control when we need to render our data on the server side or not
fetch
hook.
In September 2019 DEV opened their public API that we can now use to access articles, users and other resources data. Please note that it’s still Beta, so it could change in future or some things might not work as expected.
For creating our DEV clone we are interested in such API endpoints:
tag
, state
, top
, username
and paginated with page
parameters
To keep it simple, for communication with the DEV API we will use native JavaScript Fetch API .
If you are an experienced developer you can skip this part and get straight to the point .
Make sure you have Node and npm installed. We will use create-nuxt-app
to initialize the project, so just type the following command in terminal:
npx create-nuxt-app nuxt-dev-to-clone
# leave the default answers for each question
Now cd nuxt-dev-to-clone/
and run npm run dev
. Congrats, your Nuxt app is running on http://localhost:3000 !
Let’s install necessary packages and discuss how we will build our app next.
For styling we will use the most common CSS pre-processor Sass/SCSS and leverage Vue.js Scoped CSS feature, to keep our components styles encapsulated. To use Sass/SCSS with Nuxt run:
yarn add sass sass-loader@10 -D
npm install sass sass-loader@10 --save-dev
We also will use @nuxtjs/style-resources module that will help us to use our design tokens defined in SCSS variables in any Vue file without the necessity of using @import
statements in each file.
yarn add @nuxtjs/style-resources
npm install @nuxtjs/style-resources
Now tell Nuxt to use it by adding this code to nuxt.config.js
buildModules: ['@nuxtjs/style-resources']
Read more about this module here , regarding buildModules
, you can learn more on it in the modules vs buildModules section of the documentation.
Let’s define our design tokens as SCSS variables, put them in ~/assets/styles/tokens.scss
and tell @nuxtjs/style-resources
to load them by adding to nuxt.config.js
styleResources: {
scss: ['~/assets/styles/tokens.scss']
}
Our design tokens are now available through SCSS variables in every Vue component.
It will be kinda boring just to copy the existing DEV design and layout, so why don’t we experiment a little bit. You have probably already heard of the new UI trend — neumorphism. If you missed it somehow, read more about it here .
We can find a lot of Dribbble shots (from where this trend came from), but still only a few examples of real-world web apps built with neumorphism style interface, so we just can’t miss the chance to recreate it with CSS and Vue.js. It’s simple, clean and fresh.
I am not going to describe the styling aspect of this application in detail, but if you are interested, you can check this awesome article from CSS Tricks about neumorphism and CSS.
For SVG icons lets use @nuxt/svg . This module allows us to import .svg
files as inline SVG, while keeping SVG sources in single place and not polluting Vue template markup with loads of SVG code.
yarn add @nuxtjs/svg -D
npm install @nuxtjs/svg -D
buildModules: ['@nuxtjs/svg', '@nuxtjs/style-resources']
To keep the frontend app fast and simple we will use only two dependencies, both from Vue.js core members:
Let’s add them as Nuxt plugins , by creating two files.
import Vue from 'vue'
import VueObserveVisibility from 'vue-observe-visibility'
Vue.use(VueObserveVisibility)
import Vue from 'vue'
import VueContentPlaceholders from 'vue-content-placeholders'
Vue.use(VueContentPlaceholders)
And add them to ``
plugins: [
'~/plugins/vue-placeholders.js',
'~/plugins/vue-observe-visibility.client.js'
]
Now we finally can start developing our DEV clone powered by Nuxt and new fetch .
Let’s imitate the DEV URL structure for our simple app. Our pages folder should look like this:
├── index.vue
├── t
│ └── _tag.vue
├── top.vue
└── _username
├── _article.vue
└── index.vue
We will have 2 static pages :
index.vue
: latest articles about Nuxt will be listed
top.vue
: most popular articles for last year period.
For the rest of the app URL’s we will use convenient Nuxt file based dynamic routes feature to scaffold necessary pages by creating such file structure:
_username/index.vue
- user profile page with list of his published articles
_username/_article.vue
- this is where article, author profile and comments will be rendered
t/_tag.vue
- list of best articles by any tag that exist on DEV
That’s all. Pretty simple, right?
keep-alive
and activated
hook One of the coolest practical features of the new fetch
is its ability to work with keep-alive
directive to save fetch
calls on pages you have already visited. Let’s apply this feature in layouts/default.vue
layout like this.
<template>
<nuxt keep-alive />
</template>
With this directive fetch
will trigger only on the first page visit, after that Nuxt will save rendered components in memory, and on every subsequent visit it will be just reused from the cache. Could it be simpler than that?
Moreover, Nuxt gives us fine grained control over keep-alive
with the keep-alive-props
property where you can set the number of components which you want to cache, and activated
hook, where you can control TTL (time to live) of the cache. We will use the latest one in our app in the next sections.
fetch
in page components Let’s dive into the fetch
feature itself.
Currently as you can see from the final result we have 3 page components that basically reuse the same code — it’s the index.vue
, top.vue
and t/_tag.vue
page components. They simply render a list of article preview cards.
<template>
<div class="page-wrapper">
<div class="article-cards-wrapper">
<article-card-block
v-for="(article, i) in articles"
:key="article.id"
:article="article"
class="article-card-block"
/>
</div>
</div>
</template>
<script>
import ArticleCardBlock from '@/components/blocks/ArticleCardBlock'
export default {
components: {
ArticleCardBlock
},
data() {
return {
currentPage: 1,
articles: []
}
},
async fetch() {
const articles = await fetch(
`https://dev.to/api/articles?tag=nuxt&state=rising&page=${this.currentPage}`
).then(res => res.json())
this.articles = this.articles.concat(articles)
}
}
</script>
Pay attention to this code block:
async fetch() {
const articles = await fetch(`https://dev.to/api/articles?tag=nuxt&state=rising&page=${this.currentPage}`).then((res) => res.json())
this.articles = this.articles.concat(articles)
}
Here we are making a request to the DEV /articles
endpoint, with query parameters that API understands. Don’t confuse the fetch
hook with the JavaScript fetch interface which simply helps us to send a request to the DEV API, and then parse the response with res.json()
.
Also notice that the new fetch
hook doesn’t serve just to dispatch Vuex store action or committing mutation to set state, now it has access to this
context, and is able to mutate component’s data directly. This is a very important new feature, and you can read more about it in the previous article about fetch
.
Now let’s markup the <article-card-block>
component which receives article
prop and renders its data nicely.
<template>
<nuxt-link
:to="{ name: 'username-article', params: { username: article.user.username, article: article.id } }"
tag="article"
>
<div class="image-wrapper">
<img
v-if="article.cover_image"
:src="article.cover_image"
:alt="article.title"
/>
<img v-else :src="article.social_image" :alt="article.title" />
</div>
<div class="content">
<nuxt-link
:to="{name: 'username-article', params: { username: article.user.username, article: article.id } }"
>
<h1>{{ article.title }}</h1>
</nuxt-link>
<div class="tags">
<nuxt-link
v-for="tag in article.tag_list"
:key="tag"
:to="{ name: 't-tag', params: { tag } }"
class="tag"
>
#{{ tag }}
</nuxt-link>
</div>
<div class="meta">
<div class="scl">
<span>
<heart-icon />
{{ article.positive_reactions_count }}
</span>
<span>
<comments-icon />
{{ article.comments_count }}
</span>
</div>
<time>{{ article.readable_publish_date }}</time>
</div>
</div>
</nuxt-link>
</template>
<script>
import HeartIcon from '@/assets/icons/heart.svg?inline'
import CommentsIcon from '@/assets/icons/comments.svg?inline'
export default {
components: {
HeartIcon,
CommentsIcon
},
props: {
article: {
type: Object,
default: null
}
}
}
</script>
fetch
with this.$fetch()
It already should display a list of articles fetched from DEV - but it feels like we are not making full use of this API. Let’s add lazy loading to the articles list, and use the pagination parameter provided by this API. So when we scroll to the bottom of the page a new chunk of articles will be fetched and rendered.
To efficiently detect when to fetch the next page it’s better to use Intersection Observer API . For that we will use a previously installed Vue plugin called vue-observe-visibility
which is basically a wrapper around this API and it will detect when an element is becoming visible or hidden on the page. This plugin provides us a possibility to use v-observe-visibility
directive on any element, so let’s add it to last <article-card-block>
component:
<article-card-block
v-for="(article, i) in articles"
:key="article.id"
v-observe-visibility="
i === articles.length - 1 ? lazyLoadArticles : false
"
:article="article"
class="article-card-block"
/>
As you can guess from the code above, when the last <article-card-block>
appears in viewport lazyLoadArticles
will be fired. Let’s look at it:
lazyLoadArticles(isVisible) {
if (isVisible) {
if (this.currentPage < 5) {
this.currentPage++
this.$fetch()
}
}
}
And here we see the power of the new fetch
hook. We can just reuse $fetch
as a method and fetch the next page when lazy loading is triggered.
$fetchState
If you already applied code from the previous section and tried client-side navigation between index.vue
, top.vue
and t/_tag.vue
page components you probably noticed that it shows an empty page for the moment, while it’s waiting for the API request to complete. This is intended behavior, and it’s different from the old fetch
and asyncData
hooks that triggered before page navigation.
Thanks to $fetchState.pending
wisely provided by the fetch
hook we can use this flag to display a placeholder when fetch is being called on client-side. vue-content-placeholders
plugin will be used as a placeholder.
<template>
<div class="page-wrapper">
<template v-if="$fetchState.pending">
<div class="article-cards-wrapper">
<content-placeholders
v-for="p in 30"
:key="p"
rounded
class="article-card-block"
>
<content-placeholders-img />
<content-placeholders-text :lines="3" />
</content-placeholders>
</div>
</template>
<template v-else-if="$fetchState.error">
<p>{{ $fetchState.error.message }}</p>
</template>
<template v-else>
<div class="article-cards-wrapper">
<article-card-block
v-for="(article, i) in articles"
:key="article.id"
v-observe-visibility="
i === articles.length - 1 ? lazyLoadArticles : false
"
:article="article"
class="article-card-block"
/>
</div>
</template>
</div>
</template>
We imitate how <article-card-block>
looks with vue-content-placeholders components , and as you could see in source code it will be used in almost every component that uses the fetch
hook, so let’s not pay attention on those parts of code anymore (they are basically the same in each component).
fetch
in any other component 🔥 This is probably the most interesting feature of the new fetch
hook. We can now use the fetch
hook in any Vue component without worrying about breaking SSR! This means far less headache about how to structure your async API calls and components.
To explore this great functionality let’s move to _username/_article.vue
page component.
<template>
<div class="page-wrapper">
<div class="article-content-wrapper">
<article-block class="article-block" />
<div class="aside-username-wrapper">
<aside-username-block class="aside-username-block" />
</div>
</div>
<comments-block class="comments-block" />
</div>
</template>
<script>
import ArticleBlock from '@/components/blocks/ArticleBlock'
import CommentsBlock from '@/components/blocks/CommentsBlock'
import AsideUsernameBlock from '@/components/blocks/AsideUsernameBlock'
export default {
components: {
ArticleBlock,
CommentsBlock,
AsideUsernameBlock
}
}
</script>
Here we see no data fetching at all, only a template layout consisting of 3 components: <article-block/>
, <aside-username-block/>
, <comments-block/>
. And each of those components has its own fetch
hook. With old fetch
or current asyncData
earlier we would have to make all three requests to three different DEV endpoints and then pass them to each component as a prop. But now those components are completely encapsulated.
In <article-block/>
we use fetch
just like we’d use it in a page component.
async fetch() {
const article = await fetch(
`https://dev.to/api/articles/${this.$route.params.article}`
).then((res) => res.json())
if (article.id && article.user.username === this.$route.params.username) {
this.article = article
this.$store.commit('SET_CURRENT_ARTICLE', this.article)
} else {
// set status code on server
if (process.server) {
this.$nuxt.context.res.statusCode = 404
}
// throwing an error will set $fetchState.error
throw new Error('Article not found')
}
}
Now, remember in the section about caching I mentioned that there’s an activated
hook that can be used for managing TTL of fetch
? This is example of such usage:
activated() {
if (this.$fetchState.timestamp <= Date.now() - 60000) {
this.$fetch()
}
}
With this code in place we will call fetch again if last fetch was more than 60 sec ago. All other requests within this period will be cached.
There’s also interesting usage of another fetch
feature called fetchOnServer
in the <comments-block>
component. We don’t really want to render this content on the server side, because comments are user generated and could be irrelevant or spammy. We don’t need any SEO for this content block. Now, with the help of mentioned fetchOnServer
we have such control:
async fetch() {
this.comments = await fetch(
`https://dev.to/api/comments?a_id=${this.$route.params.article}`
).then((res) => res.json())
},
fetchOnServer: false
Last thing that should be mentioned is error handling. You probably already saw that we used error handling above, but let’s pay more attention to this important topic.
As you know, fetch
is handled at the component level, when doing server-side rendering, the parent (virtual) dom tree is already rendered when rendering the component, so we cannot change it by calling $nuxt.error(...)
, instead we have to handle the error at the component level.
$fetchState.error
is set if an error is thrown in the fetch
hook, so we can use it in our template to display an error message:
<template>
<div class=“page-wrapper”>
<template v-if="$fetchState.pending">
<!— placeholders goes here —>
</template>
<template v-else-if=“$fetchState.error">
<p>{{ $fetchState.error.message }}</p>
</template>
<template v-else>
<!— fetched content goes here —>
</template>
</div>
</template>
Then, in our fetch
hook, we will throw the error if we don't find the article corresponding for the defined author:
async fetch() {
const article = await fetch(
`https://dev.to/api/articles/${this.$route.params.article}`
).then((res) => res.json())
if (article.id && article.user.username === this.$route.params.username) {
this.article = article
} else {
// set status code on server
if (process.server) {
this.$nuxt.context.res.statusCode = 404
}
throw new Error('Article not found')
}
}
Note here that we wrap this.$nuxt.context.res.statusCode = 404
around process.server
, this is used to set the HTTP status code on the server-side for correct SEO.
In this article we explored Nuxt new fetch
and built an app with the basic DEV content features and structure using only this fetch
hook. I hope you've got some inspiration to build your own version of DEV.TO. Don’t forget to check out the source code for a more complete example and functionality.
What to do next:
fetch
hook works