Skip to content

10. Beispiel [nuxt-07]: Client- und Serverkontexte

10.1. Einführung

Beispiel [nuxt-07] zielt darauf ab, das [context]-Objekt sowohl auf der Server- als auch auf der Client-Seite zu untersuchen. Es ist wichtig, sich vor Augen zu halten, dass diese beiden Komponenten von [nuxt]-Anwendungen voneinander getrennt sind: Sie teilen nichts außer:

  • dem, was der Server an den Client sendet (in der HTTP-Antwort und der gerenderten Seite);
  • was der Client an den Server sendet (in seiner HTTP-Anfrage);

Wie wir also sehen werden, haben die vom Server und die vom Client verwalteten Objekte zwar denselben Namen, sind aber nicht identisch: Sie können manchmal Kopien voneinander sein, haben aber niemals dieselbe Referenz. Das Ändern eines clientseitigen Objekts hat keine Auswirkungen auf das gleichnamige Objekt auf der Serverseite und umgekehrt.

Das Beispiel [nuxt-07] wird zunächst durch Kopieren des Beispiels [nuxt-01] erstellt:

Image

  • In [2] fügen wir ein Plugin hinzu, das von Client und Server gemeinsam genutzt wird;
  • in [3] werden wir die Seite [index] leicht modifizieren;

10.2. Das Plugin [common/main.js]

Das Plugin [common / main.js] wird sowohl vom Client als auch vom Server ausgeführt: Dies ist auf die folgende Konfiguration in [nuxt.config.js] zurückzuführen:


/*
   ** Plugins to load before mounting the App
   */
plugins: [{ src: '~/plugins/common/main.js' }],
  • Zeile 4: Das Fehlen der Eigenschaft [mode] bedeutet, dass das Plugin [~/plugins/common/main.js] sowohl vom Client als auch vom Server ausgeführt wird: zuerst vom Server, dann vom Client;

Dieses Plugin sieht wie folgt aus:


/* eslint-disable no-undef */
/* eslint-disable no-console */
export default function(...args) {
  // who executes this code?
  console.log('[main server], process.server=', process.server, 'process.client=', process.client)
  const who = process.server ? 'server' : 'client'
  const main = '[main ' + who + ']'
  // number of arguments
  console.log(main + ', il y a', args.length, 'arguments')
 
  // 1st argument
  const context = args[0]
  // key context
  dumpkeys(main + ', context', context)
  // the application
  dumpkeys(main + ', context.app', context.app)
  // the road
  dumpkeys(main + ', context.route', context.route)
  console.log(main + ', context.route=', context.route)
  // the router
  dumpkeys(main + ', context.app.router', context.app.router)
  // on router.options.routes
  dumpkeys(main + ', context.app.router.options.routes', context.app.router.options.routes)
  console.log(main + ', context.app.router.options.routes=', context.app.router.options.routes)
 
  // 2nd argument
  const inject = args[1]
  console.log('inject=', typeof inject)
}
 
function dumpkeys(message, object) {
  // list of [object] keys
  const ligne = 'Liste des clés [' + message + ']'
  console.log(ligne)
  // kEY LIST
  if (object) {
    console.log(Object.keys(object))
  }
}
  • Zeilen 31–39: Die Funktion [dumpkeys] listet die Eigenschaften des Objekts auf, das als zweiter Parameter übergeben wurde. Dieser Liste geht die als erster Parameter übergebene Meldung voraus;
  • Zeile 3: Wir möchten wissen, wie viele Argumente die Funktion erhält. Dazu verwenden wir die Notation [...args], die die tatsächlichen Funktionsparameter in das Array [args] einfügt. Wir werden feststellen, dass es zwei Argumente gibt;
  • Zeile 5: Wir zeigen an, wer den Code ausführt, der Server oder der Client;
  • Zeile 6: der Code-Ausführer, Client oder Server;
  • Zeile 7: eine in den Protokollen verwendete Zeichenfolgenkonstante;
  • Zeile 12: Wir werden sehen, dass das erste vom Plugin empfangene Argument der Kontext des Ausführenden ist;
  • Zeile 14: Liste der Schlüssel für das [context]-Objekt;
  • Zeile 16: Wir werden sehen, dass das [context]-Objekt eine [app]-Eigenschaft besitzt, die die [nuxt]-Anwendung repräsentiert;
  • Zeile 18: Wir sehen, dass das [context]-Objekt eine [route]-Eigenschaft hat, die die aktuelle Route des Routers darstellt;
  • Zeile 21: Wir werden feststellen, dass das [app]-Objekt eine [router]-Eigenschaft hat, die den Router repräsentiert;
  • Zeile 23: Das Objekt [router.options.routes] repräsentiert die verschiedenen Routen der Anwendung;
  • Zeilen 27–28: Das zweite Argument des Plugins ist die Funktion [inject], die wir im Beispiel [nuxt-06] verwendet haben;

