123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291 |
- /*
- * This module handles displaying the UI. The most important one is the main
- * area which contains the text to be typed in. Progress is displayed on the
- * kana part of the area, while errors are shown via the romaji section. The
- * kanji is simply just for reading.
- */
- /// <reference path="kana.ts" />
- /// <reference path="state.ts" />
- /// <reference path="util.ts" />
- namespace display {
- import InputState = kana.KanaInputState;
- import TransitionResult = state.TransitionResult;
- interface Component {
- element: HTMLElement;
- destroy(): void;
- }
- class KanaDisplayComponent implements Component {
- element: HTMLElement;
- state: state.StateMachine;
- observer: state.Observer;
- remove: () => void;
- constructor(kana: string, state: state.StateMachine) {
- this.state = state;
- this.observer = result => this.rerender(result);
- this.state.addObserver(this.observer);
- this.element = document.createElement('span');
- this.element.classList.add('kana');
- this.element.textContent = kana;
- this.element.setAttribute('data-text', kana);
- }
- rerender(result: TransitionResult): void {
- if (result != TransitionResult.FAILED) {
- if (this.state.isFinished()) {
- this.element.classList.remove('half');
- this.element.classList.add('full');
- } else {
- this.element.classList.add('half');
- }
- }
- }
- destroy(): void {
- this.state.removeObserver(this.observer);
- }
- }
- export class KanaDisplayController implements Component {
- children: KanaDisplayComponent[];
- constructor(readonly element: HTMLElement) {
- this.children = [];
- }
- setInputState(inputState: InputState) {
- this.clearChildren();
- if (inputState == null) {
- this.children = [];
- } else {
- this.children = inputState.map((kana, machine) => {
- return new KanaDisplayComponent(kana, machine);
- });
- this.children.forEach(child => this.element.appendChild(child.element));
- }
- }
- private clearChildren(): void {
- this.children.forEach(child => {
- this.element.removeChild(child.element);
- child.destroy();
- });
- }
- destroy(): void {
- this.clearChildren();
- }
- }
- export class RomajiDisplayController {
- observer: state.Observer;
- inputState: InputState | null;
- constructor(
- readonly firstElement: HTMLElement,
- readonly restElement: HTMLElement
- ) {
- this.observer = (result) => this.rerender(result);
- }
- setInputState(inputState: InputState) {
- this.clearObservers();
- this.inputState = inputState;
- if (this.inputState != null) {
- this.inputState.map((_, machine) => {
- machine.addObserver(this.observer);
- });
- this.rerender(TransitionResult.SUCCESS);
- } else {
- this.firstElement.textContent = '';
- this.restElement.textContent = '';
- }
- }
- private clearObservers(): void {
- if (this.inputState != null) {
- this.inputState.map((_, machine) => {
- machine.removeObserver(this.observer);
- });
- }
- }
- rerender(result: TransitionResult): void {
- if (result === TransitionResult.FAILED) {
- this.firstElement.classList.remove('error');
- this.firstElement.offsetHeight; // trigger reflow
- this.firstElement.classList.add('error');
- } else {
- let remaining = this.inputState.getRemainingInput();
- this.firstElement.textContent = remaining.charAt(0);
- this.restElement.textContent = remaining.substring(1);
- }
- }
- destroy(): void {
- this.clearObservers();
- }
- }
- export class TrackProgressController {
- totalBar: HTMLElement;
- intervalBar: HTMLElement;
- listener: (event: AnimationEvent) => void;
- constructor(private element: HTMLElement, lines: level.Line[]) {
- this.totalBar = element.querySelector('.total .shade');
- this.intervalBar = element.querySelector('.interval .shade');
- 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 {
- if (this.listener) {
- this.intervalBar.removeEventListener('animationend', func);
- }
- this.intervalBar.addEventListener('animationend', func);
- this.listener = func;
- }
- destroy(): void {
- if (this.listener) {
- this.intervalBar.removeEventListener('animationend', this.listener);
- }
- this.intervalBar.style.animationName = '';
- this.totalBar.style.animationName = '';
- }
- }
- export class Score {
- combo: number = 0;
- score: number = 0;
- maxCombo: number = 0;
- finished: number = 0;
- hit: number = 0;
- missed: number = 0;
- skipped: number = 0;
- intervalEnd(finished: boolean): void {
- if (finished) {
- this.finished += 1;
- } else {
- this.combo = 0;
- }
- }
- update(result: TransitionResult): void {
- switch (result) {
- case TransitionResult.SUCCESS:
- this.hit += 1;
- this.score += 100 + this.combo;
- this.combo += 1;
- break;
- case TransitionResult.FAILED:
- this.missed += 1;
- this.combo = 0;
- break;
- case TransitionResult.SKIPPED:
- this.skipped += 1;
- this.combo = 0;
- break;
- }
- if (this.combo > this.maxCombo) {
- this.maxCombo = this.combo;
- }
- }
- }
- export class ScoreController {
- comboElement: HTMLElement;
- scoreElement: HTMLElement;
- maxComboElement: HTMLElement;
- finishedElement: HTMLElement;
- hitElement: HTMLElement;
- missedElement: HTMLElement;
- skippedElement: HTMLElement;
- inputState: InputState | null;
- observer: state.Observer;
- score: Score;
- constructor(
- private scoreContainer: HTMLElement,
- private statsContainer: HTMLElement
- ) {
- this.comboElement = scoreContainer.querySelector('.combo');
- this.scoreElement = scoreContainer.querySelector('.score');
- this.maxComboElement = scoreContainer.querySelector('.max-combo');
- this.finishedElement = scoreContainer.querySelector('.finished');
- this.hitElement = statsContainer.querySelector('.hit');
- this.missedElement = statsContainer.querySelector('.missed');
- this.skippedElement = statsContainer.querySelector('.skipped');
- this.observer = result => this.update(result);
- this.score = new Score();
- this.setValues();
- }
- setInputState(inputState: InputState): void {
- this.clearObservers();
- this.inputState = inputState;
- if (this.inputState != null) {
- this.inputState.map((_, m) => {
- m.addObserver(this.observer);
- });
- }
- }
- intervalEnd(finished: boolean): void {
- this.score.intervalEnd(finished);
- this.setValues();
- }
- update(result: TransitionResult): void {
- this.score.update(result);
- this.setValues();
- }
- setValues(): void {
- this.comboElement.textContent = this.score.combo == 0 ? '' : this.score.combo+' combo';
- this.scoreElement.textContent = this.score.score+'';
- this.maxComboElement.textContent = this.score.maxCombo+'';
- this.finishedElement.textContent = this.score.finished+'';
- this.hitElement.textContent = this.score.hit+'';
- this.missedElement.textContent = this.score.missed+'';
- this.skippedElement.textContent = this.score.skipped+'';
- }
- private clearObservers(): void {
- if (this.inputState != null) {
- this.inputState.map((_, machine) => {
- machine.removeObserver(this.observer);
- });
- }
- }
- destroy(): void {
- this.clearObservers();
- }
- }
- }
|