display.ts 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. /*
  2. * This module handles displaying the UI. The most important one is the main
  3. * area which contains the text to be typed in. Progress is displayed on the
  4. * kana part of the area, while errors are shown via the romaji section. The
  5. * kanji is simply just for reading.
  6. */
  7. /// <reference path="kana.ts" />
  8. /// <reference path="state.ts" />
  9. /// <reference path="util.ts" />
  10. namespace display {
  11. import InputState = kana.KanaInputState;
  12. import TransitionResult = state.TransitionResult;
  13. interface Component {
  14. element: HTMLElement;
  15. destroy(): void;
  16. }
  17. class KanaDisplayComponent implements Component {
  18. element: HTMLElement;
  19. state: state.StateMachine;
  20. observer: state.Observer;
  21. remove: () => void;
  22. constructor(kana: string, state: state.StateMachine) {
  23. this.state = state;
  24. this.observer = result => this.rerender(result);
  25. this.state.addObserver(this.observer);
  26. this.element = document.createElement('span');
  27. this.element.classList.add('kana');
  28. this.element.textContent = kana;
  29. this.element.setAttribute('data-text', kana);
  30. }
  31. rerender(result: TransitionResult): void {
  32. if (result != TransitionResult.FAILED) {
  33. if (this.state.isFinished()) {
  34. this.element.classList.remove('half');
  35. this.element.classList.add('full');
  36. } else {
  37. this.element.classList.add('half');
  38. }
  39. }
  40. }
  41. destroy(): void {
  42. this.state.removeObserver(this.observer);
  43. }
  44. }
  45. export class KanaDisplayController implements Component {
  46. children: KanaDisplayComponent[];
  47. constructor(readonly element: HTMLElement) {
  48. this.children = [];
  49. }
  50. setInputState(inputState: InputState) {
  51. this.clearChildren();
  52. if (inputState == null) {
  53. this.children = [];
  54. } else {
  55. this.children = inputState.map((kana, machine) => {
  56. return new KanaDisplayComponent(kana, machine);
  57. });
  58. this.children.forEach(child => this.element.appendChild(child.element));
  59. }
  60. }
  61. private clearChildren(): void {
  62. this.children.forEach(child => {
  63. this.element.removeChild(child.element);
  64. child.destroy();
  65. });
  66. }
  67. destroy(): void {
  68. this.clearChildren();
  69. }
  70. }
  71. export class RomajiDisplayController {
  72. observer: state.Observer;
  73. inputState: InputState | null;
  74. constructor(
  75. readonly firstElement: HTMLElement,
  76. readonly restElement: HTMLElement
  77. ) {
  78. this.observer = (result) => this.rerender(result);
  79. }
  80. setInputState(inputState: InputState) {
  81. this.clearObservers();
  82. this.inputState = inputState;
  83. if (this.inputState != null) {
  84. this.inputState.map((_, machine) => {
  85. machine.addObserver(this.observer);
  86. });
  87. this.rerender(TransitionResult.SUCCESS);
  88. } else {
  89. this.firstElement.textContent = '';
  90. this.restElement.textContent = '';
  91. }
  92. }
  93. private clearObservers(): void {
  94. if (this.inputState != null) {
  95. this.inputState.map((_, machine) => {
  96. machine.removeObserver(this.observer);
  97. });
  98. }
  99. }
  100. rerender(result: TransitionResult): void {
  101. if (result === TransitionResult.FAILED) {
  102. this.firstElement.classList.remove('error');
  103. this.firstElement.offsetHeight; // trigger reflow
  104. this.firstElement.classList.add('error');
  105. } else {
  106. let remaining = this.inputState.getRemainingInput();
  107. this.firstElement.textContent = remaining.charAt(0);
  108. this.restElement.textContent = remaining.substring(1);
  109. }
  110. }
  111. destroy(): void {
  112. this.clearObservers();
  113. }
  114. }
  115. export class TrackProgressController {
  116. totalBar: HTMLElement;
  117. intervalBar: HTMLElement;
  118. listener: (event: AnimationEvent) => void;
  119. constructor(private element: HTMLElement, lines: level.Line[]) {
  120. this.totalBar = element.querySelector('.total .shade');
  121. this.intervalBar = element.querySelector('.interval .shade');
  122. let totalDuration = lines[lines.length - 1].end;
  123. this.totalBar.style.animationName = 'progress';
  124. this.totalBar.style.animationDuration = totalDuration + 's';
  125. let names = lines.map(line => 'progress').join(',');
  126. let delays = lines.map(line => line.start + 's').join(',');
  127. let durations = lines.map(line => (line.end - line.start) + 's').join(',');
  128. this.intervalBar.style.animationName = names;
  129. this.intervalBar.style.animationDelay = delays;
  130. this.intervalBar.style.animationDuration = durations;
  131. }
  132. start(): void {
  133. this.intervalBar.style.width = '100%';
  134. this.totalBar.style.width = '100%';
  135. this.intervalBar.style.animationPlayState = 'running';
  136. this.totalBar.style.animationPlayState = 'running';
  137. }
  138. setListener(func: (event: AnimationEvent) => void): void {
  139. if (this.listener) {
  140. this.intervalBar.removeEventListener('animationend', func);
  141. }
  142. this.intervalBar.addEventListener('animationend', func);
  143. this.listener = func;
  144. }
  145. destroy(): void {
  146. if (this.listener) {
  147. this.intervalBar.removeEventListener('animationend', this.listener);
  148. }
  149. this.intervalBar.style.animationName = '';
  150. this.totalBar.style.animationName = '';
  151. }
  152. }
  153. export class Score {
  154. combo: number = 0;
  155. score: number = 0;
  156. maxCombo: number = 0;
  157. finished: number = 0;
  158. hit: number = 0;
  159. missed: number = 0;
  160. skipped: number = 0;
  161. intervalEnd(finished: boolean): void {
  162. if (finished) {
  163. this.finished += 1;
  164. } else {
  165. this.combo = 0;
  166. }
  167. }
  168. update(result: TransitionResult): void {
  169. switch (result) {
  170. case TransitionResult.SUCCESS:
  171. this.hit += 1;
  172. this.score += 100 + this.combo;
  173. this.combo += 1;
  174. break;
  175. case TransitionResult.FAILED:
  176. this.missed += 1;
  177. this.combo = 0;
  178. break;
  179. case TransitionResult.SKIPPED:
  180. this.skipped += 1;
  181. this.combo = 0;
  182. break;
  183. }
  184. if (this.combo > this.maxCombo) {
  185. this.maxCombo = this.combo;
  186. }
  187. }
  188. }
  189. export class ScoreController {
  190. comboElement: HTMLElement;
  191. scoreElement: HTMLElement;
  192. maxComboElement: HTMLElement;
  193. finishedElement: HTMLElement;
  194. hitElement: HTMLElement;
  195. missedElement: HTMLElement;
  196. skippedElement: HTMLElement;
  197. inputState: InputState | null;
  198. observer: state.Observer;
  199. score: Score;
  200. constructor(
  201. private scoreContainer: HTMLElement,
  202. private statsContainer: HTMLElement
  203. ) {
  204. this.comboElement = scoreContainer.querySelector('.combo');
  205. this.scoreElement = scoreContainer.querySelector('.score');
  206. this.maxComboElement = scoreContainer.querySelector('.max-combo');
  207. this.finishedElement = scoreContainer.querySelector('.finished');
  208. this.hitElement = statsContainer.querySelector('.hit');
  209. this.missedElement = statsContainer.querySelector('.missed');
  210. this.skippedElement = statsContainer.querySelector('.skipped');
  211. this.observer = result => this.update(result);
  212. this.score = new Score();
  213. this.setValues();
  214. }
  215. setInputState(inputState: InputState): void {
  216. this.clearObservers();
  217. this.inputState = inputState;
  218. if (this.inputState != null) {
  219. this.inputState.map((_, m) => {
  220. m.addObserver(this.observer);
  221. });
  222. }
  223. }
  224. intervalEnd(finished: boolean): void {
  225. this.score.intervalEnd(finished);
  226. this.setValues();
  227. }
  228. update(result: TransitionResult): void {
  229. this.score.update(result);
  230. this.setValues();
  231. }
  232. setValues(): void {
  233. this.comboElement.textContent = this.score.combo == 0 ? '' : this.score.combo+' combo';
  234. this.scoreElement.textContent = this.score.score+'';
  235. this.maxComboElement.textContent = this.score.maxCombo+'';
  236. this.finishedElement.textContent = this.score.finished+'';
  237. this.hitElement.textContent = this.score.hit+'';
  238. this.missedElement.textContent = this.score.missed+'';
  239. this.skippedElement.textContent = this.score.skipped+'';
  240. }
  241. private clearObservers(): void {
  242. if (this.inputState != null) {
  243. this.inputState.map((_, machine) => {
  244. machine.removeObserver(this.observer);
  245. });
  246. }
  247. }
  248. destroy(): void {
  249. this.clearObservers();
  250. }
  251. }
  252. }