10.3. Das vom Server ausgeführte Plugin

Bei der Ausführung zeigt der Server Folgendes an:

[main server], process.server= true process.client= false
[main server], il y a 2 arguments
Liste des clés [[main server], context]
[
  'isStatic',
  'isDev',
  'isHMR',
  'app',
  'payload',
  'error',
  'base',
  'env',
  'req',
  'res',
  'ssrContext',
  'redirect',
  'beforeNuxtRender',
  'route',
  'next',
  '_redirected',
  '_errored',
  'params',
  'query',
  '$axios'
]
  • Zeilen 11–12: Wir hatten bereits Gelegenheit, die Eigenschaften [base] und [env] zu verwenden, deren Werte aus der Datei [nuxt.config.js] stammen;
  • Zeile 8: Die Eigenschaft [app] bezieht sich auf die [nuxt]-Anwendung;
  • Zeile 18: Die Eigenschaft [route] bezieht sich auf die aktuelle Route des Routers, d. h. die Seite, die der Server senden wird;
  • Zeile 13: die HTTP-Anfrage vom Client-Browser;
  • Zeile 14: die HTTP-Antwort des Servers;

Die Liste der [context.app]-Eigenschaften lautet wie folgt:

Liste des clés [[main server], context.app]
[
  'router',
  'nuxt',
  'head',
  'render',
  'data',
  'beforeCreate',
  'created',
  'mounted',
  'watch',
  'computed',
  'methods',
  'components',
  'context',
  '$axios'
]
  • Zeile 3: Die Eigenschaft [router] ermöglicht uns den Zugriff auf den Router der Anwendung. Dies ist in [Nuxt] wichtig, da der Router von [Nuxt] selbst und nicht vom Entwickler definiert wird. Diese Eigenschaft ermöglicht es dem Entwickler, den Router zu ändern;

Die Liste der [context.route]-Eigenschaften lautet wie folgt:

Liste des clés [[main server], context.route]
[
  'name',
  'meta',
  'path',
  'hash',
  'query',
  'params',
  'fullPath',
  'matched'
]

Die Route [context.route] beim Start des Servers lautet wie folgt:

[main server], context.route= {
  name: 'index',
  meta: [
    {}
  ],
  path: '/',
  hash: '',
  query: {},
  params: {},
  fullPath: '/',
  matched: [
    {
      path: '',
      regex: /^(?:\/(?=$))?$/i,
      components: [Object],
      instances: {},
      name: 'index',
      parent: undefined,
      matchAs: undefined,
      redirect: undefined,
      beforeEnter: undefined,
      meta: {},
      props: {}
    }
  ]
}
  • Zeile 2: Wir sehen, dass die nächste Seite auf dem Server [index] ist und dass ihr Pfad [/] lautet (Zeile 10);
  • Zeile 22: Mit der Eigenschaft [meta] können Sie Routen Eigenschaften hinzufügen;

Die Eigenschaften von [context.app.router] des Servers lauten wie folgt:

Liste des clés [[main server], context.app.router]
[
  'app',
  'apps',
  'options',
  'beforeHooks',
  'resolveHooks',
  'afterHooks',
  'matcher',
  'fallback',
  'mode',
  'history'
]

Die verschiedenen Routen der Anwendung befinden sich in der Eigenschaft [context.app.router.options.routes]:

Liste des clés [[main server], context.app.router.options.routes]
[
  '0',
  '1',
  '2'
]
[main server], context.app.router.options.routes= [
  {
    path: '/page1',
    component: [Function: _d7b6c762],
    name: 'page1'
  },
  {
    path: '/page2',
    component: [Function: _d79a9860],
    name: 'page2'
  },
  {
    path: '/',
    component: [Function: _31eaad9f],
    name: 'index'
  }
]

Und schließlich das zweite Argument:

inject= function

10.4. Die [index]-Seite des Servers

Die [index]-Seite sieht wie folgt aus:


<!-- page principale -->
<template>
  <Layout :left="true" :right="true">
    <!-- navigation -->
    <Navigation slot="left" />
    <!-- message-->
    <b-alert slot="right" show variant="warning">
      Home
    </b-alert>
  </Layout>
</template>
 
<script>
/* eslint-disable no-undef */
/* eslint-disable no-console */
/* eslint-disable nuxt/no-env-in-hooks */
 
import Navigation from '@/components/navigation'
import Layout from '@/components/layout'
 
export default {
  name: 'Home',
  // components used
  components: {
    Layout,
    Navigation
  },
  data() {
    return {
      who: process.server ? 'server' : 'client'
    }
  },
 
  // life cycle
  beforeCreate() {
    // client and server
    console.log('[home beforeCreate]')
  },
  created() {
    // client and server
    console.log('[home ' + this.who + ' created]')
    this.dumpkeys('[home ' + this.who + ' created], this.$nuxt', this.$nuxt)
    this.dumpkeys('[home ' + this.who + ' created], this.$nuxt.context', this.$nuxt.context)
  },
  beforeMount() {
    // customer only
    console.log('[home ' + this.who + ' beforeMount]')
  },
  mounted() {
    // customer only
    console.log('[home ' + this.who + ' mounted]')
  },
  methods: {
    dumpkeys(message, object) {
      // list of [object] keys
      const ligne = 'Liste des clés [' + message + ']'
      console.log(ligne)
      if (object) {
        console.log(Object.keys(object))
      }
    }
  }
}
</script>
  • Zeile 43: Wir haben bereits gesehen, dass der Kontext einer Seite in [this.$nuxt.context] zu finden ist;
  • Zeile 42: Wir zeigen auch die Eigenschaften des Objekts [this.$nuxt] an;

Bei der Ausführung durch den Server generiert diese Seite die folgenden Protokolleinträge:

[home beforeCreate]
[home server created]
Liste des clés [[home server created], this.$nuxt]
[
  '_uid',
  '_isVue',
  '$options',
  '_renderProxy',
  '_self',
  '$parent',
  '$root',
  '$children',
  '$refs',
  '_watcher',
  '_inactive',
  '_directInactive',
  '_isMounted',
  '_isDestroyed',
  '_isBeingDestroyed',
  '_events',
  '_hasHookEvent',
  '_vnode',
  '_staticTrees',
  '$vnode',
  '$slots',
  '$scopedSlots',
  '_c',
  '$createElement',
  '$attrs',
  '$listeners',
  '_routerRoot',
  '_router',
  '_route',
  '_bv__modal',
  '_bv__toast',
  '_vueMeta',
  'nuxt',
  '_watchers',
  'refreshOnlineStatus',
  'refresh',
  'errorChanged',
  'setLayout',
  'loadLayout',
  '_data',
  'layoutName',
  'layout',
  'isOnline',
  '_computedWatchers',
  'isOffline',
  'error',
  'context'
]

Die Serverkontext-Eigenschaften auf der [index]-Seite lauten wie folgt:

Liste des clés [[home server created], this.$nuxt.context]
[
  'isStatic',
  'isDev',
  'isHMR',
  'app',
  'payload',
  'error',
  'base',
  'env',
  'req',
  'res',
  'ssrContext',
  'redirect',
  'beforeNuxtRender',
  'route',
  'next',
  '_redirected',
  '_errored',
  'params',
  'query',
  '$axios'
]

Dies sind dieselben Eigenschaften wie im [context]-Objekt des Plugins.

