Browse Source

Implement audio

Thomas Dy 7 years ago
parent
commit
f4252fead7
5 changed files with 269 additions and 17 deletions
  1. 31 1
      dist/style.css
  2. 76 0
      src/audio.ts
  3. 157 14
      src/display.ts
  4. 3 1
      src/index.ts
  5. 2 1
      tsconfig.json

+ 31 - 1
dist/style.css

@@ -1,7 +1,6 @@
 .kana {
   display: inline-block;
   position: relative;
-  color: black;
 }
 
 .kana::after {
@@ -46,3 +45,34 @@
     transform: scale(1, 1)
   }
 }
+
+.progress-bar {
+  position: relative;
+  height: 5px;
+  background-color: lightgrey;
+}
+
+.progress-bar .shade {
+  position: absolute;
+  height: 5px;
+  background-color: red;
+  animation-timing-function: linear;
+  animation-play-state: paused;
+}
+
+@keyframes progress {
+  from {
+    width: 0%;
+  }
+  to {
+    width: 100%;
+  }
+}
+
+.level-control.waiting {
+  color: grey;
+}
+
+.level-control.waiting .romaji {
+  color: transparent;
+}

+ 76 - 0
src/audio.ts

@@ -0,0 +1,76 @@
+namespace audio {
+  export class AudioManager {
+    context: AudioContext;
+    volume: GainNode;
+    output: AudioNode;
+
+    constructor() {
+      this.context = new AudioContext();
+      this.volume = this.context.createGain();
+      this.volume.connect(this.context.destination);
+      this.output = this.volume;
+    }
+
+    getTime(): number {
+      return this.context.currentTime;
+    }
+
+    loadTrack(url: string): Promise<Track> {
+      return window.fetch(url)
+        .then(response => response.arrayBuffer())
+        .then(buffer => this.context.decodeAudioData(buffer))
+        .then(audioBuffer => new Track(this, audioBuffer))
+    }
+  }
+
+  export class Track {
+    manager: AudioManager;
+    buffer: AudioBuffer;
+    source: AudioBufferSourceNode | null;
+    playStartTime: number;
+    hasStarted: boolean;
+    isFinished: boolean;
+
+    constructor(manager: AudioManager, buffer: AudioBuffer) {
+      this.manager = manager;
+      this.buffer = buffer;
+      this.playStartTime = 0;
+      this.hasStarted = false;
+      this.isFinished = false;
+    }
+
+    play(): void {
+      this.source = this.manager.context.createBufferSource();
+      this.source.buffer = this.buffer;
+      this.source.connect(this.manager.output);
+      this.source.onended = () => {
+        this.isFinished = true;
+      }
+      this.isFinished = false;
+      this.hasStarted = true;
+      this.playStartTime = this.manager.getTime();
+      this.source.start();
+    }
+
+    stop(): void {
+      this.isFinished = true;
+      if (this.source) {
+        this.source.stop();
+      }
+    }
+
+    getTime(): number {
+      if (this.isFinished) {
+        return this.getDuration();
+      } else if (!this.hasStarted) {
+        return 0;
+      } else {
+        return this.manager.getTime() - this.playStartTime;
+      }
+    }
+
+    getDuration(): number {
+      return this.buffer.duration;
+    }
+  }
+}

+ 157 - 14
src/display.ts

@@ -7,6 +7,7 @@
 
 /// <reference path="kana.ts" />
 /// <reference path="state.ts" />
+/// <reference path="audio.ts" />
 
 namespace display {
   import InputState = kana.KanaInputState;
@@ -181,42 +182,124 @@ namespace display {
     }
   }
 
