index.html 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582
  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <title>Your Electron App</title>
  5. <style>
  6. #nav {
  7. /* 导航栏样式 */
  8. }
  9. #content {
  10. /* 内容区样式 */
  11. }
  12. #level-list {
  13. width: 200px;
  14. float: left;
  15. }
  16. #level-details {
  17. margin-left: 210px;
  18. }
  19. </style>
  20. </head>
  21. <body>
  22. <div id="hdr" style="background-color: antiquewhite;">
  23. <input type="text" id="group-suffix" placeholder="分组文件的后缀">
  24. <button id="load_grp">加载分组信息</button>
  25. </div>
  26. <div id="panels-area" style="display: flex; height: 95vh;">
  27. <div id="nav" style="overflow-y: scroll; height: 90%; width: 5%; background-color: coral;">
  28. <ul>
  29. <li>
  30. <button id="levels">关卡</button>
  31. </li>
  32. <li>
  33. <button id="examples">实例</button>
  34. </li>
  35. <li>
  36. <button id="templates">模板</button>
  37. </li>
  38. </ul>
  39. </div>
  40. <div id="content" style="overflow-y: scroll; height: 90%; width: 85%; background-color: beige;">
  41. <!-- 动态内容 -->
  42. </div>
  43. </div>
  44. <script>
  45. var levels = null;
  46. var instances = null;
  47. var templateFNs = null;
  48. var templateFNsByDir = null;
  49. function getInstanceByName(name) {
  50. for (let i = 0; i < instances.length; i++) {
  51. if (instances[i].lvinst == name) {
  52. return instances[i];
  53. }
  54. }
  55. return null;
  56. }
  57. function getTemplateByName(name) {
  58. for (let i = 0; i < templateFNs.length; i++) {
  59. if (templateFNs[i] == name) {
  60. return templateFNs[i];
  61. }
  62. }
  63. return null;
  64. }
  65. function getAllLvIdsByInstName(name) {
  66. // 返回所有用到了该实例的关卡的id
  67. const ids = [];
  68. for (let i = 0; i < levels.length; i++) {
  69. if (levels[i].lvinst == name) {
  70. ids.push(levels[i].lvid);
  71. }
  72. }
  73. return ids;
  74. }
  75. function saveInst(div, inst) {
  76. // 从 div中获取实例信息
  77. const instName = div.querySelector('#instid-in-inst-detail').textContent.split(':')[1].trim();
  78. var inst = null;
  79. for (let i = 0; i < instances.length; i++) {
  80. if (instances[i].lvinst == instName) {
  81. inst = instances[i];
  82. break;
  83. }
  84. }
  85. if (inst == null) {
  86. alert('实例不存在');
  87. return;
  88. }
  89. inst['template'] = div.querySelector('#template-select').value;
  90. for (let i = 1; i <= 10; i++) {
  91. const tileCnt = div.querySelector('#try' + i + '_tileCnt').value;
  92. const seed = div.querySelector('#try' + i + '_seed').value;
  93. const param_template = div.querySelector('#try' + i + '_param_template').value;
  94. const trynParams = tileCnt + '|' + seed + '|' + param_template;
  95. inst['try' + i + '(tileCnt|seed|param_template)'] = trynParams;
  96. }
  97. window.electron.send('save2csv', '../../conf/levelInstInfo.csv', instances);
  98. }
  99. function showInstanceInfoAt(div, inst) {
  100. // 将其显示在页面上,其中template应该是一个下拉框
  101. // try1-10应该是一个表格, 表格的头部应该是:尝试次数,瓦片数,种子,参数模板,表格里面的每个值是可以编辑的
  102. div.innerHTML = `
  103. <h2 id='instid-in-inst-detail'>实例名: ${inst.lvinst}</h2>
  104. `;
  105. // 显示所有用到了该实例的关卡
  106. const lvIds = getAllLvIdsByInstName(inst.lvinst);
  107. div.innerHTML += `
  108. <p style="display: inline;">用到了该实例的关卡: ${lvIds}</p>
  109. `;
  110. // 加入 template 下拉框
  111. {
  112. var div1 = document.createElement('div');
  113. const p = document.createElement('p');
  114. p.style.display = 'inline';
  115. p.appendChild(document.createTextNode('模板名称: '));
  116. div1.appendChild(p);
  117. const templateSelect = document.createElement('select');
  118. templateSelect.id = 'template-select';
  119. templateFNs.forEach(fn => {
  120. const option = document.createElement('option');
  121. option.value = fn;
  122. option.textContent = fn;
  123. if (fn == inst.template) {
  124. option.selected = true;
  125. }
  126. templateSelect.appendChild(option);
  127. });
  128. div1.appendChild(templateSelect);
  129. // 增加一个编辑模板的按钮
  130. const editTemplateButton = document.createElement('button');
  131. editTemplateButton.id = 'edit-template-button';
  132. editTemplateButton.textContent = '编辑模板';
  133. editTemplateButton.addEventListener('click', () => {
  134. // 判断模板是在哪个目录下
  135. for (let dir in templateFNsByDir) {
  136. if (templateFNsByDir[dir].has(inst.template)) {
  137. window.electron.send('open-app', '/Applications/Tiled.app/Contents/MacOS/Tiled ' + dir + '/' + inst.template + '.json');
  138. return;
  139. }
  140. }
  141. });
  142. div1.appendChild(editTemplateButton);
  143. div.appendChild(div1);
  144. }
  145. // 加入表格
  146. const table = document.createElement('table');
  147. const thead = document.createElement('thead');
  148. const tbody = document.createElement('tbody');
  149. const tr = document.createElement('tr');
  150. const th1 = document.createElement('th');
  151. const th2 = document.createElement('th');
  152. const th3 = document.createElement('th');
  153. const th4 = document.createElement('th');
  154. th1.textContent = '尝试次数';
  155. th2.textContent = '瓦片数';
  156. th3.textContent = '种子';
  157. th4.textContent = '参数模板';
  158. tr.appendChild(th1);
  159. tr.appendChild(th2);
  160. tr.appendChild(th3);
  161. tr.appendChild(th4);
  162. thead.appendChild(tr);
  163. table.appendChild(thead);
  164. table.appendChild(tbody);
  165. div.appendChild(table);
  166. // 加入表格内容
  167. for (let i = 1; i <= 10; i++) {
  168. var trynParams = inst['try' + i + '(tileCnt|seed|param_template)'].split('|');
  169. const tr = document.createElement('tr');
  170. const td1 = document.createElement('td');
  171. const td2 = document.createElement('td');
  172. const td3 = document.createElement('td');
  173. const td4 = document.createElement('td');
  174. const input1 = document.createElement('input');
  175. const input2 = document.createElement('input');
  176. const input3 = document.createElement('input');
  177. const input4 = document.createElement('input');
  178. input1.type = 'text';
  179. input1.value = trynParams[0];
  180. input1.id = 'try' + i + '_tileCnt';
  181. input2.type = 'text';
  182. input2.value = trynParams[1];
  183. input2.id = 'try' + i + '_seed';
  184. input3.type = 'text';
  185. input3.value = trynParams[2];
  186. input3.id = 'try' + i + '_param_template';
  187. td1.textContent = i;
  188. td2.appendChild(input1);
  189. td3.appendChild(input2);
  190. td4.appendChild(input3);
  191. tr.appendChild(td1);
  192. tr.appendChild(td2);
  193. tr.appendChild(td3);
  194. tr.appendChild(td4);
  195. tbody.appendChild(tr);
  196. }
  197. // 增加一个按钮,点击后可以保存
  198. const saveButton = document.createElement('button');
  199. saveButton.id = 'save-instance-button';
  200. saveButton.textContent = '保存';
  201. div.appendChild(saveButton);
  202. saveButton.addEventListener('click', () => {
  203. // 保存实例信息
  204. saveInst(div, inst);
  205. });
  206. }
  207. function showTemplateInfoAt(div, templateInfo) {
  208. }
  209. // 过滤实例列表
  210. function filterInstance() {
  211. // 获取输入框的值
  212. var filter = document.getElementById('instance-filter').value;
  213. // 获取实例列表
  214. var list = document.getElementById('instance-list').getElementsByTagName('li');
  215. // 遍历实例列表
  216. for (var i = 0; i < list.length; i++) {
  217. // 获取实例的名字
  218. var name = list[i].innerHTML;
  219. // 如果实例的名字包含输入框的值,显示实例
  220. if (name.indexOf(filter) > -1) {
  221. list[i].style.display = '';
  222. } else {
  223. // 否则隐藏实例
  224. list[i].style.display = 'none';
  225. }
  226. }
  227. }
  228. function initData(suffix) {
  229. // 读取关卡信息并填充到页面
  230. if (levels == null) {
  231. window.electron.send('parse-csv', '../../conf/levelInfo'+suffix+'.csv')
  232. window.electron.receive('csv-data', (data) => {
  233. if (levels == null) {
  234. levels = data;
  235. window.electron.send('parse-csv', '../../conf/levelInstInfo'+suffix+'.csv')
  236. } else {
  237. if (instances == null) {
  238. instances = data
  239. }
  240. }
  241. });
  242. }
  243. if (templateFNs == null) {
  244. templateFNs = [];
  245. templateFNsByDir = {};
  246. window.electron.send('get-file-list', '../../tf_templates;../../miniGame')
  247. window.electron.receive('file-list-error', (data) => {
  248. console.error('Error reading directory: ', err);
  249. });
  250. window.electron.receive('file-list-data', (data, path) => {
  251. templateFNsByDir[path] = new Set();
  252. // 过滤掉非 .json 后缀的文件
  253. for (let i = 0; i < data.length; i++) {
  254. if (data[i].endsWith('.json')) {
  255. // 去掉后缀
  256. const name = data[i].substring(0, data[i].length - 5);
  257. templateFNs.push(name);
  258. templateFNsByDir[path].add(name);
  259. }
  260. }
  261. });
  262. }
  263. //
  264. }
  265. // 创建一个新实例,并将其关联到指定的关卡中
  266. function addNewInst(level) {
  267. const instDetails = document.getElementById('instance-details');
  268. // 获取实例名称
  269. window.electron.send('open-newinst-prompt', 'some data');
  270. }
  271. function saveLvInfo() {
  272. const lvDetails = document.getElementById('level-details');
  273. // 保存关卡的详细信息
  274. const lvid = lvDetails.querySelector('#lvid-in-lvdetails').textContent.split(':')[1].trim();
  275. const instid = lvDetails.querySelector('#inst-select').value;
  276. const type = lvDetails.querySelector('#type-select-in-lvdetails').value;
  277. const ddaType = lvDetails.querySelector('#ddatype-in-lvdetails').textContent.split(':')[1].trim();
  278. const ddaPara = lvDetails.querySelector('#ddapara-in-lvdetails').textContent.split(':')[1].trim();
  279. var lv = null;
  280. for (let i = 0; i < levels.length; i++) {
  281. if (levels[i].lvid == lvid) {
  282. lv = levels[i];
  283. break;
  284. }
  285. }
  286. lv['lvinst'] = instid;
  287. lv['type(normal=0,hard=1,superHard=2)'] = type;
  288. lv['dda_type'] = ddaType;
  289. lv['dda_para'] = ddaPara;
  290. // 保存到文件
  291. window.electron.send('save2csv', '../../conf/levelInfo.csv', levels);
  292. }
  293. function fillLvInfo() {
  294. // 读取关卡信息并填充到页面
  295. if (levels != null) {
  296. const levelList = document.getElementById('level-list');
  297. const levelDetails = document.getElementById('level-details');
  298. const instDetails = document.getElementById('instance-details');
  299. const templateDetails = document.getElementById('tempalte-details');
  300. // 先清空
  301. levelList.innerHTML = '';
  302. // 将levels里面的信息填充到页面
  303. levels.forEach(level => {
  304. const listItem = document.createElement('li');
  305. listItem.textContent = level.lvid;
  306. listItem.addEventListener('click', () => {
  307. // 将选中的关卡id条目的背景颜色设置为蓝色
  308. Array.from(levelList.children).forEach(node => {
  309. if (node.textContent == level.lvid) {
  310. node.style.backgroundColor = 'lightblue';
  311. } else {
  312. node.style.backgroundColor = 'white';
  313. }
  314. });
  315. levelDetails.innerHTML = `
  316. <h2 id='lvid-in-lvdetails'>关卡ID: ${level.lvid}</h2>
  317. <p id='instid-in-lvdetails'>关卡实例名称: ${level.lvinst}</p>`
  318. //
  319. if (level.lvinst == '') {
  320. // 实例不存在,
  321. // 显示一个下来列表,选择实例
  322. const instSelect = document.createElement('select');
  323. instSelect.id = 'inst-select';
  324. instances.forEach(inst => {
  325. const option = document.createElement('option');
  326. option.value = inst.lvinst;
  327. option.textContent = inst.lvinst;
  328. instSelect.appendChild(option);
  329. });
  330. levelDetails.appendChild(instSelect);
  331. }
  332. // 加入一个按钮,点击后可以创建实例
  333. {
  334. const div = document.createElement('div');
  335. const createInstButton = document.createElement('button');
  336. createInstButton.id = 'create-instance-button';
  337. createInstButton.textContent = '创建新的实例';
  338. div.appendChild(createInstButton);
  339. levelDetails.appendChild(div);
  340. // createInstButton.addEventListener('click', () => {
  341. // addNewInst(level);
  342. // });
  343. console.log(createInstButton.onclick);
  344. }
  345. {
  346. const div = document.createElement('div');
  347. // 增加一个下拉框,选择类型:normal,hard,super-hard
  348. const type = level['type(normal=0,hard=1,superHard=2)'];
  349. const typeSelect = document.createElement('select');
  350. typeSelect.id = 'type-select-in-lvdetails';
  351. const normalOption = document.createElement('option');
  352. normalOption.value = '0';
  353. normalOption.textContent = 'normal';
  354. normalOption.selected = type == '0';
  355. const hardOption = document.createElement('option');
  356. hardOption.value = '1';
  357. hardOption.textContent = 'hard';
  358. hardOption.selected = type == '1';
  359. const superHardOption = document.createElement('option');
  360. superHardOption.value = '2';
  361. superHardOption.textContent = 'super-hard';
  362. superHardOption.selected = type == '2';
  363. typeSelect.appendChild(normalOption);
  364. typeSelect.appendChild(hardOption);
  365. typeSelect.appendChild(superHardOption);
  366. const p = document.createElement('p');
  367. p.style.display = 'inline';
  368. p.appendChild(document.createTextNode('关卡难度级别:'));
  369. div.appendChild(p);
  370. div.appendChild(typeSelect);
  371. levelDetails.appendChild(div);
  372. }
  373. // 显示dda类型和参数
  374. const ddatype = document.createElement('p');
  375. ddatype.id = 'ddatype-in-lvdetails';
  376. ddatype.textContent = 'DDA类型: ' + level.dda_type;
  377. levelDetails.appendChild(ddatype);
  378. const ddapara = document.createElement('p');
  379. ddapara.id = 'ddapara-in-lvdetails';
  380. ddapara.textContent = 'DDA参数: ' + level.dda_para;
  381. levelDetails.appendChild(ddapara);
  382. // 增加一个按钮,点击后可以保存
  383. const saveButton = document.createElement('button');
  384. saveButton.id = 'save-lv-button';
  385. saveButton.textContent = '保存';
  386. levelDetails.appendChild(saveButton);
  387. levelDetails.addEventListener('click', (event) => {
  388. if (event.target.id === 'create-instance-button') {
  389. addNewInst(level);
  390. } else if (event.target.id === 'save-lv-button') {
  391. saveLvInfo();
  392. }
  393. });
  394. // 显示实例信息
  395. const inst = getInstanceByName(level.lvinst);
  396. if (inst != null) {
  397. showInstanceInfoAt(instDetails, inst);
  398. // 显示模板信息
  399. const template = getTemplateByName(inst.template);
  400. if (template != null) {
  401. showTemplateInfoAt(templateDetails, getTemplateByName(inst.template));
  402. }
  403. } else {
  404. instDetails.innerHTML = `
  405. <h2>实例名: Error:未定义</h2>
  406. `
  407. }
  408. });
  409. levelList.appendChild(listItem);
  410. });
  411. }
  412. }
  413. function fillInstanceInfo() {
  414. // 读取关卡信息并填充到页面
  415. if (instances != null) {
  416. const lst = document.getElementById('instance-list');
  417. const details = document.getElementById('instance-details');
  418. // 将levels里面的信息填充到页面
  419. instances.forEach(inst => {
  420. const listItem = document.createElement('li');
  421. listItem.textContent = inst.lvinst;
  422. listItem.addEventListener('click', () => {
  423. showInstanceInfoAt(details, inst)
  424. });
  425. lst.appendChild(listItem);
  426. });
  427. }
  428. }
  429. function fillTemplateInfo() {
  430. if (templateFNs != null) {
  431. const lst = document.getElementById('template-list');
  432. const details = document.getElementById('template-details');
  433. // 将levels里面的信息填充到页面
  434. templateFNs.forEach(fn => {
  435. const listItem = document.createElement('li');
  436. listItem.textContent = fn;
  437. listItem.addEventListener('click', () => {
  438. // 放入截图
  439. });
  440. lst.appendChild(listItem);
  441. });
  442. }
  443. }
  444. // 处理新增关卡的回复
  445. window.electron.receive('prompt-newlv-reply', (event, arg) => {
  446. const lvid = event;
  447. // 检查是否已经存在
  448. for (let i = 0; i < levels.length; i++) {
  449. if (levels[i].lvid == lvid) {
  450. alert('关卡id已经存在');
  451. return;
  452. }
  453. }
  454. // 不存在则加入
  455. const level = {
  456. lvid: lvid,
  457. lvinst: '',
  458. type: 0,
  459. dda_type: '0',
  460. dda_para: '0|10000'
  461. };
  462. levels.push(level);
  463. // 刷新页面
  464. fillLvInfo();
  465. // 选中该关卡
  466. const levelList = document.getElementById('level-list');
  467. levelList.childNodes.forEach(node => {
  468. if (node.textContent == lvid) {
  469. node.click();
  470. }
  471. });
  472. })
  473. // 处理新增实例的回复
  474. window.electron.receive('prompt-newinst-reply', (event, arg) => {
  475. const instName = event;
  476. // 检查是否已经存在
  477. for (let i = 0; i < instances.length; i++) {
  478. if (instances[i].lvinst == instName) {
  479. alert('实例已经存在');
  480. return;
  481. }
  482. }
  483. // 不存在则加入
  484. const inst = {
  485. lvinst: instName,
  486. template: '',
  487. 'try1(tileCnt|seed|param_template)': '0|0|0',
  488. 'try2(tileCnt|seed|param_template)': '0|0|0',
  489. 'try3(tileCnt|seed|param_template)': '0|0|0',
  490. 'try4(tileCnt|seed|param_template)': '0|0|0',
  491. 'try5(tileCnt|seed|param_template)': '0|0|0',
  492. 'try6(tileCnt|seed|param_template)': '0|0|0',
  493. 'try7(tileCnt|seed|param_template)': '0|0|0',
  494. 'try8(tileCnt|seed|param_template)': '0|0|0',
  495. 'try9(tileCnt|seed|param_template)': '0|0|0',
  496. 'try10(tileCnt|seed|param_template)': '0|0|0'
  497. };
  498. instances.push(inst);
  499. showInstanceInfoAt(instDetails, inst);
  500. });
  501. document.getElementById('load_grp').addEventListener('click', function() {
  502. const suffix = document.getElementById('group-suffix').value;
  503. // 初始化数据
  504. if (suffix == '') {
  505. initData("");
  506. } else {
  507. initData("-"+suffix);
  508. }
  509. });
  510. document.getElementById('levels').addEventListener('click', function() {
  511. fetch('level.html')
  512. .then(response => response.text())
  513. .then(data => {
  514. document.getElementById('content').innerHTML = data;
  515. fillLvInfo();
  516. // 绑定新增关卡按钮
  517. document.getElementById('add-level-button').addEventListener('click', function() {
  518. window.electron.send('open-newlv-prompt', 'some data')
  519. });
  520. })
  521. .catch(error => {
  522. console.error('Error loading the levels content:', error);
  523. });
  524. });
  525. document.getElementById('examples').addEventListener('click', function() {
  526. fetch('instance.html')
  527. .then(response => response.text())
  528. .then(data => {
  529. document.getElementById('content').innerHTML = data;
  530. fillInstanceInfo();
  531. // 给过滤按钮添加事件,点击时调用filterInstance函数
  532. document.getElementById('intance-btn-filter').addEventListener('click', filterInstance);
  533. })
  534. .catch(error => {
  535. console.error('Error loading the intance content:', error);
  536. });
  537. });
  538. document.getElementById('templates').addEventListener('click', function() {
  539. fetch('template.html')
  540. .then(response => response.text())
  541. .then(data => {
  542. document.getElementById('content').innerHTML = data;
  543. fillTemplateInfo();
  544. })
  545. .catch(error => {
  546. console.error('Error loading the intance content:', error);
  547. });
  548. });
  549. </script>
  550. </body>
  551. </html>