10.5. Das vom Client ausgeführte Plugin

Sobald der Server die [index]-Seite an den Browser des Clients gesendet hat, übernehmen die clientseitigen Skripte. Das [main.js]-Plugin wird dann ausgeführt. Diese Protokolle lauten wie folgt:

[main server], process.server= false process.client= true
[main client], il y a 2 arguments
Liste des clés [[main client], context]
Array(17)0: "isStatic"1: "isDev"2: "isHMR"3: "app"4: "payload"5: "error"6: "base"7: "env"8: "redirect"9: "nuxtState"10: "route"11: "next"12: "_redirected"13: "_errored"14: "params"15: "query"16: "$axios"length: 17__proto__: Array(0)
Liste des clés [[main client], context.app]
Array(14)0: "router"1: "nuxt"2: "head"3: "render"4: "data"5: "beforeCreate"6: "created"7: "mounted"8: "watch"9: "computed"10: "methods"11: "components"12: "context"13: "$axios"length: 14__proto__: Array(0)
Liste des clés [[main client], context.route]
Array(8)0: "name"1: "meta"2: "path"3: "hash"4: "query"5: "params"6: "fullPath"7: "matched"length: 8__proto__: Array(0)
[main client], context.route= ObjectfullPath: "/"hash: ""matched: [{…}]meta: [{…}]name: "index"params: {}path: "/"query: {}__proto__: Object
Liste des clés [[main client], context.app.router]
Array(10)0: "app"1: "apps"2: "options"3: "beforeHooks"4: "resolveHooks"5: "afterHooks"6: "matcher"7: "fallback"8: "mode"9: "history"length: 10__proto__: Array(0)
Liste des clés [[main client], context.app.router.options.routes]
Array(3)0: "0"1: "1"2: "2"length: 3__proto__: Array(0)
[main client], context.app.router.options.routes= Array(3)0: {path: "/page1", name: "page1", component: ƒ}1: {path: "/page2", name: "page2", component: ƒ}2: {path: "/", name: "index", component: ƒ}length: 3__proto__: Array(0)
inject= function

Wir finden Eigenschaften, die denen auf der Serverseite ähneln, wobei einige Eigenschaften fehlen und andere hinzukommen. In Zeile 4 fehlen beispielsweise die Eigenschaften [req, res] – die HTTP-Anfragen des Client-Browsers und die HTTP-Antwort des Servers.

10.6. Die [index]-Seite des Clients

Die [index]-Seite des Clients erzeugt die folgenden Protokolle:

1
2
3
4
5
6
7
8
[home beforeCreate]
[home client created]
Liste des clés [[home client created], this.$nuxt]
(51) ["_uid", "_isVue", "$options", "_renderProxy", "_self", "$parent", "$root", "$children", "$refs", "_watcher", "_inactive", "_directInactive", "_isMounted", "_isDestroyed", "_isBeingDestroyed", "_events", "_hasHookEvent", "_vnode", "_staticTrees", "$vnode", "$slots", "$scopedSlots", "_c", "$createElement", "$attrs", "$listeners", "_routerRoot", "_router", "_route", "_bv__modal", "_bv__toast", "_vueMeta", "nuxt", "_watchers", "refreshOnlineStatus", "refresh", "errorChanged", "setLayout", "loadLayout", "_data", "layoutName", "layout", "isOnline", "_computedWatchers", "isOffline", "error", "context", "_name", "setTransitions", "$loading", "$el"]
Liste des clés [[home client created], this.$nuxt.context]
(17) ["isStatic", "isDev", "isHMR", "app", "payload", "error", "base", "env", "redirect", "nuxtState", "route", "next", "_redirected", "_errored", "params", "query", "$axios"]
[home client beforeMount]
[home client mounted]
  • Zeile 4: die Eigenschaften des Objekts [this.$nuxt]. Es handelt sich um ein umfangreiches Objekt mit 51 Eigenschaften;
  • Zeile 6: die Eigenschaften des Objekts [this.$nuxt.context]. Dies sind dieselben Eigenschaften, die im Objekt [context] des Client-Plugins zu finden sind;