Spaces:
Runtime error
Runtime error
File size: 2,546 Bytes
3deaf02 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 |
<script lang="ts">
import { fade } from 'svelte/transition';
import { audioBlob, notesImage } from './stores';
let currentTime: number;
let duration: number;
let paused = true;
let container: HTMLElement;
let visualisation: HTMLImageElement;
$: imageWidth = visualisation && visualisation.clientWidth;
$: if (currentTime || duration || !paused) {
imageWidth = visualisation && visualisation.clientWidth;
}
const handleMove = (e: MouseEvent): void => {
if (!duration) {
return;
}
if (e.type !== 'touchmove' && !(e.buttons & 1)) {
return;
}
const { left, right } = container.getBoundingClientRect();
currentTime = (duration * (e.clientX - left)) / (right - left);
};
const handleKeydown = (e: KeyboardEvent): void => {
if (e.code === 'Space') {
paused = !paused;
}
};
</script>
<svelte:window on:keydown={handleKeydown} />
{#if $audioBlob}
<section transition:fade>
<div
class="container"
bind:this={container}
on:mousemove={handleMove}
on:touchmove|preventDefault={handleMove}
style:width={imageWidth ? imageWidth + 'px' : '512px'}
>
<img
class="visualisation"
src={$notesImage}
alt="MIDI notes of composition"
draggable="false"
bind:this={visualisation}
on:click={() => (paused = !paused)}
/>
<audio bind:currentTime bind:duration bind:paused src={$audioBlob} />
<div class="handle" style:transform="translateX({imageWidth * (currentTime / duration)}px)" />
{#if paused}
<img
class="play"
src="play.svg"
alt="Play button"
draggable="false"
transition:fade
on:click={() => (paused = !paused)}
/>
{/if}
</div>
</section>
{/if}
<style>
section {
border: 2px solid hsl(0 0% 80%);
border-radius: 0.375rem;
padding: 1rem;
}
.container {
position: relative;
padding: 1rem;
margin: auto;
}
.visualisation {
display: block;
margin: auto;
height: 100%;
aspect-ratio: 1 / 1;
cursor: pointer;
}
audio {
width: 100%;
margin: 1rem auto;
}
.play {
position: absolute;
left: 50%;
top: 50%;
width: 20%;
aspect-ratio: 1 / 1;
transform: translate(-50%, -60%);
filter: drop-shadow(0 0 5px black);
cursor: pointer;
}
.handle {
position: absolute;
left: 0;
top: -2.5%;
height: 105%;
width: 0.2rem;
margin-left: 1rem;
border-radius: 0.1rem;
background-color: white;
cursor: pointer;
}
@media (min-width: 600px) {
audio {
max-width: 40rem;
}
}
</style>
|