typing.ts 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360
  1. /// <reference path="../audio.ts" />
  2. /// <reference path="../kana.ts" />
  3. /// <reference path="../level.ts" />
  4. /// <reference path="../display.ts" />
  5. /// <reference path="common.ts" />
  6. namespace game {
  7. import Level = level.Level;
  8. class TypingScreenContext {
  9. track: audio.Track | null = null;
  10. constructor(
  11. readonly context: GameContext,
  12. readonly level: Level,
  13. readonly switchClosure: (screen: Screen | null) => void
  14. ) {}
  15. get container() {
  16. return this.context.container;
  17. }
  18. get audioManager() {
  19. return this.context.audioManager;
  20. }
  21. get bgManager() {
  22. return this.context.bgManager;
  23. }
  24. switchScreen(screen: Screen | null): void {
  25. this.switchClosure(screen);
  26. }
  27. }
  28. export class TypingScreen extends ScreenManager implements Screen {
  29. readonly name: string = 'game';
  30. constructor(
  31. readonly context: GameContext,
  32. readonly level: Level,
  33. readonly prevScreen: Screen
  34. ) {
  35. super(context.container);
  36. }
  37. enter(): void {
  38. if (this.level.background) {
  39. this.context.bgManager.setBackground(this.level.background);
  40. }
  41. let context = new TypingScreenContext(this.context, this.level, (screen) => this.switchScreen(screen));
  42. let loadingScreen = new TypingLoadingScreen(context);
  43. this.switchScreen(loadingScreen);
  44. }
  45. handleInput(key: string): void {
  46. if (this.activeScreen !== null) {
  47. this.activeScreen.handleInput(key);
  48. }
  49. }
  50. switchScreen(screen: Screen | null): void {
  51. super.switchScreen(screen);
  52. if (screen == null) {
  53. this.context.switchScreen(this.prevScreen);
  54. }
  55. }
  56. exit(): void {}
  57. transitionExit(): void {}
  58. }
  59. class TypingLoadingScreen implements Screen {
  60. readonly name: string = 'game-loading';
  61. barElement: HTMLElement | null = null;
  62. textElement: HTMLElement | null = null;
  63. readyElement: HTMLElement | null = null;
  64. isReady: boolean = false;
  65. fnContext: util.FnContext = new util.FnContext();
  66. constructor(readonly context: TypingScreenContext) {}
  67. enter(): void {
  68. let loader: HTMLElement = util.getElement(this.context.container, '#loader');
  69. this.readyElement = util.getElement(this.context.container, '#ready');
  70. if (this.context.level.audio != null) {
  71. loader.style.visibility = 'visible';
  72. this.barElement = util.getElement(loader, '.progress-bar .shade');
  73. this.textElement = util.getElement(loader, '.label');
  74. this.barElement.style.width = '0%';
  75. this.textElement.textContent = 'music loading';
  76. this.readyElement.querySelector('.status')!.textContent = 'Loading';
  77. this.readyElement.querySelector('.message')!.textContent = 'please wait';
  78. this.fnContext.invalidate();
  79. const videoId = youtube.getVideoId(this.context.level.audio);
  80. const progressListener = this.fnContext.wrap((percentage: number) => {
  81. this.barElement!.style.width = `${percentage}%`;
  82. });
  83. let trackPromise;
  84. if (videoId !== null) {
  85. const ytElement = document.createElement('div');
  86. trackPromise = this.context.audioManager.loadTrackFromYoutube(
  87. videoId,
  88. ytElement,
  89. progressListener,
  90. );
  91. this.context.bgManager.setVideo(ytElement);
  92. if (this.context.level.background == undefined) {
  93. trackPromise.then((track) => {
  94. track.playPromise.then(track.fnContext.wrap(() => {
  95. this.context.bgManager.showVideo();
  96. }));
  97. track.finishPromise.then(track.fnContext.wrap(() => {
  98. this.context.bgManager.hideVideo();
  99. }));
  100. });
  101. }
  102. } else {
  103. trackPromise = this.context.audioManager.loadTrackWithProgress(
  104. this.context.level.audio,
  105. progressListener,
  106. )
  107. }
  108. trackPromise.then(this.fnContext.wrap((track: audio.Track) => {
  109. this.context.track = track;
  110. this.barElement!.style.width = '100%';
  111. this.textElement!.textContent = 'music loaded';
  112. this.setReady();
  113. }));
  114. } else {
  115. loader.style.visibility = 'hidden';
  116. this.setReady();
  117. }
  118. }
  119. setReady(): void {
  120. this.readyElement!.querySelector('.status')!.textContent = 'Ready';
  121. this.readyElement!.querySelector('.message')!.textContent = 'press space to start';
  122. this.isReady = true;
  123. }
  124. handleInput(key: string): void {
  125. if (key == 'Escape') {
  126. this.context.switchScreen(null);
  127. } else if (this.isReady && key === ' ') {
  128. this.context.switchScreen(new TypingPlayingScreen(this.context));
  129. }
  130. }
  131. exit(): void {
  132. this.fnContext.invalidate();
  133. }
  134. transitionExit(): void {
  135. if (this.barElement) {
  136. this.barElement.style.width = '0%';
  137. }
  138. }
  139. }
  140. class TypingPlayingScreen implements Screen {
  141. readonly name: string = 'game-playing';
  142. gameContainer: HTMLElement;
  143. currentIndex: number;
  144. inputState: kana.KanaInputState | null;
  145. isWaiting: boolean;
  146. kanjiElement: HTMLElement;
  147. kanaController: display.KanaDisplayController;
  148. romajiController: display.RomajiDisplayController;
  149. progressController: display.TrackProgressController | null;
  150. scoreController: display.ScoreController;
  151. lines: level.Line[];
  152. constructor(readonly context: TypingScreenContext) {
  153. this.gameContainer = util.getElement(this.context.container, '#game');
  154. this.currentIndex = -1;
  155. this.inputState = null;
  156. this.isWaiting = false;
  157. this.kanjiElement = util.getElement(this.gameContainer, '.kanji-line');
  158. this.romajiController = new display.RomajiDisplayController(
  159. util.getElement(this.gameContainer, '.romaji-first'),
  160. util.getElement(this.gameContainer, '.romaji-line')
  161. );
  162. this.kanaController = new display.KanaDisplayController(
  163. util.getElement(this.gameContainer, '.kana-line')
  164. );
  165. this.progressController = null;
  166. this.scoreController = new display.ScoreController(
  167. util.getElement(this.gameContainer, '.score-line'),
  168. util.getElement(this.gameContainer, '.stats-line')
  169. );
  170. this.lines = this.context.level.lines;
  171. }
  172. enter(): void {
  173. let progressElement: HTMLElement = this.gameContainer.querySelector<HTMLElement>('.track-progress')!;
  174. if (this.context.level.audio == null) {
  175. progressElement.style.visibility = 'hidden';
  176. this.lines = this.context.level.lines.filter(line => line.kana != "@");
  177. } else {
  178. progressElement.style.visibility = 'visible';
  179. this.progressController = new display.TrackProgressController(
  180. progressElement,
  181. this.lines
  182. );
  183. this.progressController.setListener(event => this.onIntervalEnd());
  184. }
  185. this.onStart();
  186. }
  187. setWaiting(waiting: boolean): void {
  188. this.gameContainer.classList.toggle('waiting', waiting);
  189. this.isWaiting = waiting;
  190. }
  191. onStart(): void {
  192. this.nextLine();
  193. if (this.context.track !== null) {
  194. this.progressController!.start();
  195. this.context.track.play();
  196. }
  197. this.setWaiting(false);
  198. this.checkComplete();
  199. }
  200. checkComplete(): void {
  201. let currentLine = this.lines[this.currentIndex];
  202. if (currentLine != null && currentLine.kana == '@' && currentLine.kanji == '@') {
  203. this.onComplete(true);
  204. }
  205. }
  206. onIntervalEnd(): void {
  207. if (this.isWaiting) {
  208. this.setWaiting(false);
  209. } else {
  210. this.nextLine();
  211. this.scoreController.intervalEnd(false);
  212. }
  213. if (this.currentIndex >= this.lines.length) {
  214. this.finish();
  215. }
  216. this.checkComplete();
  217. }
  218. onComplete(autoComplete: boolean = false): void {
  219. this.nextLine();
  220. if (!autoComplete) {
  221. this.scoreController.intervalEnd(true);
  222. }
  223. if (this.context.track !== null) {
  224. this.setWaiting(true);
  225. } else {
  226. if (this.currentIndex >= this.lines.length) {
  227. this.finish();
  228. }
  229. }
  230. }
  231. handleInput(key: string): void {
  232. if (key === 'Escape') {
  233. this.finish();
  234. } else if (!this.isWaiting) {
  235. if (this.inputState !== null && /^[-_ a-z]$/.test(key)) {
  236. if (this.inputState.handleInput(key)) {
  237. this.onComplete();
  238. }
  239. }
  240. }
  241. }
  242. nextLine(): void {
  243. if (this.currentIndex < this.lines.length) {
  244. this.currentIndex += 1;
  245. }
  246. if (this.currentIndex < this.lines.length) {
  247. this.setLine(this.lines[this.currentIndex]);
  248. } else {
  249. this.setLine({ kanji: '@', kana: '@' });
  250. }
  251. }
  252. setLine(line: level.Line): void {
  253. let kanji, inputState;
  254. if (line.kanji === '@') {
  255. kanji = '';
  256. } else {
  257. kanji = line.kanji;
  258. }
  259. if (line.kana === '@') {
  260. inputState = null;
  261. } else {
  262. inputState = new kana.KanaInputState(line.kana);
  263. }
  264. this.inputState = inputState;
  265. this.kanjiElement.textContent = kanji;
  266. this.kanaController.setInputState(this.inputState);
  267. this.romajiController.setInputState(this.inputState);
  268. this.scoreController.setInputState(this.inputState);
  269. }
  270. finish(): void {
  271. this.context.switchScreen(new TypingFinishScreen(
  272. this.context,
  273. this.scoreController.score
  274. ));
  275. }
  276. exit(): void {}
  277. transitionExit(): void {
  278. this.kanaController.destroy();
  279. this.romajiController.destroy();
  280. if (this.context.track !== null) {
  281. this.progressController!.destroy();
  282. }
  283. this.scoreController.destroy();
  284. }
  285. }
  286. class TypingFinishScreen implements Screen {
  287. name: string = 'game-finished';
  288. constructor(
  289. readonly context: TypingScreenContext,
  290. readonly score: display.Score
  291. ) {
  292. let container = this.context.container.querySelector('#score')!;
  293. container.querySelector('.score')!.textContent = this.score.score+'';
  294. container.querySelector('.max-combo')!.textContent = this.score.maxCombo+'';
  295. container.querySelector('.finished')!.textContent = this.score.finished+'';
  296. container.querySelector('.hit')!.textContent = this.score.hit+'';
  297. container.querySelector('.missed')!.textContent = this.score.missed+'';
  298. container.querySelector('.skipped')!.textContent = this.score.skipped+'';
  299. }
  300. enter(): void {}
  301. handleInput(key: string): void {
  302. if (key === ' ' || key === 'Escape') {
  303. this.context.switchScreen(null);
  304. }
  305. }
  306. exit(): void {
  307. if (this.context.track !== null) {
  308. this.context.track.exit();
  309. }
  310. }
  311. transitionExit(): void {}
  312. }
  313. }