Oggi vediamo un progetto interessante, una tastiera virtuale da pianoforte, che possiamo usare nel browser. Progetto tratto da codepen.io. Ho semplificato leggermente il codice (nel codice originale compaiono diversi stili, decisamente più carini, diciamo non indispensabili per i nostri scopi e in particolare per focalizzarci su un collegamento che vorrei approfondire in seguito).
Il funzionamento del programma è questo:
- creazione dei tasti bianchi e neri del pianoforte (ottava centrale)
- al clic dei tasti A,W,S,D... il suono corrispondente (DO, REb, RE, ecc)
- il concetto di base è ovviamente un
addEventListener()
in questo caso "keydown" (premo un tasto, A,S,W,D...) che fa partire in questo caso la funzione playNote()
, quindi riproduce il file musicale corrispondente a ciò che rappresenta il tasto cliccato
- per comodità il suono viene caricato tramite tag HTML
<audio>
, da file musicali precaricati online, ad esempio la nota DO ("C" nella notazione anglosassone) si trova all'indirizzo http://carolinegabriel.com/demo/js-keyboard/sounds/040.wav
Un'alternativa interessante sarebbe quella di svincolare il programma da risorse esterne, ovvero convertendo anche i file audio in Data64 e quindi scrivendone il relativo codice inline: analogo al caso includere immagine SVG all'interno del file HTML+CSS. Questo è possibile tramite il sito web base64.guru, "Audio to Base64". Poi si procede nel seguente modo: anziché ad esempio il codice (per la nota DO) <audio data-key="65" src="http://carolinegabriel.com/demo/js-keyboard/sounds/040.wav"></audio>
, al posto di src=" "
dobbiamo mettere src="data:audio/mpeg;base64, [...] "
. Il termine [...]
è la conversione in Base64 del file audio, è generalmente molto pesante; vediamo ad esempio sempre per il caso della nota DO, cosa si ottiene:
UklGRtzxAABXQVZFZm10IBAAAAABAAIAESsAAESsAAAEABAAZGF0YbjxAAAAAAAARf8z/37/XP9n/zb/J/3o/MP+dP72BZoFRw7qDdcPhA/pDqIO1gmjCfoB0gEXBPkDxQe0B3IGbAYxDUANvQjcCPkELQXkCikL2w0pDl8PsA+2DgkP
Ho riportato SOLO una piccolissima parte del codice, in quanto sarebbe lungo in totale diverse migliaia di caratteri!!
Dunque questa è la procedura, riportare il codice completo con la conversione di tutti i file audio, per tutti i tasti, produrrebbe un codice troppo pesante da scrivere qui. Pertanto riporto la versione di base del codice (semplificata senza alcune decorazioni grafiche), che è la seguente (unico file HTML che contiene CSS e JavaScript in blocco):
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<style>
.nowplaying {
font-size: 120px;
line-height: 1;
color: #eee;
text-shadow: 0 0 5rem #028ae9;
transition: all .07s ease;
min-height: 120px;
}
.keys {
display: block;
width: 100%;
height: 350px;
max-width: 880px;
position: relative;
margin: 40px auto 0;
cursor: none;
}
.key {
position: relative;
border: 4px solid black;
border-radius: .5rem;
transition: all .07s ease;
display: block;
box-sizing: border-box;
z-index: 2;
}
.key:not(.sharp) {
float: left;
width: 10%;
height: 100%;
background: rgba(255, 255, 255, .8);
}
.key.sharp {
position: absolute;
width: 6%;
height: 60%;
background: #000;
color: #eee;
top: 0;
z-index: 3;
}
.key[data-key="87"] {
left: 7%;
}
.key[data-key="69"] {
left: 17%;
}
.key[data-key="84"] {
left: 37%;
}
.key[data-key="89"] {
left: 47%;
}
.key[data-key="85"] {
left: 57%;
}
.key[data-key="79"] {
left: 77%;
}
.key[data-key="80"] {
left: 87%;
}
.playing {
transform: scale(.95);
border-color: #028ae9;
box-shadow: 0 0 1rem #028ae9;
}
.hints {
display: block;
width: 100%;
opacity: 0;
position: absolute;
bottom: 7px;
transition: opacity .3s ease-out;
font-size: 20px;
}
.keys:hover .hints {
opacity: 1;
}
</style>
</head>
<body>
<div class="nowplaying"></div>
<div class="keys">
<div data-key="65" class="key" data-note="C">
<span class="hints">A</span>
</div>
<div data-key="87" class="key sharp" data-note="C#">
<span class="hints">W</span>
</div>
<div data-key="83" class="key" data-note="D">
<span class="hints">S</span>
</div>
<div data-key="69" class="key sharp" data-note="D#">
<span class="hints">E</span>
</div>
<div data-key="68" class="key" data-note="E">
<span class="hints">D</span>
</div>
<div data-key="70" class="key" data-note="F">
<span class="hints">F</span>
</div>
<div data-key="84" class="key sharp" data-note="F#">
<span class="hints">T</span>
</div>
<div data-key="71" class="key" data-note="G">
<span class="hints">G</span>
</div>
<div data-key="89" class="key sharp" data-note="G#">
<span class="hints">Y</span>
</div>
<div data-key="72" class="key" data-note="A">
<span class="hints">H</span>
</div>
<div data-key="85" class="key sharp" data-note="A#">
<span class="hints">U</span>
</div>
<div data-key="74" class="key" data-note="B">
<span class="hints">J</span>
</div>
<div data-key="75" class="key" data-note="C">
<span class="hints">K</span>
</div>
<div data-key="79" class="key sharp" data-note="C#">
<span class="hints">O</span>
</div>
<div data-key="76" class="key" data-note="D">
<span class="hints">L</span>
</div>
<div data-key="80" class="key sharp" data-note="D#">
<span class="hints">P</span>
</div>
<div data-key="186" class="key" data-note="E">
<span class="hints">;</span>
</div>
</div>
<audio data-key="65" src="http://carolinegabriel.com/demo/js-keyboard/sounds/040.wav"></audio>
<audio data-key="87" src="http://carolinegabriel.com/demo/js-keyboard/sounds/041.wav"></audio>
<audio data-key="83" src="http://carolinegabriel.com/demo/js-keyboard/sounds/042.wav"></audio>
<audio data-key="69" src="http://carolinegabriel.com/demo/js-keyboard/sounds/043.wav"></audio>
<audio data-key="68" src="http://carolinegabriel.com/demo/js-keyboard/sounds/044.wav"></audio>
<audio data-key="70" src="http://carolinegabriel.com/demo/js-keyboard/sounds/045.wav"></audio>
<audio data-key="84" src="http://carolinegabriel.com/demo/js-keyboard/sounds/046.wav"></audio>
<audio data-key="71" src="http://carolinegabriel.com/demo/js-keyboard/sounds/047.wav"></audio>
<audio data-key="89" src="http://carolinegabriel.com/demo/js-keyboard/sounds/048.wav"></audio>
<audio data-key="72" src="http://carolinegabriel.com/demo/js-keyboard/sounds/049.wav"></audio>
<audio data-key="85" src="http://carolinegabriel.com/demo/js-keyboard/sounds/050.wav"></audio>
<audio data-key="74" src="http://carolinegabriel.com/demo/js-keyboard/sounds/051.wav"></audio>
<audio data-key="75" src="http://carolinegabriel.com/demo/js-keyboard/sounds/052.wav"></audio>
<audio data-key="79" src="http://carolinegabriel.com/demo/js-keyboard/sounds/053.wav"></audio>
<audio data-key="76" src="http://carolinegabriel.com/demo/js-keyboard/sounds/054.wav"></audio>
<audio data-key="80" src="http://carolinegabriel.com/demo/js-keyboard/sounds/055.wav"></audio>
<audio data-key="186" src="http://carolinegabriel.com/demo/js-keyboard/sounds/056.wav"></audio>
<script>
const keys = document.querySelectorAll(".key"),
note = document.querySelector(".nowplaying"),
hints = document.querySelectorAll(".hints");
function playNote(e) {
const audio = document.querySelector(`audio[data-key="${e.keyCode}"]`),
key = document.querySelector(`.key[data-key="${e.keyCode}"]`);
if (!key) return;
const keyNote = key.getAttribute("data-note");
key.classList.add("playing");
note.innerHTML = keyNote;
audio.currentTime = 0;
audio.play();
}
function removeTransition(e) {
if (e.propertyName !== "transform") return;
this.classList.remove("playing");
}
function hintsOn(e, index) {
e.setAttribute("style", "transition-delay:" + index * 50 + "ms");
}
hints.forEach(hintsOn);
keys.forEach(key => key.addEventListener("transitionend", removeTransition));
window.addEventListener("keydown", playNote);
</script>
</body>
</html>
Nota aggiuntiva: a parte questo esempio, funzionante ma non troppo complesso, può essere interessante anche una personalizzazione ad-hoc, in stile DJ. Volendo, è infatti possibile aggiungere nuovi tasti con suoni specifici prestabiliti che possono essere combinazioni di note, particolari "effetti", ecc.
Vediamo infine un'immagine di come appare: