select.ts 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. /// <reference path="game.ts" />
  2. /// <reference path="util.ts" />
  3. namespace game {
  4. export class SelectScreen {
  5. controller: MainController;
  6. folderInfo: HTMLElement;
  7. songInfo: HTMLElement;
  8. songList: HTMLElement;
  9. currentFolderIndex: number;
  10. folderController: FolderSelectController;
  11. listControllers: SongListController[];
  12. init: boolean;
  13. get levelSets() {
  14. return this.controller.config.levelSets;
  15. }
  16. get currentLevelSet() {
  17. return this.levelSets[this.currentFolderIndex];
  18. }
  19. get activeListController() {
  20. return this.listControllers[this.currentFolderIndex];
  21. }
  22. constructor(controller: MainController) {
  23. this.controller = controller;
  24. let container = controller.container;
  25. this.folderInfo = container.querySelector('#folder-info');
  26. this.songInfo = container.querySelector('#song-info');
  27. this.songList = container.querySelector('#song-list');
  28. this.listControllers = [];
  29. this.levelSets.forEach(levelSet => {
  30. let controller = new SongListController(
  31. levelSet.levels,
  32. (index) => this.selectSong(index),
  33. (index) => this.chooseSong(index)
  34. );
  35. this.listControllers.push(controller);
  36. });
  37. this.init = true;
  38. this.folderController = new FolderSelectController(
  39. this.folderInfo,
  40. this.levelSets,
  41. (index) => this.selectLevelSet(index)
  42. );
  43. this.init = false;
  44. }
  45. handleInput(key: string): void {
  46. this.activeListController.handleInput(key);
  47. this.folderController.handleInput(key);
  48. }
  49. selectSong(index: number): void {
  50. if (!this.init) {
  51. this.controller.assets.selectSound.play();
  52. }
  53. let songInfoComponent = new SongInfoComponent(this.currentLevelSet.levels[index]);
  54. util.clearChildren(this.songInfo);
  55. this.songInfo.appendChild(songInfoComponent.element);
  56. }
  57. chooseSong(index: number): void {
  58. this.controller.assets.decideSound.play();
  59. this.controller.onSongSelect(this.currentLevelSet.levels[index]);
  60. }
  61. selectLevelSet(index: number): void {
  62. this.currentFolderIndex = index;
  63. util.clearChildren(this.songList);
  64. this.songList.appendChild(this.activeListController.element);
  65. this.selectSong(this.activeListController.currentIndex);
  66. }
  67. }
  68. class FolderSelectController {
  69. labelElement: HTMLElement;
  70. levelSets: level.LevelSet[];
  71. currentIndex: number;
  72. onFolderChange: (index: number) => void;
  73. constructor(element: HTMLElement, levelSets: level.LevelSet[], onFolderChange: (index: number) => void) {
  74. this.labelElement = element.querySelector('.label');
  75. this.levelSets = levelSets;
  76. this.currentIndex = 0;
  77. this.onFolderChange = onFolderChange;
  78. element.querySelector('.left').addEventListener('click', () => this.scroll(-1));
  79. element.querySelector('.right').addEventListener('click', () => this.scroll(1));
  80. this.scroll(0);
  81. }
  82. handleInput(key: string): void {
  83. if (key === 'ArrowLeft') {
  84. this.scroll(-1);
  85. } else if (key === 'ArrowRight') {
  86. this.scroll(1);
  87. }
  88. }
  89. scroll(offset: number): void {
  90. this.currentIndex += offset;
  91. while (this.currentIndex < 0) {
  92. this.currentIndex += this.levelSets.length;
  93. }
  94. this.currentIndex %= this.levelSets.length;
  95. this.labelElement.textContent = this.levelSets[this.currentIndex].name;
  96. this.onFolderChange(this.currentIndex);
  97. }
  98. }
  99. class SongInfoComponent {
  100. element: DocumentFragment;
  101. constructor(level: level.Level) {
  102. this.element = util.loadTemplate('song-info');
  103. this.element.querySelector('.genre').textContent = level.genre;
  104. this.element.querySelector('.creator').textContent = level.creator;
  105. this.element.querySelector('.title').textContent = level.name;
  106. }
  107. }
  108. class SongListController {
  109. element: HTMLElement;
  110. levels: level.Level[];
  111. currentIndex: number;
  112. onSongChange: (index: number) => void;
  113. onSongChoose: (index: number) => void;
  114. constructor(
  115. levels: level.Level[],
  116. onSongChange: (index: number) => void,
  117. onSongChoose: (index: number) => void
  118. ) {
  119. this.element = document.createElement('div');
  120. this.levels = levels;
  121. this.currentIndex = 0;
  122. this.onSongChange = onSongChange;
  123. this.onSongChoose = onSongChoose;
  124. this.element.className = 'song-list';
  125. this.element.style.marginTop = '200px';
  126. this.levels.forEach((level, index) => {
  127. let element = util.loadTemplate('song-item');
  128. element.querySelector('.creator').textContent = level.creator;
  129. element.querySelector('.title').textContent = level.name;
  130. element.querySelector('.difficulty').textContent = level.difficulty;
  131. element.querySelector('.song-item').addEventListener('click', (event) => this.click(index));
  132. this.element.appendChild(element);
  133. });
  134. this.element.children[0].classList.add('selected');
  135. }
  136. handleInput(key: string): void {
  137. if (key === 'ArrowUp') {
  138. this.scroll(-1);
  139. } else if (key === 'ArrowDown') {
  140. this.scroll(1);
  141. } else if (key === 'PageUp') {
  142. this.scroll(-5);
  143. } else if (key === 'PageDown') {
  144. this.scroll(5);
  145. } else if (key === ' ') {
  146. this.choose();
  147. }
  148. }
  149. scroll(offset: number) {
  150. let target = this.currentIndex + offset;
  151. target = Math.max(0, Math.min(this.levels.length - 1, target));
  152. this.select(target);
  153. }
  154. click(index: number) {
  155. if (this.currentIndex === index) {
  156. this.choose();
  157. } else {
  158. this.select(index);
  159. }
  160. }
  161. select(index: number) {
  162. if (this.currentIndex === index) return;
  163. let offset = 200 - index * 40;
  164. this.element.style.marginTop = offset+'px';
  165. let nextElement = this.element.children[index] as HTMLElement;
  166. let currElement = this.element.children[this.currentIndex] as HTMLElement;
  167. currElement.classList.remove('selected');
  168. nextElement.classList.add('selected');
  169. this.currentIndex = index;
  170. this.onSongChange(index);
  171. }
  172. choose() {
  173. this.onSongChoose(this.currentIndex);
  174. }
  175. }
  176. }