Browse Source

Haphazard improvements to editor

Thomas Dy 7 years ago
parent
commit
bcec1b2429
4 changed files with 184 additions and 44 deletions
  1. 23 0
      dist/editor.css
  2. 16 10
      dist/editor.html
  3. 9 9
      src/audio.ts
  4. 136 25
      src/editor.ts

+ 23 - 0
dist/editor.css

@@ -1,3 +1,20 @@
+#controls {
+  position: fixed;
+  width: 90%;
+  background-color: white;
+}
+
+.waveform-container {
+  position: relative;
+  height: 60px;
+}
+
+#waveform, #waveform-overlay {
+  position: absolute;
+  height: 100%;
+  width: 100%;
+}
+
 .scrubber {
   display: grid;
   grid-template-rows: 10px 20px;
@@ -13,6 +30,10 @@
   height: 100%;
 }
 
+li.highlight {
+  background-color: yellow;
+}
+
 .markers {
   grid-row: 2 / 3;
   position: relative;
@@ -28,6 +49,7 @@
 }
 
 .text-areas {
+  padding-top: 160px;
   display: grid;
   grid-template-columns: 33% 33% auto;
   grid-template-rows: 20px auto;
@@ -68,6 +90,7 @@
   min-height: 200px;
   white-space: pre;
   overflow-x: auto;
+  font-size: 16px;
 }
 
 #json {

+ 16 - 10
dist/editor.html

@@ -4,18 +4,24 @@
     <link rel="stylesheet" href="editor.css" />
   </head>
   <body>
-    <div id="#container">
-      <input id="audio" type="file" />
-
-      <button id="play">Play</button>
-      <button id="pause">Pause</button>
-      <button id="insert-marker">Insert Marker</button>
-      <div class="scrubber">
-        <div class="bar">
-          <div class="bar-overlay"></div>
+    <div id="container">
+      <div id="controls">
+        <input id="audio" type="file" />
+        <button id="play">Play</button>
+        <button id="pause">Pause</button>
+        <button id="insert-marker">Insert Marker</button>
+        <div class="scrubber">
+          <div class="bar">
+            <div class="bar-overlay"></div>
+          </div>
+          <div class="markers">
+          </div>
         </div>
-        <div class="markers">
+        <div class="waveform-container">
+          <canvas id="waveform"></canvas>
+          <canvas id="waveform-overlay"></canvas>
         </div>
+        <div id="display"></div>
       </div>
 
       <div class="text-areas">

+ 9 - 9
src/audio.ts

