Skip to content

3. Uma primeira aplicação [nuxt.js]

3.1. Criação da aplicação

Para os nossos desenvolvimentos [nuxt.js], continuamos a utilizar o VS Code. Criámos uma pasta [dvp] vazia, na qual iremos colocar os nossos exemplos. Em seguida, abrimos essa pasta:

Image

Guardamos a área de trabalho com o nome [intro-nuxtjs] [3-5]:

Image

Abrimos um terminal [6-7]:

Image

Até agora, temos utilizado o gestor de pacotes JavaScript [npm]. Para variar, vamos utilizar aqui o gestor [yarn]. Este está instalado, tal como o [npm], com as versões recentes do [node.js]. Para criar uma primeira aplicação [nuxt], utilizamos o comando [yarn create nuxt-app <dossier>] [1]. O comando irá solicitar algumas informações sobre o projeto a gerar e, uma vez obtidas essas informações, irá gerá-lo [2]:

Image

No [2], foi criada toda uma árvore de ficheiros. O ficheiro [package.json] apresenta a lista das bibliotecas JavaScript descarregadas para a pasta [node-modules] [4]:


{
  "name": "nuxt-intro",
  "version": "1.0.0",
  "description": "nuxt-intro",
  "author": "serge-tahe",
  "private": true,
  "scripts": {
    "dev": "nuxt",
    "build": "nuxt build",
    "start": "nuxt start",
    "generate": "nuxt generate",
    "lint": "eslint --ext .js,.vue --ignore-path .gitignore ."
  },
  "dependencies": {
    "nuxt": "^2.0.0",
    "bootstrap-vue": "^2.0.0",
    "bootstrap": "^4.1.3",
    "@nuxtjs/axios": "^5.3.6"
  },
  "devDependencies": {
    "@nuxtjs/eslint-config": "^1.0.1",
    "@nuxtjs/eslint-module": "^1.0.0",
    "babel-eslint": "^10.0.1",
    "eslint": "^6.1.0",
    "eslint-plugin-nuxt": ">=0.4.2",
    "eslint-config-prettier": "^4.1.0",
    "eslint-plugin-prettier": "^3.0.1",
    "prettier": "^1.16.4"
  }
}

Este ficheiro reflete as respostas fornecidas ao comando [create nuxt-app] para definir o projeto criado (novembro de 2019). O leitor pode ter um ficheiro [package.json] diferente:

  • pode ter dado respostas diferentes às perguntas;
  • o comando [create nuxt-app] terá sofrido alterações desde a redação deste documento: as dependências e as versões terão mudado;

A linha 8 do script é o comando que inicia a aplicação:

Image

  • em [4], verifica-se que a aplicação está disponível em URL e [localhost:3000];
  • em [5-6], verifica-se que a aplicação dá origem a um servidor [6] e a um cliente (desse servidor) [5];

