Multimediale Inhalte im Browser: Audio-, Video- und Canvas-Elemente

Multimediale Inhalte wie Audio- und Videodateien können wichtige Bestandteile interaktiver Webanwendungen sein. Der Einsatz der entsprechenden HTML-Elemente erlaubt die einfache und native Integration solcher Inhalte. Für die Realisierung einfacher 2D/3D-Inhalte steht im Browser mit der Canvas-API eine rudimentäre Komponente zur Verfügung, die die Realisierung von graphischen Inhalten auch außerhalb der festen DOM-Strukturen erlaubt.

Einleitung

Während die fünfte Fassung des HTML-Standards1 grundsätzlich nur Veränderungen und Neuerung im Sprachstandard umfasst, werden unter dem Begriff HTML5 in der Regel auch die in diesem Kontext eingeführten neuen Web-APIs sowie Aktualisierungen des CSS-Standards (Vgl. CSS3) zusammengefasst. Eine der zentralen Erweiterungen der Browser-Fähigkeiten im Kontext von HTML5 ist die erweiterte Unterstützung von Medieninhalten (Audio und Video) sowie die Einführung rudimentärer 2D/3D-Operationen im Browser. Durch die Einführung der <audio> und <video>-Elemente können entsprechende Inhalte direkt und ohne zusätzliche Plugins oder Erweiterungen im Browser abgespielt werden. Video- und Audiodateien werden gleichberechtigte "first-class citizens" (Mozilla Developer Network, HTML5) des Browsers, deren Wiedergabe durch die entsprechenden APIs programmatisch aus dem Javascript-Code heraus gesteuert werden kann. Zusätzlich erhalten Entwickler und Entwicklerinnen mit dem Canvas die Möglichkeit, Teilbereiche des User Interfaces auch außerhalb der DOM-Struktur zu gestalten. Das <canvas>-Element stellt dazu einen Knoten im DOM-Baum dar, dessen Inhalt nicht durch Text oder weitere Kindelemente definiert wird, sondern durch eine rudimentäre Grafikschnittstelle programmatisch erzeugt und aktualisiert werden kann. In dieser Lektion lernen Sie die grundsätzliche Verwendung von Audio-, Video- und Canvas-Elemente kennen und erfahren, wie die Kompatibilität zwischen Canvas- und Video-Inhalten sinnvoll für die Implementierung interessanter Anwendungskonzepte verwendet werden kann.

Audio & Video

Grundlage für die Audio- und Video-Wiedergabe sind die entsprechenden DOM-Elemente. Deren Aufbau und Verwendung unterscheidet sich dabei kaum. Das Erscheinungsbild und das Verhalten wird durch eine Reihe von Element-Attributen bestimmt. Der abzuspielende Inhalt wird durch Referenzen auf die entsprechenden Mediendateien definiert. Dabei kann entweder das src-Attribut verwendet werden oder, in Form einer Liste von Kindelementen, mit einem oder mehreren <source>-Elementen gearbeitet werden. Durch den Einsatz von mehreren <source>-Elementen ist es möglich, fallback-Lösungen zu definieren. Der Browser gibt dabei die erste interpretierbare Datei der Liste wieder - ein Verhalten, das vor allem dann wichtig ist, wenn im Browser Decoder für bestimmte Formate fehlen. Bis auf wenige Ausnahmen unterscheiden sich die Audio- und Video-Elemente hinsichtlich ihrer Verwendung nicht. Beispiele werden hier daher nur für das Video-Element aufgeführt. Etwaige Unterschiede zum Audio-Element sowie Besonderheiten können Sie hier nachlesen.

Ein einfaches Beispiel für die Integration eines Videos in ein HTML-Dokument kann so aussehen:

<video controls>
  <source src="video.mp4" type="video/mp4">
  <source src="video.webm" type="video/webm">
  <p>Ihr Browser unterstützt die HTML5-Video-Funktion leider nicht. Sie können sich das Video <a href="video.mp4">hier</a>herunterladen.</p>
</video>

