Selaa lähdekoodia

Implement select screen

Thomas Dy 7 vuotta sitten
vanhempi
commit
f851e2759a
6 muutettua tiedostoa jossa 514 lisäystä ja 16 poistoa
  1. 25 1
      dist/index.html
  2. 45 0
      dist/levels.json
  3. 186 9
      dist/style.css
  4. 3 4
      src/display.ts
  5. 43 2
      src/game.ts
  6. 212 0
      src/select.ts

+ 25 - 1
dist/index.html

@@ -2,14 +2,38 @@
   <head>
     <title>Typing Freaks</title>
     <link rel="stylesheet" href="style.css" />
+    <link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet" />
   </head>
   <body>
     <div id="container">
       <div id="background"></div>
       <div id="loading">Loading...</div>
-      <div id="select"></div>
+      <div id="folder-info">
+        <i class="left material-icons">chevron_left</i>
+        <i class="material-icons">album</i>
+        <i class="right material-icons">chevron_right</i>
+        <span class="label"></span>
+      </div>
+      <div id="song-info"></div>
+      <div id="song-list"></div>
       <div id="game"></div>
     </div>
+    <template id="song-info-template">
+      <div class="song-info">
+        <div class="genre"></div>
+        <div class="creator"></div>
+        <div class="title"></div>
+      </div>
+    </template>
+    <template id="song-item-template">
+      <div class="song-item">
+        <div class="difficulty-bg normal"></div>
+        <div class="difficulty-bg"></div>
+        <div class="difficulty"></div>
+        <div class="creator"></div>
+        <div class="title"></div>
+      </div>
+    </template>
     <script type="text/javascript" src="bundle.js"></script>
   </body>
 </html>

+ 45 - 0
dist/levels.json

@@ -54,6 +54,51 @@
           ]
         }
       ]
+    },
+    {
+      "name": "Test 2",
+      "levels": [
+        {
+          "name": "Test 1",
+          "lines": []
+        },
+        {
+          "name": "Test 2",
+          "lines": []
+        },
+        {
+          "name": "Test 3",
+          "lines": []
+        },
+        {
+          "name": "Test 4",
+          "lines": []
+        },
+        {
+          "name": "Test 5",
+          "lines": []
+        },
+        {
+          "name": "Test 6",
+          "lines": []
+        },
+        {
+          "name": "Test 7",
+          "lines": []
+        },
+        {
+          "name": "Test 8",
+          "lines": []
+        },
+        {
+          "name": "Test 9",
+          "lines": []
+        },
+        {
+          "name": "Test 10",
+          "lines": []
+        }
+      ]
     }
   ]
 }

+ 186 - 9
dist/style.css

@@ -4,30 +4,65 @@
 
   position: relative;
   color: var(--base-color);
+  font-family: sans;
   height: 450px;
   width: 800px;
   margin: 0 auto;
   overflow: hidden;
+
+  display: grid;
+  grid-template-columns: [start] auto [right] 300px [end];
+  grid-template-rows: [top] 50px [header] auto [game] 150px [bottom];
 }
 
 #container > div {
-  display: grid;
-  position: absolute;
-  height: 100%;
-  width: 100%;
+  overflow: hidden;
+  position: relative;
+  transition: top 0.5s, left 0.5s, opacity 0.5s;
+  opacity: 0;
 }
 
 #loading {
+  grid-column: start / end;
+  grid-row: top / bottom;
+  align-self: center;
+  justify-self: center;
+}
+
+#song-info {
+  grid-column: start / right;
+  grid-row: header / game;
+  display: grid;
+  left: -500px;
+}
+
+#song-list {
+  grid-column: right / end;
+  grid-row: top / bottom;
+  left: 300px;
+}
+
+#folder-info {
+  grid-column: start / right;
+  grid-row: top / header;
+  display: flex;
+  flex-direction: row;
   align-items: center;
-  justify-items: center;
-  transition: opacity 0.5s;
+  left: -500px;
 }
 
-#loading.finished {
-  opacity: 0;
+#game {
+  top: 150px;
+  grid-column: start / end;
+  grid-row: game / bottom;
 }
 
-#background {
+#container #background {
+  opacity: 1;
+  position: absolute;
+  height: 100%;
+  width: 100%;
+  z-index: -99;
   background-color: black;
 }
 
@@ -43,6 +78,148 @@
   opacity: 1;
 }
 
