HTML5-Tutorium: JavaScript: Hello World Vue 03: Unterschied zwischen den Versionen

aus GlossarWiki, der Glossar-Datenbank der Fachhochschule Augsburg
Kowa (Diskussion | Beiträge)
Kowa (Diskussion | Beiträge)
 
(15 dazwischenliegende Versionen desselben Benutzers werden nicht angezeigt)
Zeile 78: Zeile 78:


window.addEventListener
window.addEventListener
( 'load'
( 'load',
   createApp(App).use(pinia).mount('#app')
   createApp(App).use(pinia).mount('#app')
)
)
Zeile 94: Zeile 94:
import { defineStore } from 'pinia'
import { defineStore } from 'pinia'
import { ref }        from 'vue'
import { ref }        from 'vue'
const storeSection =
const storeSection =
defineStore
defineStore
Zeile 107: Zeile 108:
         currentSection.value = p_section,
         currentSection.value = p_section,


       visability =
       visibility =
         p_section =>
         p_section =>
         currentSection.value !== p_section ? 'hidden' : '',
         currentSection.value !== p_section ? 'hidden' : '',
Zeile 114: Zeile 115:
         init // change macht dasselbe wie init
         init // change macht dasselbe wie init


     return { init, visability, change }
     return { init, visibility, change }
   }
   }
)
)
Zeile 138: Zeile 139:
     () =>
     () =>
     { const
     { const
      state =
        state =
        reactive({ name: '', stranger: 'Stranger', hello: 'Hello, $1!' }),
          reactive({ name: '', stranger: 'Stranger', hello: 'Hello, $1!' }),


      init =
        init =
        (p_state = {}) =>
          (p_state = {}) =>
        { Object.assign(state, p_state) },
          { Object.assign(state, p_state) },


      name = toRef(state, 'name'),
        name =  
          toRef(state, 'name'),


      sayHello =
        sayHello =
        p_name => state.hello.replace('$1', p_name),
          p_name => state.hello.replace('$1', p_name),


      helloStranger =
        helloStranger =
        computed(() => sayHello(state.stranger)),
          computed(() => sayHello(state.stranger)),


      hello =
        hello =
        computed(() => sayHello(state.name))
          computed(() => sayHello(state.name))


       return { init, name, helloStranger, hello }
       return { init, name, helloStranger, hello }
     }
     }
)
  )


export default storeGreeting
export default storeGreeting
Zeile 170: Zeile 172:
<script setup>
<script setup>
   import { onMounted, onUnmounted } from 'vue'
   import { onMounted, onUnmounted } from 'vue'
  import ControllerKey from '@/controller/ControllerKey'
   import storeSection  from '@/store/StoreSection'
   import storeSection  from '@/store/StoreSection'
   import storeGreeting from '@/store/StoreGreeting'
   import storeGreeting from '@/store/StoreGreeting'
  import ControllerKey from '@/controller/ControllerKey'


   const
   const
Zeile 178: Zeile 180:
     greeting = storeGreeting(),
     greeting = storeGreeting(),


     sayHello = () => { section.change('hello') },
     sayHello     = () => section.change('hello'),
     controllerKey =
     controllerKey = new ControllerKey('Enter', sayHello, 'input_name')
      new ControllerKey('Enter', sayHello, 'input_name')


   section.init('form')
   section.init('form')
//greeting.init({ hello: 'Hallo, $1!', stranger: 'Fremder'})
//greeting.init({ hello: 'Hi, $1!', stranger: 'Nobody'})


   onMounted  (() => controllerKey.add())
   onMounted  (() => controllerKey.add())
Zeile 203: Zeile 204:


     hello        = computed(() => `Hello, ${name.value}!`),
     hello        = computed(() => `Hello, ${name.value}!`),
     visability   = p_section =>
     visibility   = p_section =>
                     p_section !== section.value ? 'hidden' : '',
                     p_section !== section.value ? 'hidden' : '',
     sayHello      = () => { section.value = 'hello' },
     sayHello      = () => section.value = 'hello',
     autofocus    = () => { document.getElementById('input_name').focus() },
     autofocus    = () => document.getElementById('input_name').focus(),
     controllerKey = new ControllerKey('Enter', sayHello, 'input_name')
     controllerKey = new ControllerKey('Enter', sayHello, 'input_name')


   onMounted  (...)
   onMounted  (...)
Zeile 223: Zeile 223:


<template>
<template>
   <section id="section_question" v-bind:class="section.visability('question')">
   <section id="section_form" v-bind:class="section.visibility('form')">
     <h1>{{greeting.helloStranger}}</h1>
     <h1>{{greeting.helloStranger}}</h1>
     <form>
     <form>
       <div>
       <div>
         <label for="input_name">What's your name?</label>
         <label for="input_name">What is your name?</label>
         <input id="input_name" type="text" autofocus
         <input id="input_name" type="text" autofocus
               v-model="greeting.name"
               v-model="greeting.name"