Vamos aceder a URL [http://localhost:3000/] num navegador:

Image

3.2. Descrição da estrutura de uma aplicação [nuxt]

Voltemos à estrutura da aplicação criada:

Image

A função das pastas é a seguinte:

assets
recursos não compilados da aplicação (imagens, ...);
static
os ficheiros desta pasta estarão disponíveis na raiz da aplicação. Nesta pasta colocam-se os ficheiros que devem estar na raiz da aplicação, como, por exemplo, o ficheiro [robots.txt] destinado aos motores de busca;
components
os componentes [vue] da aplicação utilizados nos [layouts] e nos [pages];
layouts
os componentes [vue] da aplicação que servem de layout para os [pages];
páginas
os componentes [vue] apresentados pelas diferentes rotas da aplicação. Poderiam ser designados como as vistas da aplicação. As páginas desempenham um papel específico no [nuxt]: as rotas são criadas dinamicamente a partir da estrutura de pastas encontrada na pasta [pages];
middleware
os scripts executados a cada mudança de rota. Permitem controlar essas rotas;
plugins
tem um nome que pode causar confusão. Pode conter plugins, mas também scripts clássicos. Os scripts encontrados nesta pasta são executados no arranque da aplicação;
store
se contiver um script [index.js], este define uma instância do store de [Vuex];

Se uma pasta estiver vazia, pode ser eliminada da árvore de pastas. No exemplo acima, as pastas [assets, static, middleware, plugins, store] e [2] podem ser eliminadas.

3.3. O ficheiro de configuração [nuxt.config]

A execução da aplicação é controlada pelo seguinte ficheiro [nuxt.config.js]:


export default {
  mode: 'universal',
  /*
   ** Headers of the page
   */
  head: {
    title: process.env.npm_package_name || '',
    meta: [
      { charset: 'utf-8' },
      { name: 'viewport', content: 'width=device-width, initial-scale=1' },
      {
        hid: 'description',
        name: 'description',
        content: process.env.npm_package_description || ''
      }
    ],
    link: [{ rel: 'icon', type: 'image/x-icon', href: '/favicon.ico' }]
  },
  /*
   ** Customize the progress-bar color
   */
  loading: { color: '#fff' },
  /*
   ** Global CSS
   */
  css: [],
  /*
   ** Plugins to load before mounting the App
   */
  plugins: [],
  /*
   ** Nuxt.js dev-modules
   */
  buildModules: [
    // Doc: https://github.com/nuxt-community/eslint-module
    '@nuxtjs/eslint-module'
  ],
  /*
   ** Nuxt.js modules
   */
  modules: [
    // Doc: https://bootstrap-vue.js.org
    'bootstrap-vue/nuxt',
    // Documentação: https://axios.nuxtjs.org/usage
    '@nuxtjs/axios'
  ],
  /*
   ** Axios module configuration
   ** See https://axios.nuxtjs.org/options
   */
  axios: {},
  /*
   ** Build configuration
   */
  build: {
    /*
     ** You can extend webpack config here
     */
    extend(config, ctx) {}
  }
}
  • linha 2: o tipo de aplicação gerada:
    • [universal]: aplicação cliente/servidor. No carregamento inicial da aplicação, bem como em cada atualização da página no navegador, o servidor é solicitado a fornecer a página;
    • [sap]: aplicação do tipo [Single Page Application]: um servidor fornece inicialmente a totalidade da aplicação. Posteriormente, o cliente opera de forma autónoma, mesmo em caso de atualização de uma página no navegador;
  • linhas 6-18: definem o cabeçalho HTML <head> das diferentes páginas da aplicação:
    • linha 7: a baliza <title> do título das páginas;
    • linhas 8-16: as tags <meta>;
    • linha 17: as tags <link>

Na aplicação gerada, a baliza <head> é a seguinte (código-fonte da página apresentada no navegador):


<title>nuxt-intro</title>
<meta data-n-head="ssr" charset="utf-8">
<meta data-n-head="ssr" name="viewport" content="width=device-width, initial-scale=1">
<meta data-n-head="ssr" data-hid="description" name="description" content="nuxt-intro">
<link data-n-head="ssr" rel="icon" type="image/x-icon" href="/favicon.ico">
<link rel="preload" href="/_nuxt/runtime.js" as="script">
<link rel="preload" href="/_nuxt/commons.app.js" as="script">
<link rel="preload" href="/_nuxt/vendors.app.js" as="script">
<link rel="preload" href="/_nuxt/app.js" as="script">

Agora, vamos alterar o ficheiro [nuxt.config] da seguinte forma:


head: {
    title: 'Introduction à [nuxt.js]',
    meta: [
      { charset: 'utf-8' },
      { name: 'viewport', content: 'width=device-width, initial-scale=1' },
      {
        hid: 'description',
        name: 'description',
        content: 'ssr routing loading asyncdata middleware plugins store'
      }
    ],
    link: [{ rel: 'icon', type: 'image/x-icon', href: '/favicon.ico' }]
},

Quando voltamos a executar a aplicação, a baliza <head> passou a ter o seguinte aspeto (código-fonte da página apresentada no navegador):


  <head >
    <title>Introduction à [nuxt.js]</title>
    <meta data-n-head="ssr" charset="utf-8">
    <meta data-n-head="ssr" name="viewport" content="width=device-width, initial-scale=1">
    <meta data-n-head="ssr" data-hid="description" name="description" content="ssr routing loading asyncdata middleware plugins store">
    <link data-n-head="ssr" rel="icon" type="image/x-icon" href="/favicon.ico">
    <link rel="preload" href="/_nuxt/runtime.js" as="script">
    <link rel="preload" href="/_nuxt/commons.app.js" as="script">
    <link rel="preload" href="/_nuxt/vendors.app.js" as="script">
<link rel="preload" href="/_nuxt/app.js" as="script">

Voltemos ao ficheiro [nuxt.config]:


export default {
  mode: 'universal',
  /*
   ** Headers of the page
   */
  head: {
    ...
  },
  /*
   ** Customize the progress-bar color
   */
  loading: { color: '#fff' },
  /*
   ** Global CSS
   */
  css: [],
  /*
   ** Plugins to load before mounting the App
   */
  plugins: [],
  /*
   ** Nuxt.js dev-modules
   */
  buildModules: [
    // Doc: https://github.com/nuxt-community/eslint-module
    '@nuxtjs/eslint-module'
  ],
  /*
   ** Nuxt.js modules
   */
  modules: [
    // Doc: https://bootstrap-vue.js.org
    'bootstrap-vue/nuxt',
    // Documentação: https://axios.nuxtjs.org/usage
    '@nuxtjs/axios'
  ],
  /*
   ** Axios module configuration
   ** See https://axios.nuxtjs.org/options
   */
  axios: {},
  /*
   ** Build configuration
   */
  build: {
    /*
     ** You can extend webpack config here
     */
    extend(config, ctx) {}
  }
}
  • linha 12: entre cada rota do cliente [nuxt], surge uma barra de carregamento (loading) se a mudança de rota demorar algum tempo. A propriedade [loading] permite configurar esta barra de carregamento, neste caso a cor da barra;
  • linha 16: os ficheiros globais [css]. Serão automaticamente incluídos em todas as páginas da aplicação;
  • linhas 24-27: os módulos JavaScript necessários para a compilação (build) da aplicação;
  • linhas 31-36: os módulos JavaScript utilizados pela aplicação;
  • linha 41: configuração da biblioteca [axios] quando esta tiver sido selecionada pelo utilizador para as caixas de diálogo HTTP com servidores de terceiros;
  • linhas 45-50: configuração da compilação (build) do projeto;

É possível adicionar outras chaves ao ficheiro de configuração. Em particular, é possível definir a porta do serviço (3000 por predefinição) e a raiz do projeto (por predefinição, a pasta raiz do projeto). É isso que vamos fazer agora, adicionando as seguintes chaves:


// diretório do código-fonte
  srcDir: '.',
  router: {
    // URL raiz das páginas da aplicação
    base: '/nuxt-intro/'
  },
  // servidor
  server: {
    // porta do serviço - por predefinição 3000
    port: 81,
    // endereços de rede em escuta - por predefinição localhost=127.0.0.1
    host: '0.0.0.0'
  }
  • linha 2: onde se encontra o código-fonte do projeto. Encontra-se aqui na pasta atual, ou seja, ao mesmo nível que o ficheiro [nuxt.config.js]. Este é o valor por predefinição;
  • linhas 8-13: configuram o servidor (não se deve esquecer que uma aplicação [nuxt] do tipo [universal] está instalada tanto num servidor como num navegador cliente desse servidor);
  • linha 10: as páginas da aplicação serão servidas na porta 81 do servidor;
  • linha 12: por predefinição, [localhost] (endereço de rede 127.0.0.1). Um computador pode ter vários endereços de rede se pertencer a várias redes. O endereço 0.0.0.0 indica que o servidor web escuta todos os endereços de rede do computador;
  • linhas 3-6: configuram o router da aplicação [nuxt];
  • linha 5: as páginas da aplicação estarão disponíveis em URL e [http://localhost:81/nuxt-intro/];

Adicionemos estas linhas ao ficheiro [nuxt.config.js] e, em seguida, executemos o projeto (script npm dev). O resultado é o seguinte:

Image

  • em [1], o endereço do computador numa rede pública;
  • em [2], a porta do serviço;
  • em [3], a raiz da aplicação;

3.4. A pasta [layouts]

Image

A pasta [layouts] destina-se aos componentes de layout. Por predefinição, é utilizado o componente denominado [default.vue]. Neste projeto, este é o seguinte:


<template>
  <div>
    <nuxt />
  </div>
</template>

<style>
html {
  font-family: 'Source Sans Pro', -apple-system, BlinkMacSystemFont, 'Segoe UI',
    Roboto, 'Helvetica Neue', Arial, sans-serif;
  font-size: 16px;
  word-spacing: 1px;
  -ms-text-size-adjust: 100%;
  -webkit-text-size-adjust: 100%;
  -moz-osx-font-smoothing: grayscale;
  -webkit-font-smoothing: antialiased;
  box-sizing: border-box;
}

*,
*:before,
*:after {
  box-sizing: border-box;
  margin: 0;
}

.button--green {
  display: inline-block;
  border-radius: 4px;
  border: 1px solid #3b8070;
  color: #3b8070;
  text-decoration: none;
  padding: 10px 30px;
}

.button--green:hover {
  color: #fff;
  background-color: #3b8070;
}

.button--grey {
  display: inline-block;
  border-radius: 4px;
  border: 1px solid #35495e;
  color: #35495e;
  text-decoration: none;
  padding: 10px 30px;
  margin-left: 15px;
}

.button--grey:hover {
  color: #fff;
  background-color: #35495e;
}
</style>

Comentários

  • linhas 1-5: o [template] do componente;
  • linha 3: a baliza <nuxt /> designa a página atual do encaminhamento;
  • linhas 7-55: o estilo incorporado pelo componente de layout. Como este contém a página atual do roteamento, este estilo será aplicado a todas as páginas roteadas da aplicação;

Vemos que o objetivo principal da página [default.vue] é, neste caso, aplicar um estilo às páginas encaminhadas.

3.5. A pasta [pages]

Image

A pasta [pages] contém as vistas encaminhadas, aquelas que o utilizador vê. A página [index.vue] é a página inicial da aplicação. No caso de [nuxt.js], não existe um ficheiro de encaminhamento. As rotas são determinadas a partir da estrutura da pasta [pages]. Neste caso, a presença de um ficheiro [index.vue] irá criar automaticamente uma rota denominada [index] e um caminho [/index], reduzido para [/], uma vez que se trata da página inicial. Assim, é criada a seguinte rota:

        { name : ‘index’, path : ‘/’}

O ficheiro [index.vue] é o seguinte:


<template>
  <div class="container">
    <div>
      <logo />
      <h1 class="title">
        nuxt-intro
      </h1>
      <h2 class="subtitle">
        nuxt-intro
      </h2>
      <div class="links">
        <a href="https://nuxtjs.org/" target="_blank" class="button--green">
          Documentation
        </a>
        <a
          href="https://github.com/nuxt/nuxt.js"
          target="_blank"
          class="button--grey"
        >
          GitHub
        </a>
      </div>
    </div>
  </div>
</template>

<script>
import Logo from '~/components/Logo.vue'

export default {
  components: {
    Logo
  }
}
</script>

<style>
.container {
  margin: 0 auto;
  min-height: 100vh;
  display: flex;
  justify-content: center;
  align-items: center;
  text-align: center;
}

.title {
  font-family: 'Quicksand', 'Source Sans Pro', -apple-system, BlinkMacSystemFont,
    'Segoe UI', Roboto, 'Helvetica Neue', Arial, sans-serif;
  display: block;
  font-weight: 300;
  font-size: 100px;
  color: #35495e;
  letter-spacing: 1px;
}

.subtitle {
  font-weight: 300;
  font-size: 42px;
  color: #526488;
  word-spacing: 5px;
  padding-bottom: 15px;
}

.links {
  padding-top: 15px;
}
</style>

O ficheiro [template], nas linhas 1 a 25, apresenta a seguinte visualização:

Image

A imagem [1] é gerada pela linha 4 do [template]. Vemos, portanto, que a página utiliza um componente denominado [logo]. Este está definido nas linhas 27 a 35 do script da página. Na linha 28, a notação [~] designa a raiz do projeto.

Image

O componente [Logo.vue] é o seguinte:


<template>
  <div class="VueToNuxtLogo">
    <div class="Triangle Triangle--two" />
    <div class="Triangle Triangle--one" />
    <div class="Triangle Triangle--three" />
    <div class="Triangle Triangle--four" />
  </div>
</template>

<style>
.VueToNuxtLogo {
  display: inline-block;
  animation: turn 2s linear forwards 1s;
  transform: rotateX(180deg);
  position: relative;
  overflow: hidden;
  height: 180px;
  width: 245px;
}

.Triangle {
  position: absolute;
  top: 0;
  left: 0;
  width: 0;
  height: 0;
}

.Triangle--one {
  border-left: 105px solid transparent;
  border-right: 105px solid transparent;
  border-bottom: 180px solid #41b883;
}

.Triangle--two {
  top: 30px;
  left: 35px;
  animation: goright 0.5s linear forwards 3.5s;
  border-left: 87.5px solid transparent;
  border-right: 87.5px solid transparent;
  border-bottom: 150px solid #3b8070;
}

.Triangle--three {
  top: 60px;
  left: 35px;
  animation: goright 0.5s linear forwards 3.5s;
  border-left: 70px solid transparent;
  border-right: 70px solid transparent;
  border-bottom: 120px solid #35495e;
}

.Triangle--four {
  top: 120px;
  left: 70px;
  animation: godown 0.5s linear forwards 3s;
  border-left: 35px solid transparent;
  border-right: 35px solid transparent;
  border-bottom: 60px solid #fff;
}

@keyframes turn {
  100% {
    transform: rotateX(0deg);
  }
}

@keyframes godown {
  100% {
    top: 180px;
  }
}

@keyframes goright {
  100% {
    left: 70px;
  }
}
</style>

Este componente é constituído essencialmente por estilos e animações para criar uma imagem animada.

3.7. Vista DevTools

[Vue DevTools] é a extensão do navegador que permite inspecionar os objetos [nuxt.js] e [vue.js] no navegador. Já a utilizámos no capítulo sobre o [vue.js]. Vejamos o que esta ferramenta encontra quando a página inicial da nossa aplicação é apresentada:

Image

  • em [1], o componente [PagesIndex] remete para a página [pages/index.vue];
  • vemos em [2] que este componente tem uma propriedade [$route], que é a rota que conduziu à página [index];

Como exercício simples, vamos apresentar esse caminho na consola.

3.8. Alteração da página inicial

Vamos modificar o ficheiro [index.vue]. Na nossa instalação do projeto, instalámos duas dependências:

  • [eslint]: que verifica a sintaxe dos ficheiros JavaScript e dos componentes Vue. Se a extensão [ESLint] do VSCode tiver sido instalada, essa sintaxe é verificada à medida que o texto é digitado e os erros são imediatamente assinalados;
  • [prettier]: que formata os códigos JavaScript de forma padronizada;

Estas dependências estão registadas no ficheiro [package.json]:


"devDependencies": {
    "@nuxtjs/eslint-config": "^1.0.1",
    "@nuxtjs/eslint-module": "^1.0.0",
    "babel-eslint": "^10.0.1",
    "eslint": "^6.1.0",
    "eslint-config-prettier": "^4.1.0",
    "eslint-plugin-nuxt": ">=0.4.2",
    "eslint-plugin-prettier": "^3.0.1",
    "prettier": "^1.16.4"
}

Reparei (novembro de 2019) que, com a instalação efetuada através do comando [yarn create nuxt-app], as ferramentas [eslint, prettier] não funcionam durante a digitação dos textos. Os erros só são assinalados na compilação. Após algumas pesquisas, encontrei uma configuração que funciona:

Image

Instala-se, na raiz do projeto, uma pasta [.vscode] contendo o seguinte ficheiro [settings.json]:


{
  "eslint.validate": [
    {
      "language": "vue",
      "autoFix": true
    },
    {
      "language": "javascript",
      "autoFix": true
    }
  ],
  "eslint.autoFixOnSave": true,
  "editor.formatOnSave": false
}
  • linhas 2-11: indicam que, quando o [eslint] valida os ficheiros .vue e .js, deve corrigir os erros que for possível corrigir;
  • linha 12: quando um ficheiro é guardado, o [eslint] deve corrigir os erros que for possível corrigir;
  • linha 13: desativa a formatação efetuada por predefinição no VSCode durante um salvamento. Será o [prettier] que o fará;

Com esta configuração:

  • os erros de sintaxe ou de formatação são assinalados logo que o texto é digitado;
  • os erros de formatação são corrigidos automaticamente ao guardar o ficheiro;

A biblioteca [prettier] é configurada pelo ficheiro [.prettierrc]:

Image

Por predefinição, este ficheiro é o seguinte:


{
  "semi": false,
  "arrowParens": "always",
  "singleQuote": true
}
  • linha 1: sem «;» no final das instruções;
  • linha 2: se uma função «seta» (arrow) tiver um único parâmetro, este deve estar entre parênteses;
  • linha 3: as cadeias de caracteres são colocadas entre apóstrofos (sem aspas);

Adicionamos as duas regras seguintes:


{
  "semi": false,
  "arrowParens": "always",
  "singleQuote": true,
  "printWidth": 120,
  "endOfLine": "auto"
}
  • linha 5: a linha de código pode ter até 120 caracteres;
  • linha 6: o marcador de fim de linha pode ser tanto CRLF (Windows) como LF (Unix);

Por fim, o ficheiro [package.json] é alterado da seguinte forma:


"scripts": {
    "dev": "nuxt",
    "build": "nuxt build",
    "start": "nuxt start",
    "generate": "nuxt generate",
    "lint": "eslint --ext .js,.vue --ignore-path .gitignore .",
    "lintfix": "eslint --fix --ext .js,.vue --ignore-path .gitignore ."
},
  • linha 7: adicionamos o comando [lintfix], que é idêntico ao comando [lint] da linha 6, com a diferença de que inclui adicionalmente o parâmetro [--fix]. O comando [lint] verifica a sintaxe e o formato de todos os ficheiros do projeto e assinala qualquer erro. O comando [lintfix] fará o mesmo, com a diferença de que os problemas de formatação que possam ser corrigidos serão corrigidos automaticamente. O comando [lintfix] deve ser utilizado se a compilação falhar devido a problemas de formatação dos ficheiros;

Feito isto, alteramos o ficheiro [index.vue] da seguinte forma:

Image


<script>
/* eslint-disable no-console */
import Logo from '~/components/Logo.vue'

export default {
  components: {
    Logo
  },
  // ciclo de vida
  created() {
    console.log('created, route=', this.$route)
  }
}
</script>
  • linhas 10-12: adicionamos a função [created], que é executada automaticamente quando o componente é criado;
  • linha 11: exibe-se a rota atual;
  • linha 2: um comentário destinado à função [eslint]. Sem este comentário, a função [eslint] sinaliza um erro linha 11: não são permitidas instruções [console] nas funções do ciclo de vida. O [eslint] é configurável. Vamos manter a sua configuração por predefinição e utilizaremos comentários como o da linha 2 para desativar uma regra específica do [eslint]. Utilizaremos dois tipos de comentários:
    • /* desativação da regra [eslint] */: desativação de uma regra para todo o ficheiro;
    • // desativação da regra [eslint]: desativação de uma regra para a linha seguinte;

Durante a digitação, os erros são assinalados e está disponível uma função [Quick Fix]:

Image

Executa-se o projeto:

Image

  • em [1], o separador [Vue] das ferramentas de desenvolvimento do navegador (F12);
  • em [2] e [3], a visualização do percurso;

Por que razão há duas visualizações e não apenas uma?

Uma aplicação [nuxt] é composta por dois elementos, um servidor e um cliente:

  1. o servidor fornece as páginas da aplicação no seu arranque e, posteriormente, sempre que uma página é atualizada no navegador (F5) ou quando o utilizador introduz manualmente um URL da aplicação;
  2. cada página fornecida pelo navegador contém a página solicitada, bem como o código JavaScript de toda a aplicação, que é posteriormente executado no navegador. Este é o cliente. Enquanto não houver atualização da página no navegador, a aplicação funciona como uma aplicação Vue clássica no modo [sap] (Aplicação de Página Única). Assim que o utilizador provocar manualmente uma atualização da página, esta é solicitada ao servidor e regressa-se à fase 1 anterior.

O que é preciso compreender é que são as mesmas páginas do dossier [pages] que são fornecidas pelo servidor ou pelo cliente. Por esta razão, os criadores do [nuxt] designam este tipo de páginas como páginas isomórficas. As mesmas páginas [.vue] podem ser interpretadas tanto pelo cliente como pelo servidor. Tomemos como exemplo a página [index]:


<template>
  <div class="container">
    <div>
      <logo />
      <h1 class="title">
        nuxt-intro
      </h1>
      <h2 class="subtitle">
        nuxt-intro
      </h2>
      <div class="links">
        <a href="https://nuxtjs.org/" target="_blank" class="button--green">
          Documentation
        </a>
        <a
          href="https://github.com/nuxt/nuxt.js"
          target="_blank"
          class="button--grey"
        >
          GitHub
        </a>
      </div>
    </div>
  </div>
</template>

<script>
/* eslint-disable no-console */
import Logo from '~/components/Logo.vue'

export default {
  components: {
    Logo
  },
  // ciclo de vida
  created() {
    console.log('created, route=', this.$route)
  }
}
</script>

Como se trata da página inicial, no arranque da aplicação esta é servida pelo servidor. A página no servidor também tem um ciclo de vida, idêntico ao de uma página [Vue] clássica, exceto no que diz respeito às funções [beforeMount, monted], que não existem no lado do servidor. A função [created] é executada, o que explica o primeiro registo. Isto significa, de passagem, que o servidor é capaz de executar scripts JavaScript. Neste caso e, em geral, este servidor é um servidor [node.js]. Depois de criada no servidor, a página chega ao navegador, onde passa novamente pelo ciclo de vida. A função [created] é executada uma segunda vez, o que dá origem ao segundo registo.

A arquitetura de uma aplicação [nuxt] poderia ser a seguinte:

Image

  • [1]: o navegador que hospeda a aplicação [nuxt] quando esta foi carregada no navegador. É o que designamos por cliente [nuxt];
  • [3]: o servidor que hospeda inicialmente a aplicação [nuxt]. Esta é carregada no navegador [1] no arranque da aplicação e sempre que o utilizador atualiza a página atual do navegador ou introduz manualmente um URL da aplicação. É aqui que reside a diferença de funcionamento em relação a uma aplicação Vue clássica. Nesta última, uma vez carregada no navegador, o servidor já não era mais consultado. Outra diferença importante que ainda não conseguimos observar é que o servidor de uma aplicação Vue é um servidor estático, incapaz de interpretar as páginas [.vue], enquanto o de uma aplicação Nuxt do tipo [universal] é um servidor JavaScript. Antes de enviar uma página para o navegador, o servidor pode executar scripts e, por exemplo, ir buscar dados no servidor [2];
  • [2]: é o servidor que fornece dados quer ao cliente [nuxt] [1], quer ao servidor [nuxt] [3];

No esquema acima, é possível distinguir três subsistemas cliente/servidor:

  • [1, 3]: aloja a aplicação [nuxt]. O [3] fornece-a no arranque da aplicação, juntamente com a página inicial, e sempre que o utilizador solicita uma página manualmente. O [1] hospeda aaplicação [nuxt] recebida de [3], que funciona então no modo [SAP] enquanto as páginas não forem solicitadas manualmente a [3];
  • [1, 2]: no modo [SAP], o cliente [nuxt] recupera dados externos de um ou mais servidores;
  • [3, 2]: durante a geração da página solicitada pelo utilizador, o servidor [3] também pode recuperar dados externos de um ou mais servidores;

É, portanto, o servidor [3] que distingue uma aplicação [nuxt] de uma aplicação [vue]. Este servidor é acionado sempre que o utilizador solicita uma página manualmente. Este servidor processa as mesmas páginas [.vue] que os clientes [vue] e [1]. Trata-se de um servidor JavaScript capaz de executar os scripts presentes na página. Isto pode, por exemplo, alterar a forma como a página inicial é gerada com dados externos: enquanto que uma aplicação [vue] obtém esses dados necessariamente a partir do cliente [1], aqui podem ser obtidos pelo servidor [3] antes de a página ser enviada ao cliente. A página inicial torna-se assim significativa e pode contribuir para melhorar o SEO da aplicação.

Nota: no modo de desenvolvimento, as três entidades [1, 2, 3] encontram-se frequentemente na mesma máquina. Será esse o caso aqui em todos os nossos exemplos.

3.9. Mudança do código-fonte da aplicação para uma pasta separada

Posteriormente, iremos criar várias aplicações [nuxt] na mesma pasta [dvp]. Com efeito, a pasta de dependências [node_modules] gerada para cada projeto [nuxt] pode atingir várias centenas de megabytes. Vamos criar várias pastas [nuxt-00, nuxt-01, ...] na pasta [dvp] para conter o código-fonte dos exemplos a testar. Em seguida, utilizaremos o ficheiro de configuração [nuxt-config.js] para indicar onde se encontra o código-fonte do projeto [dvp], que continuará a ser o único projeto [nuxt] deste tutorial.

Vamos mover o código-fonte da aplicação gerada inicialmente pelo comando [yarn create nuxt-app] para uma pasta [nuxt-00]:

Image

  • em [2], movemos as pastas [components, layouts, pages] para uma pasta [nuxt-00];
  • no [3], temos de alterar o ficheiro [nuxt.config.js];

Alteramos o ficheiro [nuxt.config.js] da seguinte forma:


export default {
  mode: 'universal',
  /*
   ** Headers of the page
   */
  ...
  /*
   ** Build configuration
   */
  build: {
    /*
     ** You can extend webpack config here
     */
    extend(config, ctx) {}
  },
  // diretório do código-fonte
  srcDir: 'nuxt-00',
  // router
  router: {
    // raiz dos URL da aplicação
    base: '/nuxt-00/'
  },
  // servidor
  server: {
    // porta de serviço, 3000 por predefinição
    port: 81,
    // endereços de rede em escuta, por predefinição localhost: 127.0.0.1
    // 0.0.0.0 = todos os endereços de rede do computador
    host: '0.0.0.0'
  }
}

O ficheiro é alterado em dois pontos:

  • linha 17: indica-se que o código-fonte do projeto [dvp] se encontra na pasta [nuxt-00];
  • linha 21: indica-se que a raiz da aplicação URL passa a ser [/nuxt-00/]. Esta alteração não era obrigatória. Poderíamos não definir esta propriedade e, nesse caso, a raiz do URL seria [/]. Aqui, isto permitir-nos-á lembrar que o código-fonte executado é o da pasta [nuxt-00];

Feito isto, o projeto [dvp] é executado tal como anteriormente:

Image

3.10. Implantação da aplicação [nuxt-00]

Vamos executar a aplicação [nuxt-00] num ambiente diferente do ambiente integrado da VSCode.

Primeiro, compilamos a aplicação:

Image

  • em [3], o resultado da compilação do cliente. Será executado pelo navegador;
  • em [4], o resultado da compilação do servidor. Será executado pelo servidor [node.js];

O resultado da compilação é colocado na pasta [.nuxt]:

Image

Copiamos as pastas [.nuxt, node_modules] e os ficheiros [package.json, nuxt.config.js] para uma pasta separada:

Image

O ficheiro [package.json] é simplificado da seguinte forma:


{
  "scripts": {
    "start": "nuxt start"
  }
}
  • mantém-se apenas o script [start], que permite executar a versão compilada do projeto;

O ficheiro [nuxt.config.js] é simplificado da seguinte forma:


export default {
  // router
  router: {
    // raiz dos URL da aplicação
    base: '/nuxt-00/'
  },
  // servidor
  server: {
    // porta de serviço, 3000 por predefinição
    port: 81,
    // endereços de rede em escuta, por predefinição localhost: 127.0.0.1
    // 0.0.0.0 = todos os endereços de rede do computador
    host: '0.0.0.0'
  }
}
  • linha 5: define-se o URL de base da aplicação compilada;
  • linhas 8-14: define-se a porta de serviço e os endereços de rede a monitorizar;

Feito isto, abre-se um terminal Laragon e navega-se até à pasta que contém a versão compilada do projeto. É possível abrir qualquer tipo de terminal, mas o executável [npm] tem de estar no diretório do terminal. É o caso do terminal Laragon.

Feito isto, digite o comando [npm run start]:

Image

Em [3], vemos que foi iniciado um servidor e que este está a escutar no URL [http://192.168.1.128:81/nuxt-00/]. Agora, vamos aceder a este endereço URL com um navegador [4]. Temos exatamente o mesmo resultado que anteriormente. No terminal, foram gravados registos [5]. Trata-se do registo inserido no método [created] da página [index.vue], que foi executado pelo servidor [node.js].

Image

No lado do navegador [6], encontra-se também o registo do método [created] da página [index.vue], mas desta vez executado pelo cliente.

3.11. Implementação de um servidor seguro

Acima, o URL da aplicação é [http://192.168.1.128/nuxt-00/]. Gostaríamos que fosse [https://192.168.1.128/nuxt-00/]. Por isso, temos de criar um servidor seguro. Mostramos como proceder.

Nota: o método foi retirado do artigo [https://stackoverflow.com/questions/56966137/how-to-run-nuxt-npm-run-dev-with-https-in-localhost].

Em primeiro lugar, criamos uma chave privada e uma chave pública com o comando [openssl]. O [openssl] é normalmente instalado em simultâneo com o servidor Laragon. Por isso, este comando está disponível em qualquer terminal Laragon. Abramos, portanto, um terminal Laragon e acedamos à pasta da aplicação implementada:

Image

Image

  • no [2], digitamos o comando [openssl genrsa 2048 > server.key];
  • em [3], é criado um ficheiro [server.key];
  • em [4], introduz-se o comando [openssl req -new -x509 -nodes -sha256 -days 365 -key server.key -out server.crt];
  • no [5], é criado um ficheiro [server.crt];

Estes dois ficheiros constituem um certificado autoassinado. A maioria dos navegadores só os aceita após aprovação do utilizador que solicitou a página.

Os ficheiros [server.key, server.crt] devem agora ser utilizados pela aplicação web. Para tal, o ficheiro [nuxt.config.js] deve ser alterado da seguinte forma:


import path from 'path'
import fs from 'fs'

export default {
  // router
  router: {
    // raiz dos URL da aplicação
    base: '/nuxt-00/'
  },
  // servidor
  server: {
    // porta de serviço, 3000 por predefinição
    port: 81,
    // endereços de rede em escuta, por predefinição localhost: 127.0.0.1
    // 0.0.0.0 = todos os endereços de rede do computador
    host: '0.0.0.0',
    // certificado autoassinado
    https: {
      key: fs.readFileSync(path.resolve(__dirname, 'server.key')),
      cert: fs.readFileSync(path.resolve(__dirname, 'server.crt'))
    }
  }
}

São as linhas 18 a 21 que implementam o protocolo [https].

Agora, vamos executar novamente a aplicação:

Image

3.12. Fim do primeiro exemplo

O primeiro exemplo está agora concluído. Ensinou-nos muitos conceitos do [nuxt]. Vamos agora desenvolver outros exemplos que iremos colocar em pastas [nuxt-01, nuxt-02, ...]. Como estes exemplos irão utilizar um ficheiro [nuxt.config.js] diferente, iremos guardar em cada uma destas pastas o ficheiro [nuxt.config.js] que serviu para os executar:

Image