+#container.loading #loading {
+  opacity: 1;
+}
+
+#container.loading #loading.finished {
+  opacity: 0;
+}
+
+#container.loading #song-info,
+#container.loading #folder-info {
+  top: 50px;
+  left: 0;
+}
+
+#container.loading #song-list {
+  left: 50px;
+}
+
+#container.select #song-info,
+#container.select #folder-info {
+  opacity: 1;
+  top: 0;
+  left: 0;
+}
+
+#container.select #song-list {
+  opacity: 1;
+  left: 0;
+}
+
+#container.game #folder-info {
+  opacity: 1;
+  top: 0;
+  left: 0;
+}
+
+#container.game #song-info {
+  opacity: 1;
+  top: -150px;
+  left: 0;
+}
+
+#container.game #game {
+  opacity: 1;
+  top: 0;
+}
+
+#folder-info .left:hover,
+#folder-info .right:hover {
+  text-shadow: 0px 0px 5px var(--base-color);
+}
+
+.song-info {
+  align-self: end;
+  margin-left: 20px;
+  text-shadow: 0px 0px 5px var(--base-color);
+}
+
+.song-info .genre {
+  font-size: 12px;
+}
+
+.song-info .creator {
+  font-size: 20px;
+  line-height: 0.5;
+}
+
+.song-info .title {
+  font-size: 30px;
+}
+
+.song-list {
+  margin-left: 20px;
+  transition: margin-top 0.2s;
+}
+
+.song-item {
+  height: 40px;
+  display: grid;
+  grid-template-columns: 40px auto;
+  grid-template-rows: 12px 28px;
+  grid-template-areas:
+    "diff creator"
+    "diff title";
+  transition: margin-left 0.2s;
+}
+
+.song-item .difficulty {
+  grid-area: diff;
+  align-self: end;
+  justify-self: end;
+  margin-bottom: 1px;
+  margin-right: 5px;
+  font-style: italic;
+  font-weight: bold;
+}
+
+.song-item .difficulty-bg {
+  grid-area: diff;
+  align-self: center;
+  justify-self: center;
+  height: 30px;
+  width: 30px;
+  background: radial-gradient(circle at 10% 10%, rgba(0, 255, 255, 1), transparent);
+  border-radius: 20%;
+  opacity: 0;
+  transition: opacity 0.2s ease-in-out;
+}
+
+.song-item .difficulty-bg.normal {
+  background: radial-gradient(circle at 10% 10%, rgba(255, 255, 255, 0.6), transparent);
+  opacity: 1;
+}
+
+.song-item:hover {
+  text-shadow: 0px 0px 5px var(--base-color);
+}
+
+.song-item.selected {
+  margin-left: -10px;
+  text-shadow: 0px 0px 5px var(--base-color);
+}
+
+.song-item.selected .difficulty-bg {
+  opacity: 1;
+  box-shadow: -1px -1px 10px -2px var(--base-color);
+}
+
+.song-item.selected .difficulty-bg.normal {
+  opacity: 0;
+}
+
+.song-item .creator {
+  grid-area: creator;
+  font-size: 12px;
+}
+
+.song-item .title {
+  grid-area: title;
+  font-size: 20px;
+}
+
 .kana {
   display: inline-block;
   position: relative;

+ 3 - 4
src/display.ts

@@ -197,7 +197,6 @@ namespace display {
     inputState: InputState | null;
     mainAreaController: MainAreaController;
     progressController: TrackProgressController | null;
-    listener: (event: KeyboardEvent) => void;
     state: LevelState;
     track: audio.Track | null;
 
@@ -208,13 +207,11 @@ namespace display {
       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 != "@");
@@ -326,7 +323,9 @@ namespace display {
     }
 
     destroy(): void {
-      document.removeEventListener('keydown', this.listener);
+      if (this.track != null) {
+        this.track.stop();
+      }
     }
   }
 

+ 43 - 2
src/game.ts

@@ -2,6 +2,7 @@
 /// <reference path="audio.ts" />
 /// <reference path="display.ts" />
 /// <reference path="background.ts" />
+/// <reference path="select.ts" />
 
 namespace game {
   enum GameState {
@@ -90,15 +91,34 @@ namespace game {
     audioManager: audio.AudioManager;
     bgManager: background.BackgroundManager;
     assets: GameSounds | null;
+    state: GameState;
+    selectScreen: SelectScreen | null;
+    gameController: display.LevelController | null;
 
     constructor(container: HTMLElement, configUrl: string) {
       this.container = container;
       this.configUrl = configUrl;
       this.audioManager = new audio.AudioManager();
       this.bgManager = new background.BackgroundManager(container.querySelector('#background'));
+      this.state = GameState.LOADING;
+
+      document.addEventListener('keydown', (event) => {
+        if (!event.ctrlKey && !event.metaKey) {
+          if (this.state === GameState.SELECT) {
+            this.selectScreen.handleInput(event.key);
+          } else if (this.state === GameState.PLAYING) {
+            if (event.key === 'Escape') {
+              this.onBackToSelect();
+            } else {
+              this.gameController.handleInput(event.key);
+            }
+          }
+        }
+      });
     }
 
     start(): void {
+      this.container.classList.add('loading');
       let loadingScreen = new LoadingScreen(this);
       loadingScreen.load();
     }
@@ -108,8 +128,29 @@ namespace game {
       this.container.style.setProperty('--base-color', config.baseColor);
       this.container.style.setProperty('--highlight-color', config.highlightColor);
 
-      let controller = new display.LevelController(this.audioManager, this.config.levelSets[0].levels[0]);
-      this.container.querySelector('#game').appendChild(controller.element);
+      this.selectScreen = new SelectScreen(this);
+      this.container.classList.remove('loading');
+      this.container.classList.add('select');
+      this.state = GameState.SELECT;
+    }
+
+    onSongSelect(level: level.Level): void {
+      this.container.classList.remove('select');
+      this.container.classList.add('game');
+      this.gameController = new display.LevelController(this.audioManager, level);
+      let gameContainer = this.container.querySelector('#game');
+      while (gameContainer.lastChild != null) {
+        gameContainer.removeChild(gameContainer.lastChild);
+      }
+      gameContainer.appendChild(this.gameController.element);
+      this.state = GameState.PLAYING;
+    }
+
+    onBackToSelect(): void {
+      this.container.classList.remove('game');
+      this.container.classList.add('select');
+      this.gameController.destroy();
+      this.state = GameState.SELECT;
     }
   }
 }

+ 212 - 0
src/select.ts

@@ -0,0 +1,212 @@
+/// <reference path="game.ts" />
+
+namespace game {
+  export class SelectScreen {
+    controller: MainController;
+    folderInfo: HTMLElement;
+    songInfo: HTMLElement;
+    songList: HTMLElement;
+    currentFolderIndex: number;
+    folderController: FolderSelectController;
+    listControllers: SongListController[];
+    init: boolean;
+
+    get levelSets() {
+      return this.controller.config.levelSets;
+    }
+
+    get currentLevelSet() {
+      return this.levelSets[this.currentFolderIndex];
+    }
+
+    get activeListController() {
+      return this.listControllers[this.currentFolderIndex];
+    }
+
+    constructor(controller: MainController) {
+      this.controller = controller;
+      let container = controller.container;
+      this.folderInfo = container.querySelector('#folder-info');
+      this.songInfo = container.querySelector('#song-info');
+      this.songList = container.querySelector('#song-list');
+
+      this.listControllers = [];
+      this.levelSets.forEach(levelSet => {
+        let controller = new SongListController(
+          levelSet.levels,
+          (index) => this.selectSong(index),
+          (index) => this.chooseSong(index)
+        );
+        this.listControllers.push(controller);
+      });
+
+      this.init = true;
+
+      this.folderController = new FolderSelectController(
+        this.folderInfo,
+        this.levelSets,
+        (index) => this.selectLevelSet(index)
+      );
+
+      this.init = false;
+    }
+
+    handleInput(key: string): void {
+      this.activeListController.handleInput(key);
+      this.folderController.handleInput(key);
+    }
+
+    selectSong(index: number): void {
+      if (!this.init) {
+        this.controller.assets.selectSound.play();
+      }
+      let songInfoComponent = new SongInfoComponent(this.currentLevelSet.levels[index]);
+      while (this.songInfo.firstChild) {
+        this.songInfo.removeChild(this.songInfo.firstChild);
+      }
+      this.songInfo.appendChild(songInfoComponent.element);
+    }
+
+    chooseSong(index: number): void {
+      this.controller.assets.decideSound.play();
+      this.controller.onSongSelect(this.currentLevelSet.levels[index]);
+    }
+
+    selectLevelSet(index: number): void {
+      this.currentFolderIndex = index;
+      if (this.songList.lastChild) {
+        this.songList.removeChild(this.songList.lastChild);
+      }
+      this.songList.appendChild(this.activeListController.element);
+      this.selectSong(this.activeListController.currentIndex);
+    }
+  }
+
+  class FolderSelectController {
+    labelElement: HTMLElement;
+    levelSets: level.LevelSet[];
+    currentIndex: number;
+    onFolderChange: (index: number) => void;
+
+    constructor(element: HTMLElement, levelSets: level.LevelSet[], onFolderChange: (index: number) => void) {
+      this.labelElement = element.querySelector('.label');
+      this.levelSets = levelSets;
+      this.currentIndex = 0;
+      this.onFolderChange = onFolderChange;
+
+      element.querySelector('.left').addEventListener('click', () => this.scroll(-1));
+      element.querySelector('.right').addEventListener('click', () => this.scroll(1));
+      this.scroll(0);
+    }
+
+    handleInput(key: string): void {
+      if (key === 'ArrowLeft') {
+        this.scroll(-1);
+      } else if (key === 'ArrowRight') {
+        this.scroll(1);
+      }
+    }
+
+    scroll(offset: number): void {
+      this.currentIndex += offset;
+      while (this.currentIndex < 0) {
+        this.currentIndex += this.levelSets.length;
+      }
+      this.currentIndex %= this.levelSets.length;
+      this.labelElement.textContent = this.levelSets[this.currentIndex].name;
+      this.onFolderChange(this.currentIndex);
+    }
+  }
+
+  class SongInfoComponent {
+    element: DocumentFragment;
+
+    constructor(level: level.Level) {
+      let template: HTMLTemplateElement = document.querySelector('#song-info-template');
+      this.element = document.importNode(template.content, true);
+      this.element.querySelector('.genre').textContent = level.genre;
+      this.element.querySelector('.creator').textContent = level.creator;
+      this.element.querySelector('.title').textContent = level.name;
+    }
+  }
+
+  class SongListController {
+    element: HTMLElement;
+    levels: level.Level[];
+    currentIndex: number;
+    onSongChange: (index: number) => void;
+    onSongChoose: (index: number) => void;
+
+    constructor(
+      levels: level.Level[],
+      onSongChange: (index: number) => void,
+      onSongChoose: (index: number) => void
+    ) {
+      this.element = document.createElement('div');
+      this.levels = levels;
+      this.currentIndex = 0;
+      this.onSongChange = onSongChange;
+      this.onSongChoose = onSongChoose;
+
+      this.element.className = 'song-list';
+      this.element.style.marginTop = '200px';
+
+      let template: HTMLTemplateElement = document.querySelector('#song-item-template');
+      this.levels.forEach((level, index) => {
+        let element = document.importNode(template.content, true);
+        element.querySelector('.creator').textContent = level.creator;
+        element.querySelector('.title').textContent = level.name;
+        element.querySelector('.difficulty').textContent = level.difficulty;
+        element.querySelector('.song-item').addEventListener('click', (event) => this.click(index));
+        this.element.appendChild(element);
+      });
+      this.element.children[0].classList.add('selected');
+    }
+
+    handleInput(key: string): void {
+      if (key === 'ArrowUp') {
+        this.scroll(-1);
+      } else if (key === 'ArrowDown') {
+        this.scroll(1);
+      } else if (key === 'PageUp') {
+        this.scroll(-5);
+      } else if (key === 'PageDown') {
+        this.scroll(5);
+      } else if (key === ' ') {
+        this.choose();
+      }
+    }
+
+    scroll(offset: number) {
+      let target = this.currentIndex + offset;
+      target = Math.max(0, Math.min(this.levels.length - 1, target));
+      this.select(target);
+    }
+
+    click(index: number) {
+      if (this.currentIndex === index) {
+        this.choose();
+      } else {
+        this.select(index);
+      }
+    }
+
+    select(index: number) {
+      if (this.currentIndex === index) return;
+
+      let offset = 200 - index * 40;
+      this.element.style.marginTop = offset+'px';
+
+      let nextElement = this.element.children[index] as HTMLElement;
+      let currElement = this.element.children[this.currentIndex] as HTMLElement;
+      currElement.classList.remove('selected');
+      nextElement.classList.add('selected');
+      this.currentIndex = index;
+      this.onSongChange(index);
+    }
+
+    choose() {
+      this.onSongChoose(this.currentIndex);
+    }
+  }
+}