Über das Attribut controls wird dem Browser mitgeteilt, dass die nativen Steuerelement des Browser eingeblendet werden sollen. Diese werden vom jeweiligen Browser-Hersteller implementiert, sollten aber immer Schaltflächen zum Starten und Pausieren, eine Zeitleiste, Lautstärke- und Vollbildregler sowie ggf. Auswahlmöglichkeiten für Untertitel und Tracks beinhalten. Eine Übersicht über die weiteren Attribute des Elements finden Sie hier. Im Beispiel wird die abzuspielende Datei durch Kindelemente vom Typ <source> definiert. Der Browser prüft, beginnend beim ersten <source>-Element, ob die verlinkte Datei abspielbar ist, d.h. de-codiert und wiedergegeben werden kann. Ist dies nicht der Fall, werden die angegebenen Alternativen geprüft. Kann keine der aufgeführten Dateien verwendet werden, wird der übrige Inhalt des Elements (hier der Inhalt des <p>-Abschnitts) angezeigt.

Korrespondierend zu den <video> und <audio>-Elementen existieren Javascript-Objekte, die die programmatische Steuerung der Elemente erlauben. Deren Verwendung folgt dabei dem bekannten Prinzip, das bereits für andere HTML-Elemente verwendet wurde: Innerhalb Ihrer Javascript-Anwendung referenzieren Sie die entsprechenden DOM-Elemente über die Selektor-Funktionen und arbeiten anschließend mit den zurückgegebenen Objekten. Grundlage für beide Objekte ist das HTMLMediaElement. Spezifische Methoden und Eigenschafte werden im HTMLVideoElement bzw. HTMLAudioElement definiert. Die programmatische Steuerung der selektierten Elemente erfolgt durch die Verwendung der Eigenschaften, Methoden und Events der entsprechenden Objekte.

Die nun folgenden Beispiele beziehen sich auf ein HTML-Dokument mit diesem (Teil-)Inhalt:

 <div id="player">
        <video>
            <source src="example.mp4" type="video/mp4">
        </video>
        <ul id="controls">
            <li class="play"><i class="fas fa-play"></i></li>
            <li class="pause"><i class="fas fa-pause"></i></li>
            <li class="stop"><i class="fas fa-stop"></i></li>
        </ul>
        <input type="range" id="seekbar" value="0">
</div> 

Die Schaltflächen werden durch Icons aus dem Font Awesome-Projekt realisiert. Mit entsprechenden CSS-Regeln verknüpft wird das Element mit der ID player im Browser dann in dieser Form dargestellt:

Screenshot des Video-Player

Die Wiedergabe und das Pausieren des aktuellen Videos erfolgt durch den Aufruf der entsprechenden Methoden:

let videoPlayer = document.querySelector("video");
videoPlayer.start(); // starts playback
videoPlayer.pause(); // pauses playback

Die aktuelle Abspielposition kann mit Hilfe der Eigenschaft currentTime ausgelesen und gesetzt werden. Die Angabe erfolgt dabei immer in Sekunden und einer Browser-abhängigen Präzision (z.B. 2ms im Firefox ab Version 60). Wenn Sie eigene Kontrollflächen, wie z.B. eine Zeitleiste für die Auswahl einer bestimmten Abspielposition erstellen wollen, müssen Sie aus der relativen Position des verwendeten Eingabeinstruments (z.B. einem range-Element) eine absolute (zeitliche) Position innerhalb der Mediendatei berechnen. Dazu können Sie über die Eigenschaft duration des Video-Elements auf die (zeitliche) Länge des aktuell geladenen Videos zurückgreifen:

let seekbar = document.querySelector("#seekbar");

seekbar.addEventListener("change", function () {
    let selectedPosition =  videoPlayer.duration * (seekbar.value / 100));
    videoPlayer.currentTime = selectedPosition; 
});