Zeile 240: Zeile 240:
     </form>
     </form>
   </section>
   </section>
   <section id="section_hello" v-bind:class="section.visability('hello')">
   <section id="section_hello" v-bind:class="section.visibility('hello')">
     <h1>{{greeting.hello}}</h1>
     <h1>{{greeting.hello}}</h1>
     <p>Welcome to Web Programming!</p>
     <p>Welcome to Web Programming!</p>
   </section>
   </section>
</template>
</template>
</source>
== Stores als Klassen ==
Stores können auch mit Hilfe von Klassen definiert werden. Dazu benötigt man zunächst eine Basisklasse, die von jedem Store wiederverwendet werden kann.
=== Basisklasse <code>Store</code> ===
<source lang="ecmascript">
// src/common/Store/Store.js
// cmp. https://medium.com/@mario.brendel1990/vue-3-the-new-store-a7569d4a546f
import { reactive } from 'vue'
class Store
{ #initState
  state
  reset()
  { if (this.state == null)
    { this.state = reactive({...this.initState}) }
    else
    { Object.assign(this.state, this.initState) }
  }
  init(p_state)
  { Object.assign(this.initState, p_state);
    this.reset();
  }
  constructor(p_state)
  { this.initState = p_state;
    this.reset();
  }
}
export default Store
</source>
=== Definition der Store-Klasse StoreSection ===
Nun kann der Section-Store als Klasse realisiert werden.
<source lang="ecmascript">
// src/store/StoreSection.js
import Store from '@/common/store/Store'
class StoreSection extends Store
{ constructor()
  { super({ name: '' }) }
  visability(p_section)
  { return this.state.name !== p_section ? 'hidden' : '' }
  change(p_section)
  { this.state.name = p_section }
}
const storeSection = new StoreSection();
export default storeSection;
</source>
=== Definition der Store-Klasse StoreGreeting ===
Den Greeting-Store kann man analog definieren.
<source lang="ecmascript">
// src/store/StoreGreeting.js
import Store        from '@/common/store/Store'
import { computed } from 'vue'
class StoreGreeting extends Store
{ constructor(p_init = { hello: 'Hello, $1!', stranger: 'Stranger' })
  { super({ name: '', ...p_init }); }
  get name()
  { return this.state.name; }
  set name(p_name)
  { this.state.name = p_name; }
  #sayHello(p_name)
  { return this.state.hello.replace('$1', p_name); }
  get helloStranger()
  { return computed(() => this.#sayHello(this.state.stranger)).value }
  get hello()
  { return computed(() => this.#sayHello(this.state.name !== ''
                                        ? this.state.name
                                        : this.state.stranger
                                        )
                  ).value
  }
}
const stateGreeting = new StoreGreeting()
export default stateGreeting
</source>
Störend ist hier, dass man für jedes Attribut wie {{zB}} <code>name</code> eine Getter- und eine Setter-Methode definieren muss.
Dieses Problem kann man mit einer weiteren Superklasse abmildern.
=== StoreExtendible ===
Man definiert die Klasse <code>StoreExtendible</code>, die es ermöglicht, bei der Initialisierung des Stores dynamisch reaktive Getter- und Setter-Methoden hinzuzufügen.
<source lang="ecmascript">
// src/common/Store/StoreExtendible.js
import Store from './Store'
class StoreExtendible extends Store
{ #add(p_properties, p_getter, p_setter)
  { if (typeof p_properties === 'string')
    { Object.defineProperty
      ( this.constructor.prototype,
        p_properties,
        { ...( p_getter ? { get: () => this.state[p_properties] } : {} ),
          ...( p_setter ? { set: (value) =>
                                { if (this.state[p_properties] !== value)
                                  { this.state[p_properties] = value; }
                                },
                        }
                      : {}
            ),
          configurable: false, // the property cannot be redefined
          enumerable:  true
        },
      );
    }
    else
    { p_properties.forEach(e => this.addGetter(e)) }
  }
  addGetter(p_properties)
  { this.#add(p_properties, true, false) }
  addSetter(p_properties)
  { this.#add(p_properties, false, true) }
  addGetterSetter(p_properties)
  { this.#add(p_properties, true, true) }
}
export default StoreExtendible
</source>
=== Verwendung der alternativen Stores in einer Komponente ===
<source lang="ecmascript">
<!-- src/components/HelloWorld.vue with store class -->
<script setup >
  import { onMounted, onUnmounted } from 'vue'
  import ControllerKey from '@/controller/ControllerKey'
  import section  from '@/store/StoreSectionClass'
//import greeting from '@/store/StoreGreetingClass'
  import greeting from '@/store/StoreGreetingClassExtendible'
  const
    sayHello = () => { section.change('hello') },
    controllerKey =
      new ControllerKey('Enter', sayHello, 'input_name')
  section.init({ name: 'question'})
//greeting.init({ hello: 'Hallo, $1!', stranger: 'Fremder'})
  onMounted  (() => controllerKey.add())
  onUnmounted(() => controllerKey.remove())
</script>
</source>
Zum Vergleich: Die Verwendung der zuerst definierten Stores.
<source lang="ecmascript">
<!-- src/components/HelloWorld.vue -->
<script setup>
  import { onMounted, onUnmounted } from 'vue'
  import ControllerKey from '@/controller/ControllerKey'
  import storeSection  from '@/store/StoreSection'
  import storeGreeting from '@/store/StoreGreeting'
  const
    section  = storeSection(),
    greeting = storeGreeting(),
    sayHello = () => { section.change('hello') },
    controllerKey =
      new ControllerKey('Enter', sayHello, 'input_name')
  section.init('question')
//greeting.init({ hello: 'Hallo, $1!', stranger: 'Fremder'})
  onMounted  (() => controllerKey.add())
  onUnmounted(() => controllerKey.remove())
</script>
</source>
</source>



Aktuelle Version vom 30. April 2024, 13:58 Uhr

Dieser Artikel erfüllt die GlossarWiki-Qualitätsanforderungen nur teilweise:

Korrektheit: 3
(zu größeren Teilen überprüft)
Umfang: 4
(unwichtige Fakten fehlen)
Quellenangaben: 3
(wichtige Quellen vorhanden)
Quellenarten: 5
(ausgezeichnet)
Konformität: 3
(gut)

Vorlesung WebProg

Inhalt | Teil 1 | Teil 2 | Teil 3 | Teil 4 | Teil 5 | Teil 6 | Vue 1 | Vue 2 | Vue 3 | Vue 4 | Vue 5 | Vue 6

Musterlösung
Git-Repository, git checkout v03 (JavaScript)
Git-Repository (TypeScript), git checkout v03

Stores mit Hilfe von Klassen:
Git-Repository, git checkout v03b (JavaScript)
Git-Repository (TypeScript), git checkout v03b

Anwendungsfälle (Use Cases)

Gegenüber dem ersten und zweiten Teil des Vue-Tutoriums ändern sich die die Anwendungsfälle nicht. Die Anwendung leistet also genau dasselbe wie zuvor.

Aufgabe

Aufgabe: Speichere den Zustand der Anwendung (Model) in Store-Objekten, um die künftige Modularisierung (Version v04) zu unterstützen.

In diesem Tutorium sollen die Daten in einem Store außerhalb der Komponenten gespeichert werden.

Mögliche Storekonzepte

  • vuex (veraltet, wird nur noch gewartet, angeblich nicht mehr weiterentwickelt, aber Version 4 existiert)
  • Pinia (der Nachfolger von vuex; einfacher zu benutzen; Vue-3- und Vue-2-Paradigmen werden unterstützt)
  • ohne Bibliothek direkt mit ref, reactive und computed aus Vue 3.2 (vgl. Vue 3 — The New Store)

Wir verwenden Pinia.

Pinia

Zitat:

While our hand-rolled state management solution will suffice in simple scenarios, there are many more things to consider in large-scale production applications:

  1. Stronger conventions for team collaboration
  2. Integrating with the Vue DevTools, including timeline, in-component inspection, and time-travel debugging
  3. Hot Module Replacement
  4. Server-Side Rendering support

Pinia is a state management library that implements all of the above. It is maintained by the Vue core team, and works with both Vue 2 and Vue 3.

Existing users may be familiar with Vuex, the previous official state management library for Vue. With Pinia serving the same role in the ecosystem, Vuex is now in maintenance mode. It still works, but will no longer receive new features. It is recommended to use Pinia for new applications.

Pinia started out as an exploration of what the next iteration of Vuex could look like, incorporating many ideas from core team discussions for Vuex 5. Eventually, we realized that Pinia already implements most of what we wanted in Vuex 5, and decided to make it the new recommendation instead.

Compared to Vuex, Pinia provides a simpler API with less ceremony, offers Composition-API-style APIs, and most importantly, has solid type inference support when used with TypeScript.

Erstellen eines neuen Projektzweigs

Erstellen Sie einen neuen Projektzweig (branch) innerhalb von hello_world_vue:

git checkout v02     # Wechsle in den Branch v02
git checkout -b v03  # Klone v02 in einen neuen Branch v03
npm i

Initialisierung von Pinia

Pinia muss als reguläre Dependency installiert werden, da das Paket in den produktiven Code integriert wird.

npm i pinia  // NICHT  npm i -D pinia

Pinia wird als Plugin in die App integriert. Dabei ist wichtig, dass das Pina-Plugin-Objekt vor dem eigentlichen App-Objekt erzeugt wird, da Komponenten der App, die Pinia verwenden, darauf angewiesen sind, dass die Pinia-Erzeugung vor der Komponenten-Initialisierung erfolgt.

// src/main.js

import { createPinia } from 'pinia'
import { createApp }   from 'vue'
import App             from './App.vue'

const pinia = createPinia() // before createApp(...)

window.addEventListener
( 'load',
  createApp(App).use(pinia).mount('#app')
)

Erstellen der Stores

StoreSection

Im Section-Store wird gespeichert, welche Section der Single-Page-Anwendung dem Benutzer aktuell präsentiert wird.

// src/store/StoreSection.js

import { defineStore } from 'pinia'
import { ref }         from 'vue'

const storeSection =
defineStore
( 'section', // must be unique

  () =>
  { const
      currentSection =
        ref(''),
   
      init =
        p_section =>
        currentSection.value = p_section,

      visibility =
        p_section =>
        currentSection.value !== p_section ? 'hidden' : '',

      change =
        init // change macht dasselbe wie init

    return { init, visibility, change }
  }
)

export default storeSection

StoreGreeting

Im Greeting-Store werden die Daten zur Begrüßung des Benutzers gespeichert.

// src/store/StoreGreeting.js

import { defineStore }               from 'pinia'
import { reactive, toRef, computed } from 'vue'

const
  storeGreeting =
  defineStore
  ( 'greeting',

    () =>
    { const
        state =
          reactive({ name: '', stranger: 'Stranger', hello: 'Hello, $1!' }),

        init =
          (p_state = {}) =>
          { Object.assign(state, p_state) },

        name = 
          toRef(state, 'name'),

        sayHello =
          p_name => state.hello.replace('$1', p_name),

        helloStranger =
          computed(() => sayHello(state.stranger)),

        hello =
          computed(() => sayHello(state.name))

      return { init, name, helloStranger, hello }
    }
  )

export default storeGreeting

Einbinden der Stores in einer Komponente

<!-- src/components/HelloWorld.vue -->

<script setup>
  import { onMounted, onUnmounted } from 'vue'
  import storeSection  from '@/store/StoreSection'
  import storeGreeting from '@/store/StoreGreeting'
  import ControllerKey from '@/controller/ControllerKey'

  const
    section  = storeSection(),
    greeting = storeGreeting(),

    sayHello      = () => section.change('hello'),
    controllerKey = new ControllerKey('Enter', sayHello, 'input_name')

  section.init('form')
//greeting.init({ hello: 'Hi, $1!', stranger: 'Nobody'})

  onMounted  (() => controllerKey.add())
  onUnmounted(() => controllerKey.remove())
</script>

Zum Vergleich noch einmal das Skript ohne Store

<!-- src/components/HelloWorld.vue without store -->
<script setup >
  import { ref, computed, onMounted, onUnmounted } from 'vue'
  import ControllerKey from '@/controller/ControllerKey'

  const
    section       = ref('form'),
    helloStranger = ref('Hello, Stranger'),
    name          = ref(''),

    hello         = computed(() => `Hello, ${name.value}!`),
    visibility    = p_section =>
                    p_section !== section.value ? 'hidden' : '',
    sayHello      = () => section.value = 'hello',
    autofocus     = () => document.getElementById('input_name').focus(),
    controllerKey = new ControllerKey('Enter', sayHello, 'input_name')

  onMounted  (...)
  onUnmounted(...)
</script>

Anpassung des Templates

Das Template muss angepasst werden. Man muss nun auf die Store-Objekte zugreifen.

<!-- src/components/HelloWorld.vue -->

<template>
  <section id="section_form" v-bind:class="section.visibility('form')">
    <h1>{{greeting.helloStranger}}</h1>
    <form>
      <div>
        <label for="input_name">What is your name?</label>
        <input id="input_name" type="text" autofocus
               v-model="greeting.name"
        />
      </div>
      <div>
        <input id="button_reset"  type="reset"  value="Reset"/>
        <input id="button_submit" type="button" value="Say hello"
               v-on:click="sayHello"
        />
      </div>
    </form>
  </section>
  <section id="section_hello" v-bind:class="section.visibility('hello')">
    <h1>{{greeting.hello}}</h1>
    <p>Welcome to Web Programming!</p>
  </section>
</template>

Fortsetzung des Tutoriums

Sie sollten nun Teil 4 des Vue-Tutoriums bearbeiten. Nun wird die Anwendung modularisiert.

Quellen

  1. Kowarschick (WebProg): Wolfgang Kowarschick; Vorlesung „Web-Programmierung“; Hochschule: Hochschule Augsburg; Adresse: Augsburg; Web-Link; 2024; Quellengüte: 3 (Vorlesung)