+  enum LevelState {
+    LOADING,
+    READY,
+    PLAYING,
+    WAITING,
+    FINISH
+  }
+
   export class LevelController implements Component {
     element: HTMLElement;
     level: level.Level;
-    currentLine: number;
+    currentIndex: number;
     inputState: InputState | null;
     mainAreaController: MainAreaController;
+    progressController: TrackProgressController | null;
     listener: (event: KeyboardEvent) => void;
+    state: LevelState;
+    track: audio.Track | null;
 
-    constructor(level: level.Level) {
+    constructor(audioManager: audio.AudioManager, level: level.Level) {
       this.element = document.createElement('div');
       this.level = level;
-      this.currentLine = -1;
+      this.currentIndex = -1;
       this.inputState = null;
       this.mainAreaController = new MainAreaController();
+      this.progressController = null;
       this.listener = event => this.handleInput(event.key);
+      this.state = LevelState.LOADING;
+      this.track = null;
+
+      this.element.className = 'level-control';
+      this.element.appendChild(this.mainAreaController.element);
+      document.addEventListener('keydown', this.listener);
 
+      if (this.level.audio == null) {
+        this.level.lines = this.level.lines.filter(line => line.kana != "@");
+        this.onReady();
+      } else {
+        this.progressController = new TrackProgressController(this.level);
+        this.element.insertBefore(
+          this.progressController.element,
+          this.mainAreaController.element
+        );
+        this.progressController.setListener(event => this.onIntervalEnd());
+        audioManager.loadTrack(this.level.audio).then(track => {
+          this.track = track;
+          this.onReady();
+        })
+      }
+
+    }
+
+    onReady(): void {
+      this.setState(LevelState.READY);
+    }
+
+    onStart(): void {
       this.nextLine();
+      if (this.track !== null) {
+        this.progressController.start();
+        this.track.play();
+      }
 
-      document.addEventListener('keydown', this.listener);
-      this.element.appendChild(this.mainAreaController.element);
+      this.setState(LevelState.PLAYING);
+      this.checkComplete();
     }
 
-    handleInput(key: string): void {
-      if (this.inputState !== null) {
-        if (this.inputState.handleInput(key)) {
-          this.nextLine();
-        }
-      } else {
+    checkComplete(): void {
+      let currentLine = this.level.lines[this.currentIndex];
+      if (currentLine.kana == '@' && currentLine.kanji == '@') {
+        this.onComplete();
+      }
+    }
+
+    onIntervalEnd(): void {
+      if (this.state === LevelState.WAITING) {
+        this.setState(LevelState.PLAYING);
+      } else if (this.state === LevelState.PLAYING) {
         this.nextLine();
       }
+      this.checkComplete();
+    }
+
+    onComplete(): void {
+      this.nextLine();
+      if (this.track !== null) {
+        this.setState(LevelState.WAITING);
+      }
+    }
+
+    setState(state: LevelState): void {
+      if (state === LevelState.WAITING) {
+        this.element.classList.add('waiting');
+      } else {
+        this.element.classList.remove('waiting');
+      }
+      this.state = state;
+    }
+
+    handleInput(key: string): void {
+      switch (this.state) {
+        case LevelState.READY:
+          this.onStart();
+          break;
+        case LevelState.PLAYING:
+          if (this.inputState !== null) {
+            if (this.inputState.handleInput(key)) {
+              this.onComplete();
+            }
+          }
+          break;
+      }
     }
 
     nextLine(): void {
-      if (this.currentLine + 1 < this.level.lines.length) {
-        this.currentLine += 1;
-        this.setLine(this.level.lines[this.currentLine]);
+      if (this.currentIndex + 1 < this.level.lines.length) {
+        this.currentIndex += 1;
+        this.setLine(this.level.lines[this.currentIndex]);
       } else {
         this.setLine({ kanji: '@', kana: '@' });
       }
@@ -244,4 +327,64 @@ namespace display {
       document.removeEventListener('keydown', this.listener);
     }
   }
+
+  class ProgressBar implements Component {
+    element: HTMLElement;
+    barElement: HTMLElement;
+
+    constructor() {
+      this.element = document.createElement('div');
+      this.element.className = 'progress-bar';
+      this.barElement = document.createElement('div');
+      this.barElement.className = 'shade';
+      this.element.appendChild(this.barElement);
+    }
+
+    get style() {
+      return this.barElement.style;
+    }
+
+    destroy(): void {}
+  }
+
+  class TrackProgressController implements Component {
+    element: HTMLElement;
+    totalBar: ProgressBar;
+    intervalBar: ProgressBar;
+
+    constructor(level: level.Level) {
+      this.element = document.createElement('div');
+      this.totalBar = new ProgressBar();
+      this.intervalBar = new ProgressBar();
+      this.element.appendChild(this.totalBar.element);
+      this.element.appendChild(this.intervalBar.element);
+
+      let lines = level.lines;
+
+      let totalDuration = lines[lines.length - 1].end;
+      this.totalBar.style.animationName = 'progress';
+      this.totalBar.style.animationDuration = totalDuration + 's';
+
+      let names = lines.map(line => 'progress').join(',');
+      let delays = lines.map(line => line.start + 's').join(',');
+      let durations = lines.map(line => (line.end - line.start) + 's').join(',');
+      this.intervalBar.style.animationName = names;
+      this.intervalBar.style.animationDelay = delays;
+      this.intervalBar.style.animationDuration = durations;
+    }
+
+    start(): void {
+      this.intervalBar.style.width = '100%';
+      this.totalBar.style.width = '100%';
+
+      this.intervalBar.style.animationPlayState = 'running';
+      this.totalBar.style.animationPlayState = 'running';
+    }
+
+    setListener(func: (event: AnimationEvent) => void): void {
+      this.intervalBar.element.addEventListener('animationend', func);
+    }
+
+    destroy(): void {}
+  }
 }

+ 3 - 1
src/index.ts

@@ -1,9 +1,11 @@
 /// <reference path="display.ts" />
 /// <reference path="level.ts" />
+/// <reference path="audio.ts" />
 
+let audioManager = new audio.AudioManager();
 let container = document.querySelector('#container');
 
 level.loadFromJson('levels.json').then(levelsets => {
-  let controller = new display.LevelController(levelsets[0].levels[0]);
+  let controller = new display.LevelController(audioManager, levelsets[0].levels[0]);
   container.appendChild(controller.element);
 });

+ 2 - 1
tsconfig.json

@@ -3,7 +3,8 @@
     "noImplicitAny": true,
     "removeComments": true,
     "sourceMap": true,
-    "outFile": "dist/bundle.js"
+    "outFile": "dist/bundle.js",
+    "target": "es5"
   },
   "include": [
     "src/**/*"