--- title: Manipulation Audio et Vidéo slug: Web/Guide/Audio_and_video_manipulation tags: - Audio - Canvas - HTML5 - Video - Web Audio API - WebGL translation_of: Web/Guide/Audio_and_video_manipulation ---
La beauté du web est qu'on peut combiner différentes technologies pour en créer de nouvelles. Avoir de l'audio et vidéo nativement dans le navigateur nous donne la possibilité d'utiliser ces flux de données avec d'autres technologies comme {{htmlelement("canvas")}}, WebGL ou Web Audio API pour modifier le média — par exemple ajouter des effets de réverbération ou de compression à l'audio, ou encore des filtres noir & blanc/sépia aux vidéos. Cet article fournit une référence pour expliquer ce que vous pouvez faire.
La possibilité de lire les valeurs de pixels de chaque image d'une vidéo peut être très utile, cela nous permet de placer ces images dans d'autres contextes.
{{htmlelement("canvas")}} est un moyen de dessiner de manière scripté sur des pages web; c'est un outil très puissant et qui peut être couplé avec du contenu vidéo.
La technique générale est comme suit:
<canvas>
et les manipuler.<canvas>
que l'on veut afficher.On peut configurer notre lecteur vidéo et l'élément <canvas>
comme ceci:
<video id="my-video" controls="true" width="480" height="270" crossorigin="anonymous"> <source src="http://jplayer.org/video/webm/Big_Buck_Bunny_Trailer.webm" type="video/webm"> <source src="http://jplayer.org/video/m4v/Big_Buck_Bunny_Trailer.m4v" type="video/mp4"> </video> <canvas id="my-canvas" width="480" height="270"></canvas>
Et les manipuler comme ceci: (en l'occurence, on affiche une version en noir et blanc de la vidéo)
var processor = { timerCallback: function() { if (this.video.paused || this.video.ended) { return; } this.computeFrame(); var self = this; setTimeout(function () { self.timerCallback(); }, 16); // roughly 60 frames per second }, doLoad: function() { this.video = document.getElementById("my-video"); this.c1 = document.getElementById("my-canvas"); this.ctx1 = this.c1.getContext("2d"); var self = this; this.video.addEventListener("play", function() { self.width = self.video.width; self.height = self.video.height; self.timerCallback(); }, false); }, computeFrame: function() { this.ctx1.drawImage(this.video, 0, 0, this.width, this.height); var frame = this.ctx1.getImageData(0, 0, this.width, this.height); var l = frame.data.length / 4; for (var i = 0; i < l; i++) { var grey = (frame.data[i * 4 + 0] + frame.data[i * 4 + 1] + frame.data[i * 4 + 2]) / 3; frame.data[i * 4 + 0] = grey; frame.data[i * 4 + 1] = grey; frame.data[i * 4 + 2] = grey; } this.ctx1.putImageData(frame, 0, 0); return; } };
Une fois que la page est chargée, on peut appeler
processor.doLoad()
{{EmbedLiveSample("Vidéo_et_Canvas", '100%', 550)}}
Note: En raison de problèmes de sécurité potentiels, si votre vidéo se trouve sur un domaine différent de votre page, vous devez activer CORS (Cross Origin Resource Sharing) sur le serveur qui héberge la vidéo et utiliser l'attribut crossorigin
sur la balise vidéo.
Note: L'exemple présenté est un exemple minimal de manipulation vidéo avec canvas; pour plus d'efficacité, vous pouvez envisager d'utiliser requestAnimationFrame à la place de setTimeout pour les navigateurs qui le prennent en charge.
WebGL est une API puissante qui utilise canvas pour (typiquement) afficher des scènes en trois dimensions. On peut combiner WebGL et l'élément {{htmlelement("video")}} pour créer des textures vidéo, ce qui veut dire que vous pouvez placer une vidéo dans des scènes 3D.
Exemple:
{{EmbedGHLiveSample('webgl-examples/tutorial/sample8/index.html', 670, 510) }}
Note: Vous pouvez trouver le code source de cette démo sur GitHub (la voir en direct aussi).
On peut ajuster la vitesse de lecture de l'audio et vidéo en utilisant l'attribut playbackRate
(voir {{domxref("HTMLMediaElement")}}). Il prend pour valeur un nombre qui est le coefficient à appliquer à la vitesse de lecture: par exemple, 0.5 représente la moitié de la vitesse tandis que 2 représente le double.
HTML:
<video id="my-video" controls src="http://jplayer.org/video/m4v/Big_Buck_Bunny_Trailer.m4v"></video>
JavaScript:
var myVideo = document.getElementById('my-video'); myVideo.playbackRate = 2;
<video id="my-video" controls="true" width="480" height="270"> <source src="http://jplayer.org/video/webm/Big_Buck_Bunny_Trailer.webm" type="video/webm"> <source src="http://jplayer.org/video/m4v/Big_Buck_Bunny_Trailer.m4v" type="video/mp4"> </video> <div class="playable-buttons"> <input id="edit" type="button" value="Edit" /> <input id="reset" type="button" value="Reset" /> </div> <textarea id="code" class="playable-code"> var myVideo = document.getElementById('my-video'); myVideo.playbackRate = 2;</textarea>
var textarea = document.getElementById('code'); var reset = document.getElementById('reset'); var edit = document.getElementById('edit'); var code = textarea.value; function setPlaybackRate() { eval(textarea.value); } reset.addEventListener('click', function() { textarea.value = code; setPlaybackRate(); }); edit.addEventListener('click', function() { textarea.focus(); }) textarea.addEventListener('input', setPlaybackRate); window.addEventListener('load', setPlaybackRate);
{{ EmbedLiveSample('Playable_code', 700, 425) }}
Note: Essayez l' exemple playbackRate en direct.
Note : playbackRate
marche avec les éléments <audio>
et
; cependant, dans les deux cas, la vitesse change mais pas la hauteur du son. Pour manipuler la hauteur du son, vous devez utliliser l'API Web Audio — voir la propriété {{domxref("AudioBufferSourceNode.playbackRate")}}.<video>
Laissons playbackRate
de côté. Pour manipuler l'audio, on utilise typiquement l'API Web Audio.
On peut utiliser la piste audio d'un élément {{htmlelement("audio")}} ou {{htmlelement("video")}} comme source pour alimenter l'API Web Audio, ou un simple buffer audio, une onde sinusoïdale/oscillateur, un flux (comme getUserMedia de WebRTC)... Découvrez exactement comment les utiliser en lisant les pages suivantes:
L'API Web Audio a beaucoup de différents filtres/effets qui peuvent être appliqués à l'audio en utilisant {{domxref("BiquadFilterNode")}}, par exemple:
HTML:
<video id="my-video" controls src="myvideo.mp4" type="video/mp4"></video>
JavaScript:
var context = new AudioContext(), audioSource = context.createMediaElementSource(document.getElementById("my-video")), filter = context.createBiquadFilter(); audioSource.connect(filter); filter.connect(context.destination); // Configure filter filter.type = "lowshelf"; filter.frequency.value = 1000; filter.gain.value = 25;
<video id="my-video" controls="true" width="480" height="270" crossorigin="anonymous"> <source src="http://jplayer.org/video/webm/Big_Buck_Bunny_Trailer.webm" type="video/webm"> <source src="http://jplayer.org/video/m4v/Big_Buck_Bunny_Trailer.m4v" type="video/mp4"> </video> <div class="playable-buttons"> <input id="edit" type="button" value="Edit" /> <input id="reset" type="button" value="Reset" /> </div> <textarea id="code" class="playable-code"> filter.type = "lowshelf"; filter.frequency.value = 1000; filter.gain.value = 25;</textarea>
var context = new AudioContext(), audioSource = context.createMediaElementSource(document.getElementById("my-video")), filter = context.createBiquadFilter(); audioSource.connect(filter); filter.connect(context.destination); var textarea = document.getElementById('code'); var reset = document.getElementById('reset'); var edit = document.getElementById('edit'); var code = textarea.value; function setFilter() { eval(textarea.value); } reset.addEventListener('click', function() { textarea.value = code; setFilter(); }); edit.addEventListener('click', function() { textarea.focus(); }) textarea.addEventListener('input', setFilter); window.addEventListener('load', setFilter);
{{ EmbedLiveSample('Playable_code_2', 700, 425) }}
Note: À moins que CORS ne soit activé, vous devrez pour éviter les problèmes de sécurité placer la vidéo sur le même domaine que votre code.
Les filtres pouvant être appliqués sont:
Note: Voir {{domxref("BiquadFilterNode")}} pour plus d'informations.
Il est également possible d'appliquer des réponses impulsionnelles à l'audio en utilisant {{domxref("ConvolverNode")}}. Une réponse impulsionnelle (impulse response en anglais) est un son crée après une brève impulsion sonore (comme un applaudissement) et qui s'applique sur l'environnement qui l'a créée. Exemple: un écho crée en frappant des mains dans un tunnel.
Exemple:
var convolver = context.createConvolver(); convolver.buffer = this.impulseResponseBuffer; // Connect the graph. source.connect(convolver); convolver.connect(context.destination);
Note: Voir ce Codepen pour un exemple appliqué.
Note: Voir {{domxref("ConvolverNode")}} pour plus d'informations.
On peut également positionner l'audio dans l'espace en utilisant un noeud panoramique (un panner). Ce noeud permet de définir un cône source ainsi que des éléments positionnels et directionnels — le tout dans un espace 3D définit par des coordonnées cartésiennes 3D.
Exemple:
var panner = context.createPanner(); panner.coneOuterGain = 0.2; panner.coneOuterAngle = 120; panner.coneInnerAngle = 0; panner.connect(context.destination); source.connect(panner); source.start(0); // Position the listener at the origin. context.listener.setPosition(0, 0, 0);
Note: Vous pouvez trouver un exemple sur notre repo GitHub (le voir en direct aussi).
Note: Voir {{domxref("PannerNode")}} pour plus d'informations.
Il est possible de manipuler l'audio au bas niveau en utilisant JavaScript. Cela peut être utile si vous voulez créer des codecs audio.
Des bibliothèques existent actuellement pour les formats suivants:
Note: Sur AudioCogs, vous pouvez essayer quelques démos; Audiocogs fournit également un Framework, Aurora.js, qui est destiné à vous aider à créer vos propres codecs en JavaScript.