HTML5-Tutorium: JavaScript: Hello World 04: Unterschied zwischen den Versionen

aus GlossarWiki, der Glossar-Datenbank der Fachhochschule Augsburg
(36 dazwischenliegende Versionen desselben Benutzers werden nicht angezeigt)
Zeile 1: Zeile 1:
{{In Bearbeitung}}
{{HTML5-Tutorium:JavaScript:HelloWorld:Menü}}
{{HTML5-Tutorium:JavaScript:HelloWorld:Menü}}


'''Musterlösung''': [https://glossar.hs-augsburg.de/beispiel/tutorium/2021/hello_world/WK_HelloWorld04/web/index1.html <code>index1.html</code>], [https://glossar.hs-augsburg.de/beispiel/tutorium/2021/hello_world/WK_HelloWorld04/web/index2.html <code>index2.html</code>]  
'''Musterlösung''': [https://glossar.hs-augsburg.de/beispiel/tutorium/2021/hello_world/WK_HelloWorld04a/web/index0.html <code>index0.html</code>],  [https://glossar.hs-augsburg.de/beispiel/tutorium/2021/hello_world/WK_HelloWorld04a/web/index1.html <code>index1.html</code>], [https://glossar.hs-augsburg.de/beispiel/tutorium/2021/hello_world/WK_HelloWorld04a/web/index2.html <code>index2.html</code>]  
([https://gitlab.multimedia.hs-augsburg.de/kowa/WK_HelloWorld04a.git Git-Repository ])
([https://gitlab.multimedia.hs-augsburg.de/kowa/WK_HelloWorld04a.git Git-Repository v04a]),
[https://glossar.hs-augsburg.de/beispiel/tutorium/2021/hello_world/WK_HelloWorld04b/web/index.html <code>index.html</code>]
([https://gitlab.multimedia.hs-augsburg.de/kowa/WK_HelloWorld04b.git Git-Repository v04b]),
[https://glossar.hs-augsburg.de/beispiel/tutorium/2021/hello_world/WK_HelloWorld04c/web/index.html <code>index.html</code>]
([https://gitlab.multimedia.hs-augsburg.de/kowa/WK_HelloWorld04c.git Git-Repository v04c])


==Anwendungsfälle (Use Cases)==
==Anwendungsfälle (Use Cases)==
Zeile 9: Zeile 12:
ändern sich die die Anwendungsfälle nicht. Die Anwendung leistet also genau dasselbe wie zuvor.
ändern sich die die Anwendungsfälle nicht. Die Anwendung leistet also genau dasselbe wie zuvor.


In diesem Teil des Tutoriums geht es darum, die Anwendung besser zu strukturieren, {{dh}} zu [[Modul|modularisieren]].
In diesem Teil des Tutoriums geht es darum, die Anwendung besser zu strukturieren, {{dh}} zu [[Modul|modularisieren]] (vgl. [[Programmierprinzipien#Modularit.C3.A4t.2C_Modularity.2C_Teile_und_herrsche.2C_Divide_et_impera|Programmierprinzipien]]).


==Modularisierung==
==Modularisierung==
Zeile 32: Zeile 35:
==Erstellen eines neuen Projektes==
==Erstellen eines neuen Projektes==


Erstellen Sie ein neues Projekt <code>HelloWorld04</code> als Fork von <code>HelloWorld03</code> (siehe [[HTML5-Tutorium: JavaScript: Hello World 02|Hello-World-02-Tutorium]]).
Erstellen Sie ein neues Projekt <code>HelloWorld04</code> als Fork von <code>HelloWorld03</code> (siehe [[HTML5-Tutorium: JavaScript: Hello World 02|Hello-World-02-Tutorium]]):
 
<source lang="bash">
git clone https://gitlab.multimedia.hs-augsburg.de/ACCOUNT/HelloWorld03 HelloWorld04
//git clone https://gitlab.multimedia.hs-augsburg.de/kowa/WK_HelloWorld03 HelloWorld04
 
## Änderungen zur Projektidentifikation vornehmen (z.B. Titel anpassen)
git commit -m "HelloWorld03 fork created"
 
# Neues Repository in Gitlab anlegen:
git remote -v
git remote remove origin
git remote add origin https://gitlab.multimedia.hs-augsburg.de/ACCOUNT/HelloWorld04
git remote -v
git push --set-upstream origin master
</source>


==Erstellen einer Ordnerstruktur==
==Erstellen einer Ordnerstruktur==
Zeile 81: Zeile 99:


===CSS-Dateien===
===CSS-Dateien===
Damit das Problem mit dem falschen Layout nicht auftritt, empfiehlt Google, ganz wenige, wichtige CSS-Befehle direkt – {{dh}} als CSS-Befehle innerhalb eines <code>style</code>-Elements – in dem HTML-Head-Bereich einzufügen. Alle anderen CSS-Anweisungen sollen erst am Ende der HTML-Datei dynamisch mittels JavaScript  
Damit das Problem mit dem falschen Layout nicht auftritt, empfiehlt Google ([https://developers.google.com/speed/pagespeed/insights/ PageSpeed Insights]),  
(oder mittels des PageSpeed-Optimization-Modul von Google) gelesen werden.
ganz wenige, wichtige CSS-Befehle direkt – {{dh}} als CSS-Befehle innerhalb eines <code>style</code>-Elements – in dem HTML-Head-Bereich einzufügen. Alle anderen CSS-Anweisungen sollen erst am Ende der HTML-Datei dynamisch mittels JavaScript  
(oder mittels des PageSpeed-Optimization-Modul von Google) gelesen werden.


[https://developers.google.com/speed/pagespeed/insights/?url=https%3A%2F%2Fglossar.hs-augsburg.de%2Fbeispiel%2Ftutorium%2F2021%2Fhello_world%2FWK_HelloWorld04%2Fweb%2Findex1.html PageSpeed Insights (für Musterlösung index1.html)]:
[https://developers.google.com/speed/pagespeed/insights/?url=https%3A%2F%2Fglossar.hs-augsburg.de%2Fbeispiel%2Ftutorium%2F2021%2Fhello_world%2FWK_HelloWorld04%2Fweb%2Findex0.html PageSpeed Insights (für Musterlösung index0.html)]:
<blockquote cite="https://developers.google.com/speed/pagespeed/insights/?url=https%3A%2F%2Fglossar.hs-augsburg.de%2Fbeispiel%2Ftutorium%2F2021%2Fhello_world%2FWK_HelloWorld04%2Fweb%2Findex1.html">  
<blockquote cite="https://developers.google.com/speed/pagespeed/insights/?url=https%3A%2F%2Fglossar.hs-augsburg.de%2Fbeispiel%2Ftutorium%2F2021%2Fhello_world%2FWK_HelloWorld04%2Fweb%2Findex1.html">  
Empfehlung
Empfehlung
Zeile 97: Zeile 116:
Legen Sie die Datei <code>src/css/head.css</code> an und fügen Sie folgenden (sehr kurzen!) Code ein:
Legen Sie die Datei <code>src/css/head.css</code> an und fügen Sie folgenden (sehr kurzen!) Code ein:


<source lang="javascript">
<source lang="css">
html, body
body
{ background-color: #C5EFFC; }
{ background-color: #C5EFFC; }


Zeile 107: Zeile 126:
In der Datei <code>src/index.html</code> müssen Sie das Link-Element in Header folgendermaßen
In der Datei <code>src/index.html</code> müssen Sie das Link-Element in Header folgendermaßen
<source lang="html">
<source lang="html">
<link rel="stylesheet" type="text/css" href="css/head.css"/>
<link rel="stylesheet" href="css/head.css"/>
</source>
</source>
anpassen. Das heißt, Sie laden im HTML-Head-Bereich nur noch die kleine CSS-Datei.
anpassen. Das heißt, Sie laden im HTML-Head-Bereich nur noch die kleine CSS-Datei.
Zeile 115: Zeile 134:


<source lang="html">
<source lang="html">
html, body
<style>
body
{ background-color: #C5EFFC; }
{ background-color: #C5EFFC; }


Zeile 135: Zeile 155:
<source lang="html">
<source lang="html">
<link rel="stylesheet" href="css/body.css"
<link rel="stylesheet" href="css/body.css"
       type="text/css" media="none" onload="this.media='all'"  
       media="none" onload="this.media='all'"  
/>
/>
</source>
</source>
Zeile 151: Zeile 171:
* Dies Datei wird mittels eines Hacks asynchron, {{dh}} parallel zum Body-Element geladen.
* Dies Datei wird mittels eines Hacks asynchron, {{dh}} parallel zum Body-Element geladen.


===JavaScript-Dateien===
===<code>index.html</code>===
 
Da die Datei <code>head.css</code> nur die Hintergrundfarbe sowie die <code>.hidden</code>-Regel enthält, sollten alle HTML-Elemente, die gerendert werden sollen, zunächst unsichtbar gemacht werden. Bei einer Web-Anwendung, wie einem Spiel, die zum Laden länger braucht, könnte man eine Sanduhr oder einen Ladebalken einfügen, um anzuzeigen, dass man noch etwas warten muss. Üblicherweise, sollte man dafür sorgen,
dass das Laden einer Seite so schnell geht, dass der Benutzer gar nicht erst ungeduldig wird.
 
Fügen Sie die CSS-Klasse <code>hidden</code> in beide HTML-Sektions ein:
 
<source lang="html">
<section id="section_form" class="hidden">
...
</section>
<section id="section_hello" class="hidden">
...
</section>
</source>
 
Wenn man jetzt das Dokument im Browser öffnet, sieht man nichts, außer einem hellblauen Hintergrund.
 
Abhilfe schafft hier ein kleiner JavaScript-Befehl, der <code>hidden</code> von der Eingabe-Section <code>hidden-form</code> entfernt, sobald das Dokument vollständig geladen wurde (samt allen CSS- und JavaScript-Dateien</code>):
 
<source lang="html">
document.getElementById('section_form').classList.remove('hidden');
</source>


Für JavaScript-Dateien empfiehlt Google ebenfalls, diese nicht schon zu Beginn zu laden.<ref>{{Quelle|Google (Web)}}, Kapitel „Remove Render-Blocking JavaScript“, https://developers.google.com/speed/docs/insights/BlockingJS</ref> Dieser Vorschlag war bis vor kurzem noch sehr sinnvoll. Man könnte das
An welche Stelle dieser Befehl eingefügt wird, erfahren Sie im nachfolgenden Abschnitt.
Skript-Element aus dem HTML-Header-Bereich entfernen und vor dem schließenden Body-Tag einfügen.
Bis vor Kurzem war diese Vorgehensweise auch sehr weit verbreitet.
Seitdem die meisten Browser die Async- und Defer-Attribute im Skript-Element unterstützen, ist dies
jedoch nicht mehr notwendig. Und Sie haben ja schon ein Async-Attribut in Ihr Skript-Element eingefügt.


Was Sie aber machen sollten, ist, in das erste Section-Tag im Body der HRML-Datei das Attribut
===JavaScript-Dateien===
<code>class="hidden"</code> einzufügen, damit man auch dieses Element beim Laden der Anwendung
zunächst nicht sieht. In einer Anwendung, die zum Laden etwas länger dauert, würde man
einen so genannten [[Splash Screen]] sichtbar einfügen und diesen beim eigentlichen Start der App ausblenden.


Wenn Sie nun die Web-App testweise starten, sollten Sie eine himmelblaue, aber ansonsten leere Seite sehen.  
Für JavaScript-Dateien empfiehlt Google ebenfalls, diese nicht schon zu Beginn zu laden, sondern das Laden „aufzuschieben“. Das machen wir bereits mit Hilfe des Async-Attributs.


Fügen Sie nun den Befehl
Fügen Sie nun den zuvor erwähnten Befehl
<source lang="javascript">
<source lang="javascript">
document.getElementById('section_form').classList.remove('hidden');
document.getElementById('section_form').classList.remove('hidden');
</source>
</source>
als letzen Befehl in den Rumpf der Funktion <code>init</code> in der Datei
als letzen Befehl in den Rumpf der Funktion <code>init</code> in der Datei
<code>src/js/main.js</code> ein.
<code>web/js/main.js</code> ein.


Wenn Sie jetzt die Web-App starten, sollte sie wieder funktionieren (wenn auch mit einem sehr CSS-armen Layout).
Wenn Sie jetzt die Web-App starten, sollte sie wieder funktionieren .


Was haben Sie erreicht? Solange die Anwendung geladen wird, sehen Sie nur einen blauen Hintergrund ohne Inhalt. (Besser wäre – wie gesagt – ein Splash Screen.) Wenn dann alles, {{dh}} alle CSS- und JavaScript-Dateien geladen wurden, wird die eigentliche Anwendung sichtbar. Dies ist bei großen Anwendungen,
Was haben Sie erreicht? Solange die Anwendung geladen wird, sehen Sie nur einen blauen Hintergrund ohne Inhalt, bzw. {{iAllg}} einen so genannten Splash Screen. Wenn dann alles, {{dh}} alle CSS- und JavaScript-Dateien geladen wurden, wird die eigentliche Anwendung sichtbar.  
wie einem Spiel, eine übliche Vorgehensweise.


Vergessen Sie nicht:
Vergessen Sie nicht: Commit!
* Klick auf das Commit-Icon.


===Modularisierung der JavaScript-Dateien===
===Modularisierung der JavaScript-Dateien===
Zeile 187: Zeile 220:
die zweite initialisiert die App, {{dh}} aktiviert die Begrüßungsfunktion.  
die zweite initialisiert die App, {{dh}} aktiviert die Begrüßungsfunktion.  


Erstellen Sie die JavaScript-Datei <code>src/js/greet.js</code> und verschieben Sie die
Erstellen Sie die JavaScript-Datei <code>web/js/greet.js</code> und verschieben Sie die
beiden Funktionen <code>sayHello</code> und <code>sayHelloOnEnter</code>
beiden Funktionen <code>sayHello</code> und <code>sayHelloOnEnter</code>
(jeweils samt Funktionsrumpf :-) ) in diese Datei.
(jeweils samt Funktionsrumpf :-) ) in diese Datei.
Zeile 196: Zeile 229:
Export-Anweisung könnten von anderen ES-6-Dateien nicht importiert werden.)
Export-Anweisung könnten von anderen ES-6-Dateien nicht importiert werden.)


In der Datei <code>src/js/main.js</code> werden jetzt diese beiden Funktionen importiert.
In der Datei <code>web/js/main.js</code> werden jetzt diese beiden Funktionen importiert.
Fügen Sie ganz am Anfang der Datei ({{dh}} vor der verbliebenen Init-Funktion)
Fügen Sie ganz am Anfang der Datei ({{dh}} vor der verbliebenen Init-Funktion)
die folgende Import-Anweisung ein:
die folgende Import-Anweisung ein:
Zeile 209: Zeile 242:
<code>greet.sayHello</code> und <code>greet.sayHelloOnEnter</code>.
<code>greet.sayHello</code> und <code>greet.sayHelloOnEnter</code>.


Damit die Web-Anwendung wieder funktioniert. müssen Sie also in der Datei <code>main.js</code> vor jede Verwendung dieser beiden Funktionen die Zeichfolge <code>greet.</code> einfügen. (WebStorm zeigt ihnen durch Unterringeln an, dass die Funktionen <code>sayHello</code> und <code>sayHelloOnEnter</code> nicht existieren. Wenn Sie jeweils  <code>greet.</code> davor einfügen, verschwinden die Fehlermarkierungen.)
Damit die Web-Anwendung wieder funktioniert, müssen Sie also in der Datei <code>main.js</code> vor jede Verwendung dieser beiden Funktionen die Zeichenfolge <code>greet.</code> einfügen.


Fügen Sie jetzt noch das Attribut <code>type="module"</code> in das Script-Element in der
Fügen Sie jetzt noch das Attribut <code>type="module"</code> in das Script-Element in der
Datei <code>src/index.html</code> ein '''und löschen Sie gegebenenfalls das Attribut <code>type="text/javascript"</code>'''.  
Datei <code>index.html</code> ein. Damit teilen Sie dem Browser mit, dass Sie die ECMAScript-6-Befehle <code>import</code> und <code>export</code> verwenden möchten.
Anderenfalls hätten sie Zugriff auf alle Objekte, Konstanten, Variablen und Funktionen einer geladenen Datei, so wie die bis ECMAScript 5 üblich war. Das heißt, das Laden einer Datei konnte zu massiven Problemen führen, wenn in zwei verschiedenen Dateien unterschiedliche Objekte etc. zufälligerweise gleich benannt wurden. Um diese Probleme zu umgehen gab es zahlreiche ziemlich aufwändige Hacks. Mit dem Modul-Konzept von ECMAScript gehören diese Probleme der Vergangenheit an.
<!--
'''Löschen Sie gegebenenfalls das Attribut <code>type="text/javascript"</code>'''.  
-->
Damit sollte Ihre Anwendung wieder funktionieren.
Damit sollte Ihre Anwendung wieder funktionieren.
Leider macht Ihnen WebStorm einen Strich durch die Rechnung. Aus „Sicherheitsgründen“
unterbindet WebStorm das dynamische Laden der Datei <code>greet.js</code> durch den Browser.
Sie werden durch ein Pop-up-Fenster aufgefordert, eine “authorization URL” zum Clipboard zu
kopieren. Wenn Sie das tun, ist es schon zu spät. Der Browser konnte die Datei nicht laden
und daher funktioniert die Web-App nicht (wenn Sie sie von WebStorm aus starten;
auf einem anderen Server würde sie schon funktionieren, wenn Sie alles richtig gemacht haben).


Gewöhnen Sie WebStorm dieses Verhalten ab:
Vergessen Sie nicht, Ihr aktuelles Projekt zu committen und auf dem Git-Server zu speichern, sobald alles funktioniert.


* <code>File</code>/<code>WebStorm</code> → <code>Settings</code>/<code>Preferences</code> → <code>Build, Execution, Deployment</code> → <code>Debugger</code>
===Google-Pagespeed-Test===
* Häkchen bei <code>Allow unsigned requests</code>
* <code>OK</code>


Jetzt sollte die Anwendung wieder laufen (auch wenn Sie sie von WebStorm aus starten).
Wenn man die Anwendung noch einmal mit Google testet, erhält man eine neue Empfehlung


Vergessen Sie nicht:
[https://developers.google.com/speed/pagespeed/insights/?url=https%3A%2F%2Fglossar.hs-augsburg.de%2Fbeispiel%2Ftutorium%2F2021%2Fhello_world%2FWK_HelloWorld04%2Fweb%2Findex2.html PageSpeed Insights (für Musterlösung index2.html)]:
* Klick auf das Commit-Icon.
<blockquote cite="https://developers.google.com/speed/pagespeed/insights/?url=https%3A%2F%2Fglossar.hs-augsburg.de%2Fbeispiel%2Ftutorium%2F2021%2Fhello_world%2FWK_HelloWorld04%2Fweb%2Findex2.html">
 
Halten Sie die Anfrageanzahl niedrig und die Übertragungsgröße gering 5 Anfragen • 5 KiB.
===webpack improved===
</blockquote>
 
Bislang wird im Produktivmodus nur die JavaScript-Datei komprimiert. Schöner wäre es, wenn auch die CSS-Datei und die HTML-Datei komprimiert werden würden.
Und wenn man die Google-Vorgaben vollständig umsetzen möchte, sollte außerdem der komprimierte Inhalt der Datei <code>head.css</code> direkt in die
Datei <code>index.html</code> eingefügt werden. All das kann man mit webpack erreichen.
 
Installieren Sie dazu folgende Node.js-Pakete:


<source lang="bash">
Das heißt, wir sollen die Dateien komprimieren: Überflüssige Kommentare, Leerzeichen und Zeilenumbrüche
# Die folgenden Pakete wurden schon installiert.  
löschen, lange Variablen-, Konstanten- und Funktionsnamen verkürzen etc. Außerdem sollten wir die beiden Dateien
# Das brauchen Sie nicht zu wiederholen.
<code>main.js</code> und <code>greet.js</code> wieder zu einer Datei zusammenfügen. Da dies gegen das Prinzip der Modularisierung spricht, wird dies nicht im Sourcecode, sondern automatisch mit Hilfe eines geeigneten Tools ([[webpack]], [[vite]], ...) realisiert (siehe [[HTML5-Tutorium: JavaScript: Hello World 05|Tutorium: Teil 5]]). Das heißt, der Sourcecode wird weiterhin modular aufgebaut. Anschließend wird der Sourcecode mit Hilfe eines [[Transpiler]s (Source-to-Source-Compiler) in eine kompakte Darstellung transformiert. Die Anzahl der Dateien, die vom Browser geladen werden müssen, wird drastisch reduziert, überflüssige Leerzeichen und Kommentare werden entfernt, Variablennamen werden durch kurze Namen ersetzt etc. Ein Transpiler kann auch die Programmiersprache ändern. Zum Beispiel kann ECMAScript 6 in ECMAScript 5 übersetzt werden. Oder man verwendet eine Sprache wie TypeScript, CoffeeScript etc. oder sogar Java und übersetzt diese in ECMAScript, damit der Browser den Code interpretieren kann.
# Es macht aber auch nichts, wenn Sie sie noch einmal installieren.
npm install --save-dev webpack webpack-cli
npm install --save-dev html-loader css-loader style-loader 
npm install --save-dev file-loader extract-loader 
npm install --save-dev suppress-chunks-webpack-plugin


# Die folgenden Pakete werden zusätzlich benötigt:
===Weitere Modularisierung===
npm install --save-dev cssnano postcss-loader
npm install --save-dev posthtml-loader posthtml-inline-assets
npm install --save-dev html-webpack-plugin terser-webpack-plugin
npm install --save-dev
</source>


Nun müssen Sie „nur noch“ den Inhalt der Datei <code>webpack.config.js</code> ersetzen.
Teilen Sie die Datei <code>greet.js</code> in zwei Dateien <code>greet.js</code> und <code>greet_on_enter.js</code> mit jeweils eines Funktion auf, um dem
(In dieser Datei steckt stundenlange Arbeit, da die webpack-Dokumentation häufig nicht so prickelnd ist
[[Programmierprinzipien#Single_Responsibility_Principle.5B9.5D|Single Responsibility Principle]] zu genügen.
und es Dutzende von Paketen gibt, die Ähnliches leisten, aber oft die ein oder andere Schwäche haben.
Ich behaupte übrigens nicht, dass dies die perfekte Konfigurationsdatei ist, um die gewünschten Optimierungen zu erzielen.
Aber sie funktioniert! Zumindest solange webpack5 nicht veröffentlicht wurde. )


<source lang="javascript">
Musterlösung:
const
[https://glossar.hs-augsburg.de/beispiel/tutorium/2021/hello_world/WK_HelloWorld04b/web/index.html <code>index.html</code>]
  path                      = require('path'),
([https://gitlab.multimedia.hs-augsburg.de/kowa/WK_HelloWorld04b.git Git-Repository v04b])
  TerserPlugin              = require('terser-webpack-plugin'),
  SuppressChunksPlugin      = require('suppress-chunks-webpack-plugin').default;


function absolutePath(l_path)
===Konfigurierbarkeit===
{ return path.resolve(__dirname, l_path); }


module.exports = function(p_env, p_argv)
Entfernen Sie programmspezifische Konstanten aus den JavaScript-Dateien und definieren Sie diese in einer Datei <code>config.js</code> oder (sehr viel besser) <code>config.json</code>, um dem Prinzip der [[Programmierprinzipien#Konfigurierbarkeit.2C_Customizability|Konfigurierbarkeit]] zu genügen.
{ let isProd = p_argv.mode === 'production';
  // isProd === true iff webpack is run in production mode


  return {
Musterlösung:
  // store the current mode ('production', 'development')
[https://glossar.hs-augsburg.de/beispiel/tutorium/2021/hello_world/WK_HelloWorld04c/web/index.html <code>index.html</code>]
  // in the webpack object so that webpack plugins can access it
([https://gitlab.multimedia.hs-augsburg.de/kowa/WK_HelloWorld04c.git Git-Repository v04c])
  mode:   p_argv.mode,


  // https://webpack.js.org/configuration/devtool/
==Fortsetzung des Tutoriums==
  // 'eval-source-map' than 'inline-source-map' would be better.
  // However, using this devtool forces a developer using WebStorm
  // to authorize an error url created by webpack within WebStorm
  // before it can be viewed within the browser.
  devtool: isProd ? false : 'inline-source-map',
 
  entry:
  { // use the following files to create
    // web/index.html, web/css/head.css, web/js/main.js
    html: absolutePath('src/index.html'),
    head: absolutePath('src/css/head.css'),
    main: absolutePath('src/js/main.js'),
  },


  output:
Wenn man die Vorschläge von Google beachtet,  
  { // placesJavaScript files in dictionary web/js
beschleunigt dies die Übertragung der Dateien vom Server zum Client, macht aber  den Code unlesbar.
    filename: 'js/[name].js',
Die Zusammenführung zweier Dateien widerspricht dem Prinzip der Modularisierung. Das Entfernen von Kommentaren,
    path: absolutePath('web'),
Leerzeichen und Leerzeilen ist aus Entwickler-Sicht vollkommen kontraproduktiv.
  },
 
  plugins:
  [ // delete html.js, head.js from output dictionary web/js
    // (For every member of the entry list, a JavaScript file
    // is created. index.html and head.css are extracted from
    // those files. So html.js and head.js are not needed
    // any longer.)
    new SuppressChunksPlugin(['html', 'head'])
  ],
 
  optimization:
  { minimizer:
    [ // minimizes ES6 JavaScript files
      // remove all JS comments but those that are marked to be preserved
      new TerserPlugin({ terserOptions: {output: {comments: /@preserve/}}})
    ]
  },
 
  module:
  { rules:
    [ { test:    /head\.css$/,
        use:    [ // store the extracted CSS content in a file
                  { loader: 'file-loader',
                    options: { name: '[name].[ext]',
                                outputPath: 'css/'
                              }
                  },
                  // extract the CSS content from the file loaded
                  'extract-loader',
                  // load the CSS file (using the optimizer defined below)
                  { loader: 'css-loader', options: { importLoaders: 1 } },
                  // optimize the CSS file
                  { loader: 'postcss-loader',
                    options: { sourceMap: 'inline',
                                plugins:
                                  () =>
                                  [ require('cssnano')
                                    ({ preset:
                                      [ 'default',
                                        { discardComments:
                                          // remove all CSS comments but those
                                          // that are marked tpo be preserved
                                          { remove: c => !c.match(/@preserve/) }
                                        }
                                        ],
                                    })
                                  ]
                              }
                  }
                ]
      },
 
      { test:    /main\.css$/,
        exclude: [/node_modules/, /web/],
        use:    [ // enable to import CSS files into a JavaScript file
                  'style-loader',
                  // load the CSS file (using the optimizer defined below)
                  { loader: 'css-loader', options: { importLoaders: 1 } },
                  // optimize the CSS file
                  { loader: 'postcss-loader',
                    options: { sourceMap: 'inline',
                                plugins:  [ require('cssnano') ]
                                          // all CSS comments are removed
                              }
                  }
                ]
      },
 
      { test:  /\.html$/,
        use:    [ // store the extracted HTML content in a file
                  { loader: 'file-loader',
                    options: { name: '[name].[ext]',
                              outputPath: '/'
                            }
                  },
                  // extract the HTML content from the file loaded
                  'extract-loader',
                  // load the HTML file; if on production mode minimize it
                  { loader: 'html-loader',
                    options: { minimize:                    isProd,
                              removeComments:              true,
                              ignoreCustomComments:        [/@preserve/],
                              //https://gist.github.com/stalniy/d55fc7990b4340c9d0ef55fd8838ed34
                              collapseWhitespace:          true,
                              conservativeCollapse:        false,
                              collapseInlineTagWhitespace: false,
                              removeAttributeQuotes:      false,
                              caseSensitive:              true,
                            }
                  },
                  { loader: 'posthtml-loader',
                    options:
                    { plugins:
                      [ require('posthtml-inline-assets')
                              ({ from: 'web/css', // 'web/web', 'web/abc' works as well ...
                                                  // this doesn't make any sense ...
                                  inline: { script: false }
                              })
                      ],
                    }
                  }
                ]
      },
    ]
  }
}};
</source>


Testen Sie den Übersetzungsmechanismus. Führen Sie <code>npm run dev</code> aus und sehen
Daher hat es sich eingebürgert, modularen Code mit Kommentaren und Leerzeichen zu schreiben. Dieser wird dann, bevor er an einen Browser übergeben wird automatisch zusammengefasst und komprimiert (minifiziert).  
Sie sich die erzeugten Dateien an (insbesondere den Head-Bereich der Datei <code>web/index.html</code>).
Führen Sie danach <code>npm run prod</code> aus und sehen Sie sich die Dateien noch einmal an.


Führen Sie die erzeugte Datei <code>web/index.html</code> jedes Mal auch im Browser aus und
Sie sollten nun [[HTML5-Tutorium: JavaScript: Hello World 05|Teil 5 des Tutoriums]] bearbeiten.
überprüfen Sie, ob sie noch funktioniert.
Dort erfüllen wir die Google-Vorgaben mittels <code>webpack</code>, einem mächtiges Werkzeug zur Umwandlung von Web-Dateien. Leider ist die Konfiguration dieses Werkzeugs sehr komplex.


Übrigens: webpack wurde so konfiguriert, dass HTML-, CSS- und JavaScript-Kommentare, in denen das Schlüsselwort <code>@preserve</code> enthalten ist,
auch in den komprimierten Code eingefügt werden. Das ist wichtig, wenn man Lizenzinformationen in diese Dateien einfügen möchte.
==Quellen==
==Quellen==
<references/>
<references/>
<ol>
<ol>
<li value="7"> {{Quelle|Kowarschick, W.: Multimedia-Programmierung}}</li>
<li value="1"> {{Quelle|Kowarschick, W.: Multimedia-Programmierung}}</li>
</ol>
</ol>
<noinclude>[[Kategorie: HTML5-Tutorium: JavaScript: Hello World]][[Kategorie: HTML5-Beispiel]][[Kategorie:Kapitel:Multimedia-Programmierung:Beispiele]]</noinclude>
<noinclude>[[Kategorie: HTML5-Tutorium: JavaScript: Hello World]][[Kategorie: HTML5-Beispiel]][[Kategorie:Kapitel:Multimedia-Programmierung:Beispiele]]</noinclude>

Version vom 12. April 2022, 14:26 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

Musterlösung: index0.html, index1.html, index2.html (Git-Repository v04a), index.html (Git-Repository v04b), index.html (Git-Repository v04c)

Anwendungsfälle (Use Cases)

Gegenüber dem dritten Teil des Tutoriums ändern sich die die Anwendungsfälle nicht. Die Anwendung leistet also genau dasselbe wie zuvor.

In diesem Teil des Tutoriums geht es darum, die Anwendung besser zu strukturieren, d. h. zu modularisieren (vgl. Programmierprinzipien).

Modularisierung

Eine Anwendung, wie z. B. ein HTML5-Spiel, besteht aus diversen unterschiedlichen Komponenten mit ganz unterschiedlichen Aufgaben. Bei einem Web-Frontend müssen beispielsweise verschieden Komponenten (Buttons, Scrolbars, Textfelder, Datumsfelder etc.) erstellt werden. Bei einem HTML5-Spiel müssen die Spielfiguren, die Spielszenen, die Spiellogik, die Benutzereingaben, die Darstellung das Spiels im Browser etc. erstellt und verwaltet werden, wobei das Erstellen und Verwalten teilweise während der Spielentwicklung und teilweise während der Spielausführung (Runtime) erfolgt.

An der Entwicklung eines Web-Systems oder eines Spiels sind üblicherweise mehrere oder gar viele Entwickler gleichzeitig beteiligt. Oft müssen einzelnen Komponenten an neue Endgeräte oder Betriebssysteme angepasst werden. Beispielsweise muss eine Tastatursteuerung durch eine Touch- oder Gestensteuerung ersetzt werden, wenn eine Anwendung so erweitert werden soll, dass es auch auf einem Smartphone oder Tablet läuft.

Das alles ist nur machbar, wenn man die Anwendung modularisiert, d. h. in kleine, möglichst unabhängige Komponenten unterteilt. Ein großer Feind der Modularisierung sind globale Variablen.

Je mehr Dateien von unterschiedlichen Autoren erstellt werden, desto größer ist die Gefahr, dass es beim Zugriff auf globale Variablen zu Namenskollisionen kommt. Außerdem weiß niemand, welche globale Größe noch benötigt wird und welche nicht. Sicherheitshalber löscht man daher keine dieser Variablen, auch wenn sie vollkommen veraltet sind.

Daher gilt der Grundsatz: Verwende so wenig globale Größen wie möglich.

Erstellen eines neuen Projektes

Erstellen Sie ein neues Projekt HelloWorld04 als Fork von HelloWorld03 (siehe Hello-World-02-Tutorium):

git clone https://gitlab.multimedia.hs-augsburg.de/ACCOUNT/HelloWorld03 HelloWorld04
//git clone https://gitlab.multimedia.hs-augsburg.de/kowa/WK_HelloWorld03 HelloWorld04

## Änderungen zur Projektidentifikation vornehmen (z.B. Titel anpassen)
git commit -m "HelloWorld03 fork created"

# Neues Repository in Gitlab anlegen:
git remote -v
git remote remove origin
git remote add origin https://gitlab.multimedia.hs-augsburg.de/ACCOUNT/HelloWorld04
git remote -v
git push --set-upstream origin master

Erstellen einer Ordnerstruktur

Grundsätzlich gilt auch bei der Programmierung: Ordnung ist das halbe Leben.

Web-Anwendungen werden sehr schnell sehr groß. Also sollte man sich eine geeignete Ordnerstruktur überlegen. Üblicherweise legt man CSS-Dateien in einen Ordner namens „css“ und JavaScript-Dateien in einen Ordner namens „js“ oder „lib“. Sollten es viele CSS- und/oder JavaScript-Dateien werden, legt man im entsprechenden Ordner geeignete Unterordner an.

Legen Sie im web-Verzeichnis Ihres Projektes folgende Ordner an:

  • web/css: Hier werden die eigentlichen CSS-Dateien der Web-Anwendung gespeichert, die von der App dynamisch nachgeladen werden sollen. In diesem Ordner könnten zur weiteren Strukturierung Unterordner angelegt werden. Allerdings ist das hier nicht notwendig. Es wird nur die Dateien „head.css“ und „body.css“ geben. Verschieben Sie die Datei web/main.css in diesen Ordner.
  • web/js: Hierher kommen die JavaScript-Dateien, die von der Anwendung benötigt werden. Verschieben Sie die Datei web/main.js in diesen Ordner.

Nehmen Sie in der Datei web/index.html folgende beiden Ersetzungen vor:

  • main.csscss/main.css
  • main.jsjs/main.js

Wenn Sie jetzt die Datei index.html mittels run ausführen, sollte Ihre Anwendung wieder funktionieren.

Sollte dies der Fall sein, so sollten Sie die Änderungen committen und evtl. auch gleich per git push auf den Git-Server hochladen. Es schadet übrigens nie, vor einen Commit oder gar einen Push noch einmal zu überprüfen, ob die Anwendung noch läuft.

Die App

Laden der CSS- und der JavaScript-Dateien

Eine Web-Anwendung funktioniert nur – wie Sie bereits erfahren haben –, wenn nicht nur die HTML-Datei, sondern auch die zugehörigen CSS- und JavaScript-Dateien geladen werden. Allerdings gibt es dabei ein Problem: Die CSS- und JavaScript-Dateien werden im Laufe der Zeit immer zahlreicher und/oder größer. Diese Dateien zu laden, dauert seine Zeit.

In der dritten Version der Web-Anwendung stehen die Verweise auf diese Dateien im head-Bereich des Dokuments. Dieser wird vollständig geladen, bevor der body-Bereich eingelesen wird. (Genauer gesagt, nur die CSS-Datei wird vollständig geladen. Durch die Angabe von async="async" im Script-Tag wird erreicht, dass die JavaScript-Datei main.js geladen wird, während der Body-Bereich gerendert wird.)

Das heißt aber, dass der Browser keine Inhalte des HTML-Dokuments darstellen kann, solange er CSS-Datei (und evtl. weitere Dateien) lädt. Wenn dies zu lange dauert, verliert der Besucher die Geduld und verlässt die Seite vorzeitig.

Besser wäre es daher andersherum vorzugehen: Es wird zuerst der Body-Bereich geladen und dann die JavaScript- und die CSS-Dateien. Im Falle von JavaScript ist das durchaus sinnvoll (und durch die Angabe von async="async" auch machbar), aber im Falle von CSS hat das den Effekt, dass der Browser noch keine Layout-Vorgaben erhalten hat, wenn er mit dem Rendern der Seite beginnt. Also verwendet er die browserspezifischen Defaultwerte. Das heißt, die Seite sieht zunächst ganz anders aus, als vom Designer geplant. Wenn dann die CSS-Dateien geladen wurden, wird die Seite erneut gerendert und verändert ihr Aussehen. Auch das ist verwirrend und wirkt unprofessionell.

Was also machen?

CSS-Dateien

Damit das Problem mit dem falschen Layout nicht auftritt, empfiehlt Google (PageSpeed Insights), ganz wenige, wichtige CSS-Befehle direkt – d. h. als CSS-Befehle innerhalb eines style-Elements – in dem HTML-Head-Bereich einzufügen. Alle anderen CSS-Anweisungen sollen erst am Ende der HTML-Datei dynamisch mittels JavaScript (oder mittels des PageSpeed-Optimization-Modul von Google) gelesen werden.

PageSpeed Insights (für Musterlösung index0.html):

Empfehlung

Ressourcen blockieren den First Paint Ihrer Seite. Versuchen Sie, wichtiges JS und wichtige CSS inline anzugeben und alle nicht kritischen JS und Stile aufzuschieben. Weitere Informationen.

Gehen Sie so vor, wie es Google empfiehlt. Genauer gesagt, gehen Sie so vor, wie es unter Multimedia-Programmierung: Best Practices beschrieben ist. In diesem und den folgenden Tutorien werden die dort beschriebenen Prinzipien berücksichtigt.

Legen Sie die Datei src/css/head.css an und fügen Sie folgenden (sehr kurzen!) Code ein:

body
{ background-color: #C5EFFC; }

.hidden
{ display: none; }

In der Datei src/index.html müssen Sie das Link-Element in Header folgendermaßen

<link rel="stylesheet" href="css/head.css"/>

anpassen. Das heißt, Sie laden im HTML-Head-Bereich nur noch die kleine CSS-Datei. Damit funktioniert Ihre Web-Anwendung derzeit nur noch rudimentär: Es gibt kaum noch CSS-Anweisungen, die berücksichtigt werden. Noch besser ist es gemäß den Google-Empfehlungen, wenn man die CSS-Regelsätze direkt in die HTML-Datei einfügt. Ersetzen Sie also das Link-Element durch folgendes Style-Element:

<style>
body
{ background-color: #C5EFFC; }

.hidden
{ display: none; }
</style>

In der Datei src/css/main.css können Sie nun das .hidden-Element und das Farbattribut background-color: #C5EFFC; entfernen.

Es ist sinnvoll, die Datei main.css in body.css umzubenennen: head.css wird im HTML-Head-Bereich vor dem Body-Bereich geladen, body.css soll parallel zum Body-Bereich geladen werden und dann zur Verfügung stehen, wenn der Body vollständig geladen wurde.

Leider ist CSS-Link-Element nicht möglich, async="async" anzugeben, wie wir dies beim Skript-Element gemacht haben, um JavaScript-Code parallel zum Body-Element zu laden. Hier hilft derzeit nur ein kleiner Hack

<link rel="stylesheet" href="css/body.css"
      media="none" onload="this.media='all'" 
/>

Die Datei body.css wird für den Medientyp none geladen. Diesen Medientyp gibt es aber gar nicht. Es gibt die Typen screen, print, speech, all sowie ein paar weitere veraltete Typen. Da der Medientyp none vorerst nicht gebraucht wird, laden ihn moderne Browser asynchron. Sobald das HTML-Dokument vollständig geladen wurde, feuert das Ereignis onload. Sobald dies passiert, wird der Medientyp none durch all ersetzt. Daran erkennt der Browser, dass dies eine CSS-Datei ist, die durchaus zum Layouten der aktuellen Seite verwendet werden soll. Und das macht der Browser dann auch.

Zusammenfassung:

  • In head.css stehen möglichst wenig CSS-Anweisungen. Sie sollten ausreichen, um die aktuelle Seite “above the fold” („über dem Zeitungsknick“) fehlerfrei zu rendern. Das heißt, es sollte alles korrekt gerendert werden, was nach dem Laden im größtmöglichen Browserfenster sichtbar ist.
  • Der Inhalt der head.css sollte direkt in den Head-Bereich der HTML-Seite eingefügt werden.
  • Die restlichen CSS-Anweisungen werden in die Datei body.css eingetragen.
  • Dies Datei wird mittels eines Hacks asynchron, d. h. parallel zum Body-Element geladen.

index.html

Da die Datei head.css nur die Hintergrundfarbe sowie die .hidden-Regel enthält, sollten alle HTML-Elemente, die gerendert werden sollen, zunächst unsichtbar gemacht werden. Bei einer Web-Anwendung, wie einem Spiel, die zum Laden länger braucht, könnte man eine Sanduhr oder einen Ladebalken einfügen, um anzuzeigen, dass man noch etwas warten muss. Üblicherweise, sollte man dafür sorgen, dass das Laden einer Seite so schnell geht, dass der Benutzer gar nicht erst ungeduldig wird.

Fügen Sie die CSS-Klasse hidden in beide HTML-Sektions ein:

<section id="section_form" class="hidden">
...
</section>
<section id="section_hello" class="hidden">
...
</section>

Wenn man jetzt das Dokument im Browser öffnet, sieht man nichts, außer einem hellblauen Hintergrund.

Abhilfe schafft hier ein kleiner JavaScript-Befehl, der hidden von der Eingabe-Section hidden-form entfernt, sobald das Dokument vollständig geladen wurde (samt allen CSS- und JavaScript-Dateien):

document.getElementById('section_form').classList.remove('hidden');

An welche Stelle dieser Befehl eingefügt wird, erfahren Sie im nachfolgenden Abschnitt.

JavaScript-Dateien

Für JavaScript-Dateien empfiehlt Google ebenfalls, diese nicht schon zu Beginn zu laden, sondern das Laden „aufzuschieben“. Das machen wir bereits mit Hilfe des Async-Attributs.

Fügen Sie nun den zuvor erwähnten Befehl

document.getElementById('section_form').classList.remove('hidden');

als letzen Befehl in den Rumpf der Funktion init in der Datei web/js/main.js ein.

Wenn Sie jetzt die Web-App starten, sollte sie wieder funktionieren .

Was haben Sie erreicht? Solange die Anwendung geladen wird, sehen Sie nur einen blauen Hintergrund ohne Inhalt, bzw. i. Allg. einen so genannten Splash Screen. Wenn dann alles, d. h. alle CSS- und JavaScript-Dateien geladen wurden, wird die eigentliche Anwendung sichtbar.

Vergessen Sie nicht: Commit!

Modularisierung der JavaScript-Dateien

Die JavaScript-Datei main.js aus dem dritten Teil des Tutoriums wird in zwei Teile aufgespalten: greet.js und main.js. Die erste Datei enthält die eigentliche Begrüßungsfunktionalität, die zweite initialisiert die App, d. h. aktiviert die Begrüßungsfunktion.

Erstellen Sie die JavaScript-Datei web/js/greet.js und verschieben Sie die beiden Funktionen sayHello und sayHelloOnEnter (jeweils samt Funktionsrumpf :-) ) in diese Datei.

Schreiben Sie vor jede der beiden Funktionsdefinitionen das Schlüsselwort export. Mit dieser ES-6-Anweisung legen Sie fest, dass die beiden Funktionen von anderen ES-6-Dateien mittels eines Import-Befehls importiert werden können. (Funktionen ohne Export-Anweisung könnten von anderen ES-6-Dateien nicht importiert werden.)

In der Datei web/js/main.js werden jetzt diese beiden Funktionen importiert. Fügen Sie ganz am Anfang der Datei (d. h. vor der verbliebenen Init-Funktion) die folgende Import-Anweisung ein:

import * as greet from './greet.js';

Damit wird erreicht, dass alle (*) Elemente, die von der Datei greet.js exportiert werden, unter ihrem jeweiligen Namen im Objekt greet gespeichert werden. (Man könnte auch nur bestimmte Elemente von greet.js importieren, sofern man nicht alle benötigt.) Das heißt, innerhalb der Datei main.js gibt es nun die beiden Funktionen greet.sayHello und greet.sayHelloOnEnter.

Damit die Web-Anwendung wieder funktioniert, müssen Sie also in der Datei main.js vor jede Verwendung dieser beiden Funktionen die Zeichenfolge greet. einfügen.

Fügen Sie jetzt noch das Attribut type="module" in das Script-Element in der Datei index.html ein. Damit teilen Sie dem Browser mit, dass Sie die ECMAScript-6-Befehle import und export verwenden möchten. Anderenfalls hätten sie Zugriff auf alle Objekte, Konstanten, Variablen und Funktionen einer geladenen Datei, so wie die bis ECMAScript 5 üblich war. Das heißt, das Laden einer Datei konnte zu massiven Problemen führen, wenn in zwei verschiedenen Dateien unterschiedliche Objekte etc. zufälligerweise gleich benannt wurden. Um diese Probleme zu umgehen gab es zahlreiche ziemlich aufwändige Hacks. Mit dem Modul-Konzept von ECMAScript gehören diese Probleme der Vergangenheit an. Damit sollte Ihre Anwendung wieder funktionieren.

Vergessen Sie nicht, Ihr aktuelles Projekt zu committen und auf dem Git-Server zu speichern, sobald alles funktioniert.

Google-Pagespeed-Test

Wenn man die Anwendung noch einmal mit Google testet, erhält man eine neue Empfehlung

PageSpeed Insights (für Musterlösung index2.html):

Halten Sie die Anfrageanzahl niedrig und die Übertragungsgröße gering 5 Anfragen • 5 KiB.

Das heißt, wir sollen die Dateien komprimieren: Überflüssige Kommentare, Leerzeichen und Zeilenumbrüche löschen, lange Variablen-, Konstanten- und Funktionsnamen verkürzen etc. Außerdem sollten wir die beiden Dateien main.js und greet.js wieder zu einer Datei zusammenfügen. Da dies gegen das Prinzip der Modularisierung spricht, wird dies nicht im Sourcecode, sondern automatisch mit Hilfe eines geeigneten Tools (webpack, vite, ...) realisiert (siehe Tutorium: Teil 5). Das heißt, der Sourcecode wird weiterhin modular aufgebaut. Anschließend wird der Sourcecode mit Hilfe eines [[Transpiler]s (Source-to-Source-Compiler) in eine kompakte Darstellung transformiert. Die Anzahl der Dateien, die vom Browser geladen werden müssen, wird drastisch reduziert, überflüssige Leerzeichen und Kommentare werden entfernt, Variablennamen werden durch kurze Namen ersetzt etc. Ein Transpiler kann auch die Programmiersprache ändern. Zum Beispiel kann ECMAScript 6 in ECMAScript 5 übersetzt werden. Oder man verwendet eine Sprache wie TypeScript, CoffeeScript etc. oder sogar Java und übersetzt diese in ECMAScript, damit der Browser den Code interpretieren kann.

Weitere Modularisierung

Teilen Sie die Datei greet.js in zwei Dateien greet.js und greet_on_enter.js mit jeweils eines Funktion auf, um dem Single Responsibility Principle zu genügen.

Musterlösung: index.html (Git-Repository v04b)

Konfigurierbarkeit

Entfernen Sie programmspezifische Konstanten aus den JavaScript-Dateien und definieren Sie diese in einer Datei config.js oder (sehr viel besser) config.json, um dem Prinzip der Konfigurierbarkeit zu genügen.

Musterlösung: index.html (Git-Repository v04c)

Fortsetzung des Tutoriums

Wenn man die Vorschläge von Google beachtet, beschleunigt dies die Übertragung der Dateien vom Server zum Client, macht aber den Code unlesbar. Die Zusammenführung zweier Dateien widerspricht dem Prinzip der Modularisierung. Das Entfernen von Kommentaren, Leerzeichen und Leerzeilen ist aus Entwickler-Sicht vollkommen kontraproduktiv.

Daher hat es sich eingebürgert, modularen Code mit Kommentaren und Leerzeichen zu schreiben. Dieser wird dann, bevor er an einen Browser übergeben wird automatisch zusammengefasst und komprimiert (minifiziert).

Sie sollten nun Teil 5 des Tutoriums bearbeiten. Dort erfüllen wir die Google-Vorgaben mittels webpack, einem mächtiges Werkzeug zur Umwandlung von Web-Dateien. Leider ist die Konfiguration dieses Werkzeugs sehr komplex.

Quellen

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