2
0

level.ts 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. /**
  2. * This module represents the levels for the game. Each level consists of lines
  3. * that you have to complete. Each line has the kanji of the line, which is used
  4. * solely for display and the kana of the line which the input is based.
  5. */
  6. namespace level {
  7. export interface Line {
  8. kanji: string,
  9. kana: string,
  10. start?: number,
  11. end?: number
  12. }
  13. export interface Level {
  14. name: string,
  15. creator?: string,
  16. genre?: string,
  17. difficulty?: string,
  18. audio?: string,
  19. lines: Line[]
  20. }
  21. export interface LevelSet {
  22. name: string,
  23. levels: Level[]
  24. }
  25. export interface Config {
  26. background: string,
  27. selectMusic?: string,
  28. selectSound: string,
  29. decideSound: string,
  30. baseColor: string,
  31. highlightColor: string,
  32. levelSets: LevelSet[]
  33. }
  34. export function loadFromJson(url: string): Promise<Config> {
  35. return window.fetch(url)
  36. .then(response => response.json())
  37. }
  38. let parser = new DOMParser();
  39. function parseXML(response: Response): Promise<Document> {
  40. return response.text().then(text => {
  41. let normalized = text.replace(/[“”]/g, '"');
  42. return parser.parseFromString(normalized, "text/xml");
  43. });
  44. }
  45. export function loadFromTM(base: string): Promise<Config> {
  46. let settingsXML = window.fetch(base+'/settings.xml').then(parseXML);
  47. let levelSets = window.fetch(base+'/folderlist.xml')
  48. .then(parseXML)
  49. .then(dom => parseTMFolderList(base, dom));
  50. return Promise.all([settingsXML, levelSets]).then(pair => {
  51. return parseTMSettings(base, pair[1], pair[0]);
  52. })
  53. }
  54. function parseTMSettings(base: string, levelSets: LevelSet[], dom: Document): Config {
  55. function getData(tag: string): string | null {
  56. let elem = dom.querySelector(tag);
  57. if (elem === null) {
  58. return null;
  59. } else {
  60. return base+'/'+elem.getAttribute('src');
  61. }
  62. }
  63. let background = getData('background');
  64. let selectMusic = getData('selectmusic');
  65. let selectSound = getData('selectsound');
  66. let decideSound = getData('decidesound');
  67. return {
  68. background,
  69. baseColor: 'white',
  70. highlightColor: 'blue',
  71. selectMusic,
  72. selectSound,
  73. decideSound,
  74. levelSets
  75. }
  76. }
  77. function parseTMFolderList(base: string, dom: Document): Promise<LevelSet[]> {
  78. let folderList = dom.querySelectorAll('folder');
  79. let promises = [];
  80. for (let i = 0; i < folderList.length; ++i) {
  81. let folder = folderList[i];
  82. let name = folder.getAttribute('name');
  83. let path = folder.getAttribute('path');
  84. let promise = window.fetch(base+'/'+path)
  85. .then(parseXML)
  86. .then(dom => parseTMFolder(base, name, dom))
  87. promises.push(promise);
  88. }
  89. return Promise.all(promises);
  90. }
  91. function parseTMFolder(base: string, name: string, dom: Document): Promise<LevelSet> {
  92. let musicList = dom.querySelectorAll('musicinfo');
  93. let promises = [];
  94. for (let i = 0; i < musicList.length; ++i) {
  95. let musicInfo = musicList[i];
  96. let xmlPath = base+'/'+musicInfo.getAttribute('xmlpath');
  97. let audioPath = base+'/'+musicInfo.getAttribute('musicpath');
  98. function getData(tag: string): string | null {
  99. let elem = musicInfo.querySelector(tag);
  100. if (elem === null) {
  101. return null;
  102. } else {
  103. return elem.textContent;
  104. }
  105. }
  106. let name = getData('musicname');
  107. let creator = getData('artist');
  108. let genre = getData('genre');
  109. let difficulty = getData('level');
  110. let promise = window.fetch(xmlPath)
  111. .then(parseXML)
  112. .then(parseTMSong)
  113. .then(lines => {
  114. return {
  115. name,
  116. creator,
  117. genre,
  118. difficulty,
  119. audio: audioPath,
  120. lines
  121. }
  122. })
  123. promises.push(promise);
  124. }
  125. return Promise.all(promises)
  126. .then(levels => {
  127. return { name, levels }
  128. })
  129. }
  130. function parseTMSong(dom: Document): Line[] {
  131. let kanjiList = dom.querySelectorAll('nihongoword');
  132. let kanaList = dom.querySelectorAll('word');
  133. let intervalList = dom.querySelectorAll('interval');
  134. let lines: Line[] = [];
  135. let time = 0;
  136. for (let i = 0; i < intervalList.length; ++i) {
  137. let start = time;
  138. time += parseInt(intervalList[i].textContent) / 1000
  139. lines.push({
  140. kanji: kanjiList[i].textContent,
  141. kana: kanaList[i].textContent,
  142. start: start,
  143. end: time
  144. })
  145. }
  146. return lines;
  147. }
  148. }