@@ -64,7 +64,7 @@ namespace audio {
       this.playStartTime = 0;
       this.resumeTime = 0;
       this.hasStarted = false;
-      this.isFinished = false;
+      this.isFinished = true;
     }
 
     play(): void {
@@ -84,11 +84,9 @@ namespace audio {
       this.source.onended = (event) => {
         if (this.source == event.target) {
           this.isFinished = true;
-          if (duration > 0) {
-            this.resumeTime += duration;
-            if (this.resumeTime > this.getDuration()) {
-              this.resumeTime = 0;
-            }
+          this.resumeTime = this.manager.getTime() - this.playStartTime;
+          if (this.resumeTime > this.getDuration()) {
+            this.resumeTime = 0;
           }
         }
       }
@@ -99,6 +97,7 @@ namespace audio {
     }
 
     pause(): void {
+      if (this.isFinished) return;
       this.resumeTime = this.manager.getTime() - this.playStartTime;
       this.isFinished = true;
       if (this.source) {
@@ -119,14 +118,15 @@ namespace audio {
     }
 
     getTime(): number {
-      if (this.isFinished) {
+      if (!this.hasStarted) {
+        return 0;
+      }
+      else if (this.isFinished) {
         if (this.resumeTime > 0) {
           return this.resumeTime;
         } else {
           return this.getDuration();
         }
-      } else if (!this.hasStarted) {
-        return 0;
       } else {
         return this.manager.getTime() - this.playStartTime;
       }

+ 136 - 25
src/editor.ts

@@ -9,12 +9,15 @@ namespace editor {
     barElement: HTMLElement;
     markerListElement: HTMLElement;
     intervalListElement: HTMLElement;
-    kanaElement: HTMLElement;
-    kanjiElement: HTMLElement;
+    kanaElement: HTMLTextAreaElement;
+    kanjiElement: HTMLTextAreaElement;
+    displayElement: HTMLElement;
     jsonElement: HTMLInputElement;
     track: audio.Track | null = null;
     markers: Marker[] = [];
     rafId: number;
+    waveForm: WaveForm;
+    currentMarker: Marker;
 
     constructor() {
       this.audioManager = new audio.AudioManager();
@@ -27,7 +30,13 @@ namespace editor {
       this.intervalListElement = document.querySelector('#intervals');
       this.kanaElement = document.querySelector('#kana');
       this.kanjiElement = document.querySelector('#kanji');
+      this.displayElement = document.querySelector('#display');
       this.jsonElement = document.querySelector('#json');
+      this.waveForm = new WaveForm(
+        document.querySelector('#waveform'),
+        document.querySelector('#waveform-overlay'),
+        (time: number) => this.play(time)
+      );
 
       this.markerListElement.addEventListener('click', (event: MouseEvent) => this.markersClick(event));
       document.querySelector('#play').addEventListener('click', () => this.play());
@@ -36,6 +45,8 @@ namespace editor {
       document.querySelector('.bar').addEventListener('click', (event: MouseEvent) => this.scrubberClick(event));
       document.querySelector('#import').addEventListener('click', () => this.import());
       document.querySelector('#export').addEventListener('click', () => this.export());
+
+      this.update();
     }
 
     loadAudio(): void {
@@ -45,29 +56,43 @@ namespace editor {
           this.track.stop();
         }
         this.clearMarkers();
-        this.audioManager.loadTrackFromFile(file).then(t => this.track = t);
+        this.audioManager.loadTrackFromFile(file).then(t => {
+          this.track = t;
+          this.waveForm.setTrack(t);
+        });
       }
     }
 
     update(): void {
-      let percentage = this.track.getTime() / this.track.getDuration() * 100;
-      this.barElement.style.width = `${percentage}%`;
-      if (this.track.isPlaying()) {
-        this.rafId = requestAnimationFrame(() => this.update());
+      if (this.track != null) {
+        let percentage = this.track.getTime() / this.track.getDuration() * 100;
+        this.barElement.style.width = `${percentage}%`;
+        this.waveForm.update(this.markers);
+        if (this.currentMarker) {
+          this.currentMarker.liElement.className = '';
+        }
+        let index = this.markers.findIndex(m => m.time > this.track.getTime());
+        if (index < 0) index = 0;
+        this.currentMarker = this.markers[index - 1];
+        if (this.currentMarker) {
+          this.currentMarker.liElement.className = 'highlight';
+        }
+        let text = this.kanjiElement.value.split('\n')[index] || '';
+        this.displayElement.textContent = text;
       }
+      requestAnimationFrame(() => this.update());
     }
 
     scrubberClick(event: MouseEvent): void {
-      let pos = event.clientX - this.markerListElement.offsetLeft;
+      let pos = event.clientX - 10;
+      console.log(pos);
       let percentage = pos / this.markerListElement.clientWidth;
       let targetTime = percentage * this.track.getDuration();
-      this.track.stop();
-      this.track.resumeTime = targetTime;
-      this.play();
+      this.play(targetTime);
     }
 
     markersClick(event: MouseEvent): void {
-      let pos = event.clientX - this.markerListElement.offsetLeft;
+      let pos = event.clientX - 10;
       let percentage = pos / this.markerListElement.clientWidth;
       let targetTime = percentage * this.track.getDuration();
       this.insertMarker(targetTime);
@@ -98,10 +123,12 @@ namespace editor {
       }
     }
 
-    play(duration: number = undefined): void {
-      window.cancelAnimationFrame(this.rafId);
+    play(start: number = undefined, duration: number = undefined): void {
+      this.track.pause();
+      if (start != undefined) {
+        this.track.resumeTime = start;
+      }
       this.track.start(duration);
-      this.update();
     }
 
     pause(): void {
@@ -109,15 +136,16 @@ namespace editor {
     }
 
     playMarker(marker: Marker): void {
-      let start = 0;
+      let start = marker.time;
+      let end = this.track.getDuration();
       let index = this.markers.findIndex(m => m == marker);
-      if (index > 0) {
-        start = this.markers[index - 1].time;
+      if (index < this.markers.length - 1) {
+        end = this.markers[index + 1].time;
       }
-      let duration = marker.time - start;
-      this.track.stop();
-      this.track.resumeTime = start;
-      this.play(duration);
+      let duration = end - start;
+      this.play(start, duration);
+
+      this.highlightLine(this.kanjiElement, index + 1);
     }
 
     removeMarker(marker: Marker): void {
@@ -135,6 +163,17 @@ namespace editor {
       this.markers = [];
     }
 
+    highlightLine(element: HTMLTextAreaElement, line: number) {
+      let text = element.value;
+      let index = 0;
+      for (let i = 0; i < line; ++i) {
+        index = text.indexOf('\n', index + 1);
+      }
+      let endIndex = text.indexOf('\n', index + 1);
+      element.focus();
+      element.setSelectionRange(index, endIndex);
+    }
+
     import(): void {
       this.clearMarkers();
       let lines: level.Line[] = JSON.parse(this.jsonElement.value);
@@ -156,9 +195,7 @@ namespace editor {
     export(): void {
       let kanji = this.kanjiElement.value.split('\n');
       let kana = this.kanaElement.value.split('\n');
-      kanji.pop();
-      kana.pop();
-      let length = Math.max(kanji.length, kana.length, this.markers.length);
+      let length = Math.max(kanji.length, kana.length, this.markers.length - 1);
 
       let lines = [];
       let lastStart = 0;
@@ -217,4 +254,78 @@ namespace editor {
       this.markerElement.style.left = `${percentage}%`;
     }
   }
+
+  class WaveForm {
+    ctx: CanvasRenderingContext2D;
+    overlayCtx: CanvasRenderingContext2D;
+    track: audio.Track | null;
+    data: Float32Array | null;
+    stride: number;
+    currentSection: number;
+
+    constructor(
+      readonly canvas: HTMLCanvasElement,
+      readonly overlay: HTMLCanvasElement,
+      readonly setTime: (time: number) => void
+    ) {
+      canvas.height = canvas.clientHeight;
+      canvas.width = canvas.clientWidth;
+      overlay.height = overlay.clientHeight;
+      overlay.width = overlay.clientWidth;
+      this.ctx = canvas.getContext('2d');
+      this.overlayCtx = overlay.getContext('2d');
+
+      this.overlayCtx.fillStyle = 'rgba(255, 0, 0, 0.5)';
+
+      this.overlay.addEventListener('click', (event: MouseEvent) => {
+        let pos = event.clientX - this.overlay.offsetLeft;
+        let percentage = pos / this.overlay.width;
+        let time = this.currentSection * 5 + percentage * 5;
+        this.setTime(time);
+      });
+    }
+
+    setTrack(track: audio.Track): void {
+      this.track = track;
+      this.stride = Math.floor(this.track.buffer.sampleRate / this.canvas.width * 5);
+      this.currentSection = -1;
+    }
+
+    timeToX(time: number): number {
+      return (time - this.currentSection * 5) / 5 * this.canvas.width
+    }
+
+    update(markers: Marker[]): void {
+      let section = Math.floor(this.track.getTime() / 5);
+
+      let height = this.canvas.height;
+      if (this.currentSection != section) {
+        this.data = this.track.buffer.getChannelData(0);
+        this.ctx.clearRect(0, 0, this.canvas.width, height);
+        this.ctx.beginPath();
+        this.ctx.moveTo(0, height / 2);
+        let offset = section * this.canvas.width * this.stride;
+        for (let i = 0; i < this.canvas.width; ++i) {
+          let index = offset + i * this.stride;
+          let value = height / 2 + height / 2 * this.data[index];
+          this.ctx.lineTo(i, value);
+        }
+        this.ctx.stroke();
+        this.currentSection = section;
+      }
+
+      let marker = this.timeToX(this.track.getTime());
+      this.overlayCtx.clearRect(0, 0, this.canvas.width, height);
+      this.overlayCtx.fillRect(0, 0, marker, height);
+      markers.forEach(m => {
+        if (m.time > section * 5 && m.time <= section * 5 + 5) {
+          let x = this.timeToX(m.time);
+          this.overlayCtx.beginPath();
+          this.overlayCtx.moveTo(x, 0);
+          this.overlayCtx.lineTo(x, height);
+          this.overlayCtx.stroke();
+        }
+      })
+    }
+  }
 }