Video- und Audio-APIs
HTML enthält Elemente zum Einbetten von Rich Media in Dokumente — <video> und <audio> — die wiederum ihre eigenen APIs zum Steuern der Wiedergabe, zum Suchen usw. haben. Dieser Artikel zeigt Ihnen, wie Sie gängige Aufgaben wie das Erstellen von benutzerdefinierten Wiedergabesteuerelementen durchführen.
| Voraussetzungen: | Vertrautheit mit HTML, CSS und JavaScript, insbesondere Grundlagen der JavaScript-Objekte und der Abdeckung grundlegender APIs wie DOM-Skripting und Netzwerkanfragen. | 
|---|---|
| Ziel: | 
        
  | 
    
HTML-Video und -Audio
Die <video>- und <audio>-Elemente ermöglichen es uns, Video und Audio in Webseiten einzubetten. Wie wir in HTML-Video und -Audio gezeigt haben, sieht eine typische Implementierung so aus:
<video controls>
  <source src="rabbit320.mp4" type="video/mp4" />
  <source src="rabbit320.webm" type="video/webm" />
  <p>
    Your browser doesn't support HTML video. Here is a
    <a href="rabbit320.mp4">link to the video</a> instead.
  </p>
</video>
Dies erstellt einen Videoplayer im Browser, der so aussieht:
Sie können überprüfen, was alle HTML-Funktionen im oben verlinkten Artikel tun. Für unsere Zwecke ist das interessanteste Attribut controls, das die Standardsumme von Wiedergabesteuerelementen aktiviert. Wenn Sie dies nicht angeben, erhalten Sie keine Wiedergabesteuerelemente:
Dies ist für die Videowiedergabe nicht sofort nützlich, hat aber Vorteile. Ein großes Problem mit den nativen Browser-Steuerelementen ist, dass sie in jedem Browser unterschiedlich sind — nicht sehr gut für die plattformübergreifende Unterstützung! Ein weiteres großes Problem ist, dass die nativen Steuerelemente in den meisten Browsern nicht sehr tastaturzugänglich sind.
Sie können diese beiden Probleme lösen, indem Sie die nativen Steuerelemente ausblenden (durch Entfernen des controls-Attributs) und Ihre eigenen mit HTML, CSS und JavaScript programmieren. Im nächsten Abschnitt sehen wir uns die grundlegenden Tools an, die wir zur Verfügung haben, um dies zu tun.
Die HTMLMediaElement API
Als Teil der HTML-Spezifikation bietet die API HTMLMediaElement Funktionen, mit denen Sie Video- und Audioplayer programmatisch steuern können, z. B. HTMLMediaElement.play(), HTMLMediaElement.pause() usw. Diese Schnittstelle steht sowohl <audio>- als auch <video>-Elementen zur Verfügung, da die Funktionen, die Sie implementieren möchten, nahezu identisch sind. Lassen Sie uns ein Beispiel durchgehen, in dem wir Funktionen hinzufügen, während wir fortfahren.
Unser fertiges Beispiel wird folgendermaßen aussehen (und funktionieren):
Einstieg
Um mit diesem Beispiel zu beginnen, führen Sie folgende Schritte aus:
- 
Erstellen Sie ein neues Verzeichnis auf Ihrer Festplatte namens
custom-video-player. - 
Erstellen Sie eine neue Datei darin mit dem Namen
index.htmlund füllen Sie sie mit folgendem Inhalt:html<!doctype html> <html lang="en-gb"> <head> <meta charset="utf-8" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <title>Video player example</title> <link rel="stylesheet" type="text/css" href="style.css" /> </head> <body> <div class="player"> <video controls> <source src="/shared-assets/videos/sintel-short.mp4" type="video/mp4" /> <source src="/shared-assets/videos/sintel-short.webm" type="video/webm" /> </video> <div class="controls"> <button class="play" data-icon="P" aria-label="play pause toggle"></button> <button class="stop" data-icon="S" aria-label="stop"></button> <div class="timer"> <div></div> <span>00:00</span> </div> <button class="rwd" data-icon="B" aria-label="rewind"></button> <button class="fwd" data-icon="F" aria-label="fast forward"></button> </div> </div> <p> Sintel © copyright Blender Foundation | <a href="https://studio.blender.org/films/sintel/" >studio.blender.org/films/sintel/</a >. </p> <script src="custom-player.js"></script> </body> </html> - 
Erstellen Sie eine weitere neue Datei darin mit dem Namen
style.cssund füllen Sie sie mit folgendem Inhalt:css@font-face { font-family: "HeydingsControlsRegular"; src: url("https://mdn.github.io/learning-area/javascript/apis/video-audio/finished/fonts/heydings_controls-webfont.eot"); src: url("https://mdn.github.io/learning-area/javascript/apis/video-audio/finished/fonts/heydings_controls-webfont.eot?#iefix") format("embedded-opentype"), url("https://mdn.github.io/learning-area/javascript/apis/video-audio/finished/fonts/heydings_controls-webfont.woff") format("woff"), url("https://mdn.github.io/learning-area/javascript/apis/video-audio/finished/fonts/heydings_controls-webfont.ttf") format("truetype"); font-weight: normal; font-style: normal; } video { border: 1px solid black; } p { position: absolute; top: 310px; } .player { position: absolute; } .controls { visibility: hidden; opacity: 0.5; width: 400px; border-radius: 10px; position: absolute; bottom: 20px; left: 50%; margin-left: -200px; background-color: black; box-shadow: 3px 3px 5px black; transition: 1s all; display: flex; } .player:hover .controls, .player:focus-within .controls { opacity: 1; } button, .controls { background: linear-gradient(to bottom, #222222, #666666); } button::before { font-family: "HeydingsControlsRegular"; font-size: 20px; position: relative; content: attr(data-icon); color: #aaaaaa; text-shadow: 1px 1px 0px black; } .play::before { font-size: 22px; } button, .timer { height: 38px; line-height: 19px; box-shadow: inset 0 -5px 25px #0000004d; border-right: 1px solid #333333; } button { position: relative; border: 0; flex: 1; outline: none; } .play { border-radius: 10px 0 0 10px; } .fwd { border-radius: 0 10px 10px 0; } .timer { line-height: 38px; font-size: 10px; font-family: monospace; text-shadow: 1px 1px 0px black; color: white; flex: 5; position: relative; } .timer div { position: absolute; background-color: rgb(255 255 255 / 20%); left: 0; top: 0; width: 0; height: 38px; z-index: 2; } .timer span { position: absolute; z-index: 3; left: 19px; } button:hover, button:focus { box-shadow: inset 1px 1px 2px black; } button:active { box-shadow: inset 3px 3px 2px black; } .active::before { color: red; } - 
Erstellen Sie eine weitere neue Datei in dem Verzeichnis namens
custom-player.js. Lassen Sie sie vorerst leer. 
Zu diesem Zeitpunkt sollten Sie beim Laden des HTML einen vollkommen normalen HTML-Videoplayer mit den gerenderten nativen Steuerelementen sehen.
Erforschung des HTML
Öffnen Sie die HTML-Indexdatei. Sie werden eine Reihe von Funktionen sehen; das HTML wird vom Videoplayer und seinen Steuerelementen dominiert:
- Der gesamte Player ist in ein 
<div>-Element eingebettet, damit er bei Bedarf als eine Einheit gestylt werden kann. - Das 
<video>-Element enthält zwei<source>-Elemente, sodass je nach Ansicht des Browsers verschiedene Formate geladen werden können. - Das HTML der Steuerelemente ist vermutlich am interessantesten:
- Wir haben vier 
<button>s — abspielen/pause, stoppen, zurückspulen und vorspulen. - Jeder 
<button>hat einenclass-Namen, eindata-icon-Attribut zur Definition, welches Icon auf jedem Button angezeigt werden soll (wir zeigen, wie dies im folgenden Abschnitt funktioniert), und einaria-label-Attribut, um eine verständliche Beschreibung jedes Buttons bereitzustellen, da wir innerhalb der Tags keine menschenlesbaren Bezeichnungen bereitstellen. Der Inhalt vonaria-label-Attributen wird von Bildschirmlesern vorgelesen, wenn ihre Benutzer sich auf die sie enthaltenden Elemente konzentrieren. - Es gibt auch ein Timer-
<div>, das die vergangene Zeit anzeigt, wenn das Video abgespielt wird. Nur zum Spaß bieten wir zwei Berichtmechanismen — ein<span>, das die verstrichene Zeit in Minuten und Sekunden enthält, und ein zusätzliches<div>, mit dem wir eine horizontale Indikatorleiste erstellen werden, die länger wird, während die Zeit verstreicht. 
 - Wir haben vier 
 
Erforschung des CSS
Öffnen Sie nun die CSS-Datei und sehen Sie sich darin um. Das CSS für das Beispiel ist nicht allzu kompliziert, aber wir heben hier die interessantesten Teile hervor. Zuerst einmal beachten wir das .controls-Styling:
.controls {
  visibility: hidden;
  opacity: 0.5;
  width: 400px;
  border-radius: 10px;
  position: absolute;
  bottom: 20px;
  left: 50%;
  margin-left: -200px;
  background-color: black;
  box-shadow: 3px 3px 5px black;
  transition: 1s all;
  display: flex;
}
.player:hover .controls,
.player:focus-within .controls {
  opacity: 1;
}
- Wir beginnen mit der 
visibilityder benutzerdefinierten Steuerelemente, die standardmäßig aufhiddengesetzt ist. In unserem JavaScript später stellen wir die Steuerelemente aufvisibleund entfernen dascontrols-Attribut vom<video>-Element. Dies dient dazu, dass Benutzer das Video auch dann mit den nativen Steuerelementen nutzen können, wenn das JavaScript aus irgendeinem Grund nicht geladen wird. - Wir geben den Steuerelementen standardmäßig eine 
opacityvon0.5, damit sie weniger ablenkend sind, wenn Sie versuchen, das Video anzusehen. Nur wenn Sie über den Player schweben/fokussieren, erscheinen die Steuerelemente mit voller Deckkraft. - Wir gestalten die Tasten innerhalb der Steuerleiste mithilfe von Flexbox (
display: flex), um die Sache zu vereinfachen. 
Schauen wir uns als Nächstes unsere Button-Symbole an:
@font-face {
  font-family: "HeydingsControlsRegular";
  src: url("https://mdn.github.io/learning-area/javascript/apis/video-audio/finished/fonts/heydings_controls-webfont.eot");
  src:
    url("https://mdn.github.io/learning-area/javascript/apis/video-audio/finished/fonts/heydings_controls-webfont.eot?#iefix")
      format("embedded-opentype"),
    url("https://mdn.github.io/learning-area/javascript/apis/video-audio/finished/fonts/heydings_controls-webfont.woff")
      format("woff"),
    url("https://mdn.github.io/learning-area/javascript/apis/video-audio/finished/fonts/heydings_controls-webfont.ttf")
      format("truetype");
  font-weight: normal;
  font-style: normal;
}
button::before {
  font-family: "HeydingsControlsRegular";
  font-size: 20px;
  position: relative;
  content: attr(data-icon);
  color: #aaaaaa;
  text-shadow: 1px 1px 0px black;
}
Zuerst verwenden wir im oberen Bereich des CSS einen @font-face-Block, um eine benutzerdefinierte Webschriftart zu importieren. Dies ist eine Symbolschriftart — alle Buchstaben des Alphabets entsprechen gängigen Symbolen, die Sie möglicherweise in einer Anwendung verwenden möchten.
Als Nächstes verwenden wir generierten Inhalt, um ein Symbol auf jedem Button anzuzeigen:
- Wir verwenden den 
::before-Selektor, um den Inhalt vor jedem<button>-Element anzuzeigen. - Wir verwenden die 
content-Eigenschaft, um den in jedem Fall anzuzeigenden Inhalt gleich den Inhalten desdata-icon-Attributs festzulegen. Im Fall unseres Abspielbuttons enthältdata-iconein großes "P". - Wir verwenden die benutzerdefinierte Webschriftart für unsere Buttons mit 
font-family. In dieser Schriftart ist "P" tatsächlich ein "Abspielen"-Symbol, daher hat der Abspielbutton ein "Abspielen"-Symbol darauf angezeigt. 
Symbolschriftarten sind aus vielen Gründen cool — sie reduzieren HTTP-Anfragen, da Sie diese Symbole nicht als Bilddateien herunterladen müssen, sie bieten großartige Skalierbarkeit und Sie können Texteigenschaften verwenden, um sie zu gestalten — wie color und text-shadow.
Zu guter Letzt werfen wir einen Blick auf das CSS für den Timer:
.timer {
  line-height: 38px;
  font-size: 10px;
  font-family: monospace;
  text-shadow: 1px 1px 0px black;
  color: white;
  flex: 5;
  position: relative;
}
.timer div {
  position: absolute;
  background-color: rgb(255 255 255 / 20%);
  left: 0;
  top: 0;
  width: 0;
  height: 38px;
  z-index: 2;
}
.timer span {
  position: absolute;
  z-index: 3;
  left: 19px;
}
- Wir setzen das äußere 
.timer-Element aufflex: 5, damit es den größten Teil der Breite der Steuerleiste einnimmt. Wir geben ihm auchposition: relative, damit wir Elemente darin bequem entsprechend seinen Grenzen und nicht den Grenzen des<body>-Elements positionieren können. - Das innere 
<div>ist absolut positioniert, um direkt oben auf dem äußeren<div>zu sitzen. Es hat auch eine anfängliche Breite von 0, damit Sie es überhaupt nicht sehen können. Während das Video abgespielt wird, wird die Breite mit JavaScript erhöht, während das Video abläuft. - Das 
<span>ist ebenfalls absolut positioniert, um sich nahe der linken Seite der Timerleiste zu befinden. - Wir geben auch unserem inneren 
<div>und<span>den richtigenz-index, damit der Timer oben angezeigt wird und das innere<div>darunter. So stellen wir sicher, dass wir alle Informationen sehen können – ein Kasten verdeckt keinen anderen. 
Implementierung des JavaScripts
Wir haben bereits eine ziemlich vollständige HTML- und CSS-Schnittstelle; jetzt müssen wir nur noch alle Buttons verkabeln, um die Kontrollen zum Laufen zu bringen.
- 
Fügen Sie am Anfang der Datei
custom-player.jsden folgenden Code ein:jsconst media = document.querySelector("video"); const controls = document.querySelector(".controls"); const play = document.querySelector(".play"); const stop = document.querySelector(".stop"); const rwd = document.querySelector(".rwd"); const fwd = document.querySelector(".fwd"); const timerWrapper = document.querySelector(".timer"); const timer = document.querySelector(".timer span"); const timerBar = document.querySelector(".timer div");Hier erstellen wir Konstanten, um Referenzen zu allen Objekten zu halten, die wir manipulieren wollen. Wir haben drei Gruppen:
- Das 
<video>-Element und die Steuerleiste. - Die Play/Pause-, Stop-, Rückspul- und Vorspulknöpfe.
 - Das äußere Timer-Wrapper-
<div>, die digitale Timeranzeige-<span>und das innere<div>, das breiter wird, während die Zeit verstreicht. 
 - Das 
 - 
Fügen Sie dann das Folgende am Ende Ihres Codes hinzu:
jsmedia.removeAttribute("controls"); controls.style.visibility = "visible";Diese beiden Zeilen entfernen die Standardbrowsersteuerelemente vom Video und machen die benutzerdefinierten Steuerelemente sichtbar.
 
Abspielen und Pausieren des Videos
Lassen Sie uns wahrscheinlich die wichtigste Kontrolle implementieren — den Abspiel-/Pause-Knopf.
- 
Fügen Sie zunächst das Folgende am Ende Ihres Codes hinzu, so dass die Funktion
playPauseMedia()aufgerufen wird, wenn der Abspielknopf geklickt wird:jsplay.addEventListener("click", playPauseMedia); - 
Nun zur Definition von
playPauseMedia()— fügen Sie das Folgende wieder am Ende Ihres Codes hinzu:jsfunction playPauseMedia() { if (media.paused) { play.setAttribute("data-icon", "u"); media.play(); } else { play.setAttribute("data-icon", "P"); media.pause(); } }Hier verwenden wir eine
if-Anweisung, um zu prüfen, ob das Video pausiert ist. DieHTMLMediaElement.paused-Eigenschaft gibt true zurück, wenn das Medium pausiert ist, was jedes Mal der Fall ist, wenn das Video nicht abgespielt wird, einschließlich wenn es nach dem ersten Laden auf 0 Dauer eingestellt ist. Wenn es pausiert ist, setzen wir den Wert desdata-icon-Attributs auf den Play-Knopf auf "u", was ein "Pausiert"-Symbol ist, und rufen die MethodeHTMLMediaElement.play()auf, um das Medium abzuspielen.Beim zweiten Klick wird der Button wieder zurückgeschaltet — das "Abspielen"-Symbol wird erneut angezeigt und das Video wird mit
HTMLMediaElement.pause()pausiert. 
Stoppen des Videos
- 
Lassen Sie uns als Nächstes die Funktionalität zum Stoppen des Videos hinzufügen. Fügen Sie die folgenden
addEventListener()-Zeilen unter den vorherigen hinzu, die Sie hinzugefügt haben:jsstop.addEventListener("click", stopMedia); media.addEventListener("ended", stopMedia);Das
click-Event ist offensichtlich — wir möchten das Video stoppen, indem wir die FunktionstopMedia()ausführen, wenn der Stop-Button geklickt wird. Wir möchten jedoch auch das Video stoppen, wenn es zu Ende ist — dies wird durch das Auslösen desended-Events angezeigt, daher richten wir auch einen Listener ein, um die Funktion bei Auslösung dieses Events auszuführen. - 
Lassen Sie uns nun
stopMedia()definieren — fügen Sie die folgende Funktion unterplayPauseMedia()hinzu:jsfunction stopMedia() { media.pause(); media.currentTime = 0; play.setAttribute("data-icon", "P"); }Die HTMLMediaElement-API hat keine
stop()-Methode — das Äquivalent ist, das Video zupause()und seinecurrentTime-Eigenschaft auf 0 zu setzen. DascurrentTimeauf einen Wert (in Sekunden) zu setzen, springt das Medium sofort auf diese Position.Danach bleibt nur noch, das angezeigte Symbol auf das "Abspielen"-Symbol zu setzen. Unabhängig davon, ob das Video pausiert oder abgespielt wurde, als der Stopp-Button gedrückt wurde, möchten Sie, dass es danach bereit ist, abgespielt zu werden.
 
Vor- und Zurückspulen
Es gibt viele Möglichkeiten, wie Sie Rückspul- und Vorspul-Funktionalitäten implementieren können; hier zeigen wir Ihnen eine relativ komplexe Methode, bei der das Player nicht kaputtgeht, wenn die verschiedenen Tasten in unerwarteter Reihenfolge gedrückt werden.
- 
Fügen Sie zunächst die folgenden zwei
addEventListener()-Zeilen unter den vorherigen hinzu:jsrwd.addEventListener("click", mediaBackward); fwd.addEventListener("click", mediaForward); - 
Nun zu den Ereignishandlungsfunktionen — fügen Sie den folgenden Code unter Ihren vorherigen Funktionen ein, um
mediaBackward()undmediaForward()zu definieren:jslet intervalFwd; let intervalRwd; function mediaBackward() { clearInterval(intervalFwd); fwd.classList.remove("active"); if (rwd.classList.contains("active")) { rwd.classList.remove("active"); clearInterval(intervalRwd); media.play(); } else { rwd.classList.add("active"); media.pause(); intervalRwd = setInterval(windBackward, 200); } } function mediaForward() { clearInterval(intervalRwd); rwd.classList.remove("active"); if (fwd.classList.contains("active")) { fwd.classList.remove("active"); clearInterval(intervalFwd); media.play(); } else { fwd.classList.add("active"); media.pause(); intervalFwd = setInterval(windForward, 200); } }Sie werden bemerken, dass wir zuerst zwei Variablen initialisieren —
intervalFwdundintervalRwd— mehr dazu später.Lassen Sie uns
mediaBackward()durchgehen (die Funktionalität vonmediaForward()ist genau dieselbe, aber umgekehrt):- Wir löschen alle Klassen und Intervalle, die in der Vorspielfunktion festgelegt sind — wir tun dies, weil wir, wenn wir die 
rwd-Taste nach derfwd-Taste drücken, jede Vorspielfunktionalität abbrechen und durch die Rückspielfunktionalität ersetzen möchten. Wenn wir versuchen, beides gleichzeitig zu tun, würde der Player kaputtgehen. - Wir verwenden eine 
if-Anweisung, um zu prüfen, ob dieactive-Klasse auf denrwd-Button gesetzt ist, was darauf hinweist, dass er bereits gedrückt wurde. DieclassListist eine ziemlich nützliche Eigenschaft, die auf jedem Element existiert — sie enthält eine Liste aller auf dem Element festgelegten Klassen sowie Methoden zum Hinzufügen/Entfernen von Klassen usw. Wir verwenden die MethodeclassList.contains(), um zu überprüfen, ob die Liste dieactive-Klasse enthält. Dies liefert ein booleschestrue/false-Ergebnis zurück. - Wenn 
activeauf demrwd-Button gesetzt ist, entfernen wir es mitclassList.remove(), löschen das Intervall, das beim ersten Drücken der Taste festgelegt wurde (siehe unten für mehr Erklärung), und verwendenHTMLMediaElement.play(), um das Rückspulen abzubrechen und das Video normal abzuspielen. - Wenn es noch nicht gesetzt ist, fügen wir die Klasse 
activezumrwd-Button hinzu, indem wirclassList.add()verwenden, pausieren das Video mitHTMLMediaElement.pause()und setzen dann die VariableintervalRwdgleich einemsetInterval()-Aufruf. Wenn aufgerufen, erstelltsetInterval()ein aktives Intervall, was bedeutet, dass es die Funktion ausführt, die als erstes Parameter angegeben wird, alle x Millisekunden, wobei x der Wert des 2. Parameters ist. Hier führen wir also diewindBackward()-Funktion alle 200 Millisekunden aus — wir verwenden diese Funktion, um das Video kontinuierlich rückwärts zu spulen. Um ein laufendessetInterval()zu stoppen, müssen SieclearInterval()aufrufen und den Identifikationsname des zu löschenden Intervalls angeben, was in diesem Fall der VariablennameintervalRwdist (siehe denclearInterval()-Aufruf früher in der Funktion). 
 - Wir löschen alle Klassen und Intervalle, die in der Vorspielfunktion festgelegt sind — wir tun dies, weil wir, wenn wir die 
 - 
Schließlich müssen wir die
windBackward()- undwindForward()-Funktionen definieren, die in densetInterval()-Aufrufen ausgeführt werden. Fügen Sie das Folgende unter Ihre beiden vorherigen Funktionen hinzu:jsfunction windBackward() { if (media.currentTime <= 3) { rwd.classList.remove("active"); clearInterval(intervalRwd); stopMedia(); } else { media.currentTime -= 3; } } function windForward() { if (media.currentTime >= media.duration - 3) { fwd.classList.remove("active"); clearInterval(intervalFwd); stopMedia(); } else { media.currentTime += 3; } }Wir werden nur durch die erste dieser Funktionen gehen, da sie fast identisch sind, aber in umgekehrter Richtung zueinander arbeiten. In
windBackward()führen wir Folgendes durch — beachten Sie, dass diese Funktion, wenn das Intervall aktiv ist, einmal alle 200 Millisekunden ausgeführt wird.- Wir beginnen mit einer 
if-Anweisung, die überprüft, ob die aktuelle Zeit weniger als 3 Sekunden beträgt, d.h. wenn das Rückwärtsspulen um weitere drei Sekunden es hinter den Anfang des Videos zurückführen würde. Dies würde seltsames Verhalten verursachen, also stoppen wir in diesem Fall das Video, indem wirstopMedia()aufrufen, entfernen dieactive-Klasse vom Rückspulbutton und löschen das IntervallintervalRwd, um die Rückspul-Funktionalität zu stoppen. Wenn wir diesen letzten Schritt nicht machen würden, würde das Video weiterhin für immer zurückgespult werden. - Wenn die aktuelle Zeit nicht innerhalb von 3 Sekunden nach dem Start des Videos ist, entfernen wir drei Sekunden von der aktuellen Zeit, indem wir 
media.currentTime -= 3ausführen. So spulen wir das Video effektiv alle 200 Millisekunden um 3 Sekunden zurück. 
 - Wir beginnen mit einer 
 
Aktualisierung der verstrichenen Zeit
Das letzte Stück unseres Media-Players, das implementiert werden muss, sind die Zeitverläufe. Dazu führen wir eine Funktion aus, um die Zeitverläufe jedes Mal zu aktualisieren, wenn das timeupdate-Event auf dem <video>-Element ausgelöst wird. Die Häufigkeit, mit der dieses Event ausgelöst wird, hängt von Ihrem Browser, der Leistung der CPU usw. ab (siehe diesen Stack Overflow-Beitrag).
- 
Fügen Sie die folgende
addEventListener()-Zeile direkt unter die anderen hinzu:jsmedia.addEventListener("timeupdate", setTime); - 
Nun zur Definition der
setTime()-Funktion. Fügen Sie das Folgende am unteren Ende Ihrer Datei hinzu:jsfunction setTime() { const minutes = Math.floor(media.currentTime / 60); const seconds = Math.floor(media.currentTime - minutes * 60); const minuteValue = minutes.toString().padStart(2, "0"); const secondValue = seconds.toString().padStart(2, "0"); const mediaTime = `${minuteValue}:${secondValue}`; timer.textContent = mediaTime; const barLength = timerWrapper.clientWidth * (media.currentTime / media.duration); timerBar.style.width = `${barLength}px`; } 
Dies ist eine ziemlich lange Funktion, daher gehen wir sie Schritt für Schritt durch:
- Zuerst berechnen wir die Anzahl der Minuten und Sekunden im Wert 
HTMLMediaElement.currentTime. - Dann initialisieren wir zwei weitere Variablen — 
minuteValueundsecondValue. Wir verwendenpadStart(), um jeden Wert auf 2 Zeichenlänge zu bringen, selbst wenn der Wert nur eine einzige Ziffer ist. - Der tatsächliche anzuzeigende Zeitwert wird festgelegt als 
minuteValueplus einem Doppelpunkt-Zeichen plussecondValue. - Der 
Node.textContent-Wert des Timers wird auf den Zeitwert gesetzt, sodass er in der Benutzeroberfläche angezeigt wird. - Die Länge, auf die wir das innere 
<div>setzen sollten, wird bestimmt, indem zunächst die Breite des äußeren<div>herausgefunden wird (dieclientWidth-Eigenschaft eines Elements enthält seine Länge) und sie dann mitHTMLMediaElement.currentTimegeteilt durch die gesamteHTMLMediaElement.durationdes Mediums multipliziert. - Wir setzen die Breite des inneren 
<div>auf die berechnete Balkenlänge plus "px", damit es auf diese Anzahl von Pixeln gesetzt wird. 
Beheben von Abspielen und Pausieren
Es gibt ein Problem, das noch zu beheben ist. Wenn die Abspiel-/Pause- oder Stopptasten gedrückt werden, während die Rück- oder Vorspielfunktionalität aktiv ist, funktionieren sie einfach nicht. Wie können wir es so reparieren, dass sie die rwd/fwd-Tasten-Funktionalität stornieren und das Video wie erwartet abspielen/stoppen? Dies ist ziemlich einfach zu beheben.
- 
Fügen Sie zunächst die folgenden Zeilen in der
stopMedia()-Funktion ein — es ist egal, wo:jsrwd.classList.remove("active"); fwd.classList.remove("active"); clearInterval(intervalRwd); clearInterval(intervalFwd); - 
Fügen Sie nun dieselben Zeilen noch einmal ganz am Anfang der
playPauseMedia()-Funktion (direkt vor Beginn derif-Anweisung) ein. - 
An diesem Punkt können Sie die entsprechenden Zeilen aus den
windBackward()- undwindForward()-Funktionen löschen, da diese Funktionalität stattdessen in derstopMedia()-Funktion implementiert wurde. 
Hinweis: Sie könnten auch die Effizienz des Codes weiter verbessern, indem Sie eine separate Funktion erstellen, die diese Zeilen ausführt und dann überall dort aufrufen, wo es nötig ist, anstatt die Zeilen mehrmals im Code zu wiederholen. Aber wir überlassen das Ihnen.
Zusammenfassung
Ich denke, wir haben Ihnen in diesem Artikel genug beigebracht. Die HTMLMediaElement-API bietet eine Fülle von Funktionalitäten zur Erstellung einfacher Video- und Audioplayer, und das ist nur die Spitze des Eisbergs. Siehe den Abschnitt "Siehe auch" unten für Links zu komplexerer und interessanter Funktionalität.
Hier sind einige Vorschläge, wie Sie das bestehende Beispiel, das wir aufgebaut haben, verbessern könnten:
- 
Die Zeitanzeige bricht derzeit, wenn das Video eine Stunde oder länger dauert (nun, es zeigt keine Stunden an; nur Minuten und Sekunden). Können Sie herausfinden, wie Sie das Beispiel ändern können, um Stunden anzuzeigen?
 - 
Da
<audio>-Elemente dieselbeHTMLMediaElement-Funktionalität zur Verfügung haben, könnten Sie diesen Player auch für ein<audio>-Element arbeiten lassen. Versuchen Sie es. - 
Können Sie eine Möglichkeit finden, das innere Timer-
<div>-Element in eine echte Suchleiste/Scroller umzuwandeln — d.h. wenn Sie irgendwo auf die Leiste klicken, springt es zu dieser relativen Position in der Videowiedergabe? Als Hinweis, Sie können die X- und Y-Werte der Linken/Rechten und Oben/Unten-Seiten eines Elements über diegetBoundingClientRect()-Methode herausfinden, und Sie können die Koordinaten eines Mausklicks über das Ereignisobjekt des Klick-Ereignisses herausfinden, das auf demDocument-Objekt aufgerufen wird. Beispielsweise:jsdocument.onclick = function (e) { console.log(e.x, e.y); }; 
Siehe auch
HTMLMediaElement- HTML-Video und -Audio — einfacher Leitfaden zu 
<video>- und<audio>-HTML. - Audio- und Videowiedergabe — detaillierter Leitfaden zur Wiedergabe von Medien im Browser, mit vielen Tipps, Tricks und Links zu weiter fortgeschrittenen Tutorials.
 - Manipulation von Audio und Video — detaillierter Leitfaden zur Bearbeitung von Audio und Video, z. B. mit der Canvas API, der Web Audio API und mehr.
 <video>und<audio>Referenzseiten.- Leitfaden zu Medientypen und Formaten im Web