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>