In der Regel dient das Zeitleisten-Element des User Interface eines Media Players nicht nur der Auswahl eines bestimmten Abschnitts der wiederzugebenden Datei, sondern zeigt während der Wiedergabe auch deren aktuelle Abspielposition an. Um diese Funktion zu realisieren, müssen Sie das Event timeupdate abfangen (Vgl. Media Events, das während der Wiedergabe regelmäßig vom Video- bzw. Audio-Element ausgelöst wird. Das Intervall, in dem dieses Event ausgelöst wird, nimmt - bei korrekter Implementierung durch den Browser-Hersteller - Rücksicht auf die allgemeine Systemlast und sollte sich zwischen 4Hz und 66Hz bewegen. Auf Basis der aktuellen Abspielpostion kann dann der korrespondierende Wert für die Suchleiste berechnet werden. In diesem und dem vorherigen Beispiel wird davon ausgegangen, dass die Standardwerte (siehe hier und hier) des range inputs nicht verändert wurden:

video.addEventListener("timeupdate", function() {
  var value = (100 / videoPlayer.duration) * videoPlayer.currentTime;
  seekbar.value = value;
});

Die Canvas-API

Die Canvas-API erlaubt das programmatische Erzeugen von 2D- bzw. 3D-Grafiken und Animationen im Browser. Die Inhalte des <canvas>-Elements werden, anders als bei klassischen DOM-Elementen, nicht durch Textinhalte oder Kindelemente definiert, sondern durch eine manipulierbare Grafikoberfläche erzeugt. Zentrale Akteure sind dabei das eigentliche canvas-Element, das im HTML-Dokument definiert und so in den DOM-Baum eingebunden wird sowie die Javascript-API, die eine Manipulation der vom Canvas bereitgestellten Zeichenfläche erlaubt. Die Canvas-API erlaubt die Erzeugung und Darstellung statischer Grafiken (z.B. Diagrammen) oder dynamischen Animationen und Inhalten (z.B. Videospielen) und ist Grundlage für zahlreiche Frameworks und Libraries (z.B. Phaser, Fabric.js oder Chart.js).

Grundlagen

Um die Möglichkeiten der Canvas-API zu verwenden, muss zuerst ein entsprechendes Element erzeugt werden. Im folgenden Beispiel wird dabei von einem HTML-Dokument ausgegangen, das unter anderem das folgende Element beinhaltet:

<canvas id="canvas" width="500" height="500">
</canvas>

Das canvas-Element verfügt über width- und height-Attribute, mit denen die Dimensionen des DOM-Elements definiert werden. Zusätzlich werden diese Angaben verwendet, um die Auflösung der internen Zeichenfläche anzugeben, auf der die Canvas-Inhalte dargestellt werden. Das DOM-Element, nicht die Zeichenfläche, kann auch durch CSS-Eigenschaften angepasst werden. Sollte über diesen Weg eine andere als die ursprünglich definiert Elementgröße ausgewählt werden, kann es zu Darstellungsfehlern kommen, wenn Zeichenfläche und Element über unterschiedliche Seitenverhältnisse verfügen. Ähnlich wie bei den media elements, kann auch beim Canvas-Element fallback content angegeben werden, der angezeigt wird, wenn die eigentliche Canvas-Funktionalität im Browser nicht zur Verfügung steht. Neben textuellen Inhalten bieten sich hier auch statischer Bilder (<img>) an.

Die Zeichenfläche

Die eigentlichen Zeichenoperationen werden nicht direkt auf dem Canvas-Element ausgeführt. Um Inhalte auf dem Element darzustellen, muss zuerst eine Zeichenfläche, der sogenannte rendering context, aus dem Canvas ausgewählt werden:

let canvasEl = document.querySelector("#canvas"),
context = canvasEl.getContext("2d");

Über den Parameter der getContext-Methode bestimmen Sie die Art der erzeugten Zeichenfläche. Im Beispiel wird ein CanvasRenderingContext2D-Objekt zurückgegeben, das für die Darstellung zwei-dimensionaler Inhalte verwendet werden kann. Zusätzlich stehen im Rahmen der WebGL-API auch entsprechende Schnittstellen für die Darstellung drei-dimensionaler Inhalte zur Verfügung. In dieser Lektion wird nur auf die Möglichkeiten des 2D-Kontexts eingegangen.

Die Zeichenoperationen

Mit Hilfe des ausgewählten Contexts können nun die eigentlichen Zeichenoperationen ausgelöst werden. Die API erlaubt dabei das Zeichnen von einfachen Formen und Pfaden, Bildern sowie Texten. Grundlage für die Operationen ist ein Koordinatensystem, dessen Ursprung (0,0) in der oberen, linken Ecke liegt. Die Dimensionen des Gitternetzes ergeben sich aus den gewählten Werten der width- bzw. height-Attribute. Die Auflösung entspricht in der Regel einem Pixel. Für komplexere Operationen besteht die Möglichkeit, den Ursprung des Koordinatensystem zu versetzen bzw. das Gitternetz zu skalieren und zu rotieren. Im folgenden Beispiel wird nur eine Auswahl der vorhandenen Methoden vorgestellt. Machen Sie sich ggf. mit der vollständigen API vertraut.

Rechtecke stellen die einfachste Variante der Formen dar, die auf dem Canvas gezeichnet werden können. Mit Hilfe der Methode fillRect wird ein einfarbiges Rechteck erstellt. Übergeben werden Parameter, die den Ursprung sowie die Länge bzw. Höhe der zu zeichnenden Form definieren.

context.fillRect(50,50,100,100);

Das Resultat sieht im Browser so aus:

Screenshot des Canvas

Neben Rechtecken stellen Pfade die zweite primitive Form dar, die mittels der Canvas-API gezeichnet werden können. Pfade bestehen aus Linien und Bögen. Im oben gezeigten Screenshot wurden Linien verwendet, um das Koordinatensystem des Canvas zu visualisieren.

Beim Zeichnen von Pfaden sind - im Gegensatz zu den Rechtecken - weitere Teilschritte notwendig:

  1. Ein einzelner Pfad wird durch den Aufruf der beginPath-Methode geöffnet und durch die Methode closePath geschlossen.

  2. Die Definition des Pfades erfolgt durch Zeichenoperationen, die zwischen den beiden Methodenaufrufen ausgelöst werden.

  3. Der definierte Pfad wird anschließend durch Nachzeichnen (stroke) oder Füllen (fill) dargestellt.

Um das im Screenshot dargestellte Koordinatensystem zu zeichnen, wird der folgende Code verwendet:

const WIDTH = 500,
  HEIGHT = 500,
  GRID_SIZE = 10,
  GRID_COLOR = "rgb(220,220,220)";

  var canvasEl, context;

function initCanvas() {
  canvasEl = document.querySelector("#canvas");
  context = canvasEl.getContext("2d");
}

function drawGrid() {
  for (let i = GRID_SIZE; i <= WIDTH - GRID_SIZE; i += GRID_SIZE) {
    drawLine({x: i, y: 0},{x: i, y: HEIGHT},GRID_COLOR);
    drawLine({x: 0, y: i},{x: WIDTH, y: i},GRID_COLOR);
  }
}

function drawLine(from, to, color) {
  context.strokeStyle = color;
  context.beginPath();
  context.moveTo(from.x, from.y);
  context.lineTo(to.x, to.y);
  context.stroke();
  context.closePath();
}

initCanvas();
drawGrid();

Farbe und Stärke der Pfade und Formen können durch das Manipulieren der entsprechenden Eigenschaften des Context-Objekts verändert werden. Diese Manipulation muss dabei vor dem Nachzeichnen bzw. Füllen des Pfades erfolgen. Im Beispiel wird die strokeStyle-Eigenschaft geändert, um die Linienfarbe anzupassen. Die Canvas-API ist dabei kompatibel zu den bekannten CSS-Farbwerten (RGB-, HEX- oder Text-Darstellung). Neben der Farbe können auch noch weitere Eigenschaften beeinflusst werden.

Importieren von Bildern

Mittels der drawImage-Methode der Canvas-API können Bilder auf den ausgewählten Kontext gezeichnet werden. Neben der Position und den (optionalen) Dimensionen wird der Methode dazu die Bild-Quelle übergeben. Diese kann aus einem HTMLImageElement (<img>), einem HTMLVideoElement (<video>) oder einem anderen Canvas (<canvas>) bestehen2. Die jeweilige Quelle kann dabei entweder bereits im DOM vorhanden sein, oder für den Zeichenprozess neu erstellt werden. Vor allem im zweiten Fall muss berücksichtigt werden, dass das Zeichnen des Bildes erst dann möglich ist, wenn dieses vollständig aus den entsprechenden Quellen (src-Attribut) geladen ist:

// Erstellen eines leeren img-Containers mit jeweils 100px Breite und Höhe
let image = new Image(100, 100);

// Registrieren des Listeners für die Benachrichtigung über das abgeschlossene Laden des Bildes
image.addEventListener("load", function() {
  // Zeichnen des vollständig geladenen Bildes an die Postion x = 10, y = 10
  context.drawImage(image, 10,10);
});

// Setzen der Bildquelle  über einen relativen Pfad
image.src = "image.png"

Pixel-basierte Manipulation der Canvas-Inhalte

Der aktuelle Inhalt des Canvas kann als Pixel-Matrix exportiert, manipuliert und re-importiert werden. Bindeglied zwischen diesen Operationen ist der ImageData-Prototyp, der die einzelnen Pixel (bzw. deren Farbwerte) eines Canvas repräsentiert. Über die Methode getImageData wird dazu der aktuelle Inhalt des gesamten Canvas oder eines über die Parameter definierten Teilbereichs ausgelesen:

let imageData = context.getImageData();

Das zurückgegebene ImageData-Objekt enthält Informationen zu der Breite (width) und Höhe (height) des exportierten Ausschnitts. Über die Eigenschaft data, einem ein-dimensionalen-Array, werden die einzelnen Pixel zugänglich gemacht. Deren RGBA-Werte werden durch jeweils vier aufeinanderfolgende Elemente des Arrays repräsentiert. Die Pixel sind dabei zeilenweise, beginnend von der oberen, linken Ecke des Bildes sortiert:

// Auslesen aller Pixel des exportierten Ausschnitts
for(let i = 0; i < imageData.data.length; i += 4) {
  let r = imageData.data[i], // R value
  g = imageData.data[i + 1], // G value
  b = imageData.data[i + 2], // B value
  a = imageData.data[i + 3]; // Alpha value
}

Die Werte innerhalb des Arrays können manipuliert werden. Da zwischen dem exportierten ImageData-Objekt und dem ursprünglichen Canvas jedoch keine Verbindung besteht, bleiben die eigentlichen Pixel im Canvas unverändert. Um diese anzupassen, muss das manipulierte ImageData-Objekt wieder in den Canvas zurückgeschrieben werden. Ein vollständiges Beispiel zum Anwenden eines Graustufen-Effekts auf den Inhalt eines Canvas könnte so aussehen:

// Info: context enthält die Referenz auf den 2D-Kontext eines Canvas-Elements

// Auslesen des aktuellen Inhalts des Canvas
let imageData = context.getImageData(0, 0, canvas.width, canvas.height);

// Iteration über alle Pixel
for(let i = 0; i < imageData.data.length; i += 4) {
  let r = imageData.data[i],
  g = imageData.data[i + 1], 
  b = imageData.data[i + 2],
  // Berechnen der Graustufe
  grayscale = (r + g + b)/3;
  // Überschreiben der ursprünglichen Farbwerte
  imageData.data[i] = grayscale;
  imageData.data[i + 1] = grayscale;
  imageData.data[i + 2] = grayscale;
  // Der Alpha-Wert bleibt unverändert
}

// Schreiben des modifizierten ImageData-Objekts in den Canvas (an die Positon 0,0)
context.putImageData(imageData, 0, 0); 

Auf dem folgenden Screenshot sehen Sie ein Beispiel für die Anwendung des oben skizzierten Codes. Im linken Canvas wurde ein Bild (Quelle: Lucas V. Barbosa, The Gunk) eingezeichnet. Der Inhalt dieses Canvas wurde anschließend in Form eines ImageData-Objekts ausgelesen, in eine Graustufen-Repräsentation umgewandelt und in dieser Form in den rechten Canvas übertragen.

Beispiel für Graustufen-Effekt im Canvas

Export des Canvas-Inhalts

Neben dem Export der Pixel-Daten kann der aktuelle Inhalt eines Canvas-Elements auch in Form einer Data URL ausgelesen werden. Data URLs erlauben das Kodieren und Bereitstellen kleinerer Dateien. Dabei werden diese nicht mehr durch die URL referenziert sondern innerhalb der URL repräsentiert. Über die Methode toDataURL, die direkt auf dem Canvas-Element - nicht auf dessen Kontext - aufgerufen wird, kann eine solche URL erzeugt werden. Diese enthält dann dessen kodierten Inhalt. Die exportierte Data URL des oben verwendeten Canvas (mit dargestelltem Graustufen-Bild) beginnt mit der folgenden Zeichenkette:



Die URL beginnt mit dem Präfix data: an das sich der mime type (hier: image/png) und die Codierung (hier base64) anschließt. Der Rest der Zeichenkette beinhaltet den eigentlichen Dateiinhalt.

Die so erzeugte URL kann dann verwendet werden, um den codierten Canvas-Inhalt zum Download (als Link) anzubieten oder als Inhalt eines <img>-Elements genutzt werden.

let imageUrl = canvas.toDataURL(),
imageEl = document.querySelector("img");
iamgeEl.src = imageUrl;

Beispiel: Echtzeit-Videoeffekte im Browser

Die Kompatibilität zwischen Canvas- und Video-Element lässt sich nutzen, um z.B. einfache Echtzeit-Videoeffekte direkt im Browser zu realisieren. Ein naiver Ansatz dafür kann nach folgendem Schema implementiert werden:

  1. Das Originalvideo wird in einem <video>-Element abgespielt (optional auch unsichtbar über die CSS-Eigenschaft display: none).

  2. In einem regelmäßigen, an die Wiedergabefrequenz des Videos angepassten, Intervall wird der aktuelle Frame des Videos auf den Canvas gezeichnet.

  3. Nach jedem Zeichenvorgang werden die Pixel-Informationen (ImageData) des Canvas ausgelesen, manipuliert und anschließend zurück in den Canvas geschrieben.

Die wichtigsten Einzelschritte lassen sich mit Hilfe dieser Javascript-Bausteine realisieren:

Vorbereiten von Video- und Canvas-Element

let videoEl = document.querySelector("video"),
  canvasEl = document.querySelctor("canvas"),
  context = canvasEl.getContext("2d");

Regelmäßiges Auslesen und Importieren des Video-Inhalts (als Einzelbild)

// Angenommen wird eine Frequenz von 25 Bildern pro Sekunde
let fps = 1000 / 25;

function drawFrame() {
  // Zeichenen des aktuellen Einzelbildes auf den Canvas
  context.drawImage(videoEl, 0, 0, videoEl.width, videoEl.height);
  // Hier kann nun das ImageData-Objekt exportiert, manipuliert und zurückgeschrieben werden
}

setInterval(drawFrame, fps);

Die tatsächliche frame rate eines Videos kann im Browser und mit Javascript nicht eindeutig bestimmt werden. Ist die Wiedergabefrequenz des verwendeten Videomaterials nicht bekannt, muss diese geschätzt werden. Dies führt unter Umständen dann dazu, dass nicht alle Einzelbilder des Videos exportiert werden, was sich ggf. auch visuell bemerkbar macht.

Anwendungsbeispiel

Das folgende Video zeigt ein einfaches Anwendungsbeispiel für die oben beschriebenen Echtzeit-Video-Effekte. Links sehen Sie ein Video-Element. Die Einzelbilder werden regelmäßig auf den Canvas (rechts) übertragen. Die Farbwerte der Pixel des ImageData-Objekts werden manipuliert, um ein Threshold-Bild zu erzeugen. Alle Pixel, deren Graustufe einen bestimmten Wert überschreiten, werden mit der Farbe Weiß (255,255,255) überschrieben, alle anderen Pixel mit der Farbe Schwarz (0,0,0). Die modifizierten Pixel werden zurück in den Canvas geschrieben:

Beispiel für Graustufen-Effekt im Video

Übungsaufgaben

  1. Erstellen Sie einen einfachen video player: Über ein Text-Eingabefeld kann der Benutzer nacheinander verschiedene Video-URLs eingeben (z.B. URLs aus dieser Liste). Die Videos ergeben eine Playlist, deren Inhalte nacheinander in einem <video>-Element mit angezeigten Standard-Steuerelementen abgespielt wird. Nutzen Sie entsprechende Eigenschaften bzw. Events des <video>-Elements um das automatische Abspielen des jeweils nächsten Eintrags zu realisieren. Beispiel-Videos zum Testen der Anwendung finden Sie hier.

  2. Ergänzen Sie den erstellen video player mit eigenen Schaltflächen für das Starten und Stoppen des aktuellen Videos sowie das Auswählen des nächsten Eintrags der Playlist.

  3. Implementieren Sie eine einfache Web-Anwendung, die es dem Benutzer erlaubt, Zeichnungen zu erstellen. Erzeugen Sie ein HTML-Dokument und fügen Sie diesem ein <canvas>-Element hinzu. Fangen Sie die Mausinteraktion des Nutzers auf diesem Element ab (Vgl. mousemove) und übersetzen Sie die Mausbewegung in eine Zeichenoperation. Verwenden Sie die lineTo-Methode der Canvas-API bzw. um jeweils eine Linie zwischen der letzten bekannten und der aktuellen Position des Mauszeigers zu zeichnen.


  1. W3C, HTML 5.2 

  2. Zusätzlich unterstützt die drawImage-Methode auch weitere, teils experimentelle Quellen