index.html 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605
  1. <!doctype html>
  2. <html>
  3. <head>
  4. <meta charset="utf-8">
  5. <link rel="stylesheet" href="{{ url_for('static', filename='css/style.css') }}">
  6. <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@4.0.0/dist/css/bootstrap.min.css"
  7. integrity="sha384-Gn5384xqQ1aoWXA+058RXPxPg6fy4IWvTNh0E263XmFcJlSAwiGgFAW/dAiS6JXm" crossorigin="anonymous">
  8. <script src="https://code.jquery.com/jquery-3.2.1.slim.min.js"
  9. integrity="sha384-KJ3o2DKtIkvYIK3UENzmM7KCkRr/rE9/Qpg6aAZGJwFDMVNA/GpGFF93hXpG5KkN"
  10. crossorigin="anonymous"></script>
  11. <script src="https://cdn.jsdelivr.net/npm/popper.js@1.12.9/dist/umd/popper.min.js"
  12. integrity="sha384-ApNbgh9B+Y1QKtv3Rn7W3mgPxhU9K/ScQsAP7hUibX39j7fakFPskvXusvfa0b4Q"
  13. crossorigin="anonymous"></script>
  14. <script src="https://cdn.jsdelivr.net/npm/bootstrap@4.0.0/dist/js/bootstrap.min.js"
  15. integrity="sha384-JZR6Spejh4U02d8jOt6vLEHfe/JQGiRRSQQxSfFWpi1MquVdAyjUar5+76PVCmYl"
  16. crossorigin="anonymous"></script>
  17. <script type="text/javascript" src="{{ url_for('static', filename='js/index.js') }}"></script>
  18. <script type="text/javascript" src="{{ url_for('static', filename='js/jszip.min.js') }}"></script>
  19. <script type="text/javascript" src="{{ url_for('static', filename='js/FileSaver.min.js') }}"></script>
  20. <script type="text/javascript" src="{{ url_for('static', filename='js/jquery-3.6.0.min.js') }}"></script>
  21. <title>StreetForum Design Game</title>
  22. </head>
  23. <body>
  24. <div class="container">
  25. <div class="title">
  26. <h1>
  27. StreetForum Design Game
  28. </h1>
  29. <div class="logout-btn">
  30. <form id="logout" action="/logout">
  31. <input class="btn btn-primary" type="submit" value="Log out"/>
  32. </form>
  33. </div>
  34. </div>
  35. <div class="download">
  36. <div class="input-group">
  37. <div class="input-group-prepend">
  38. <span class="input-group-text">Select Game for Download:</span>
  39. </div>
  40. <select class="custom-select" id="select-game">
  41. {% for id in games %}
  42. <option value="{{ id }}">{{ id }}</option>
  43. {% endfor %}
  44. </select>
  45. <div class="input-group-append">
  46. <button class="btn btn-primary" type="button" onclick="download()">Download (images +
  47. .csv)
  48. <a id="loading-field"><span class="loader"></span></a>
  49. </button>
  50. </div>
  51. </div>
  52. </div>
  53. <div id="table">
  54. <h2> Game Table </h2>
  55. <table class="table {#table-striped#} table-responsive table-hover">
  56. <thead class="table-header">
  57. <tr>
  58. <th>game number</th>
  59. <!--<th>game images</th>-->
  60. <th>number of players</th>
  61. <th>location</th>
  62. <th>closed</th>
  63. <th>player number</th>
  64. <th>claimed</th>
  65. <th>points</th>
  66. <th>goal card</th>
  67. <th>character card</th>
  68. <th>action card</th>
  69. <th>object ID</th>
  70. <th>object name</th>
  71. <th>object points</th>
  72. </tr>
  73. </thead>
  74. <tbody>
  75. {% for row in data %}
  76. <tr>
  77. <td>{{ row.Game.id }}</td>
  78. {#
  79. {% set id = row.Game.id %}
  80. {% if images[id | string] is defined %}
  81. <td>
  82. {% for img in images[id | string] %}
  83. <img class="table-img img-responsive" src="{{ img }}">
  84. {% endfor %}
  85. </td>
  86. {% else %}
  87. <td>
  88. -
  89. </td>
  90. {% endif %}
  91. #}
  92. <td>{{ row.Game.number_of_players }}</td>
  93. <td>{{ row.Game.location }}</td>
  94. <td>{{ row.Game.closed }}</td>
  95. <td>{{ row.Player.player_number }}</td>
  96. <td>{{ row.Player.is_taken }}</td>
  97. <td>{{ row.Player.points }}</td>
  98. <td>{{ row.Player.goal_card_id }}</td>
  99. <td>{{ row.Player.character_card_id }}</td>
  100. <td>
  101. {% for ac in row.Player.drawn_action_cards %}
  102. {{ ac.card_id }},
  103. {% endfor %}
  104. </td>
  105. <td>{{ row.Object.object_type }}</td>
  106. <td>{{ row.Object.object_name }}</td>
  107. <td>{{ row.Object.object_points }}</td>
  108. </tr>
  109. {% endfor %}
  110. </tbody>
  111. </table>
  112. </div>
  113. <div id="table-img">
  114. <h2> Image Table </h2>
  115. <table class="table {#table-striped#} table-responsive table-hover">
  116. <thead class="table-header">
  117. <tr>
  118. <th>game number</th>
  119. <th>game images</th>
  120. </tr>
  121. </thead>
  122. <tbody>
  123. {% for key, row in images.items() %}
  124. <tr>
  125. <td>{{ key }}</td>
  126. <td>
  127. {% for img in row %}
  128. <img class="table-img img-responsive" src="{{ img }}">
  129. {% endfor %}
  130. </td>
  131. </tr>
  132. {% endfor %}
  133. </tbody>
  134. </table>
  135. </div>
  136. <div id="action-card-table">
  137. <div class="row-align"><h2> Action Cards </h2>
  138. <!-- Button trigger modal -->
  139. <button style="margin-left: 10px" type="button" class="btn btn-primary" data-toggle="modal"
  140. data-target="#addActionCard">
  141. Add Card
  142. </button>
  143. </div>
  144. <table class="table {#table-striped#} table-responsive table-hover">
  145. <thead class="table-header">
  146. <tr>
  147. <th>Card ID</th>
  148. <th>Action</th>
  149. <th>Image</th>
  150. </tr>
  151. </thead>
  152. <tbody>
  153. {% for card in action_cards %}
  154. <tr>
  155. <td>{{ card.card_id }}</td>
  156. <td>{{ card.action }}</td>
  157. <td>{% set filename = card.img_path.split('/')[-1] %}
  158. <a target="_blank" rel="noopener noreferrer"
  159. href="/api/get_card_image/{{ filename }}">{{ filename }}</a>
  160. </td>
  161. <td>
  162. <button type="button" class="btn btn-danger"
  163. onclick="delete_card('action', {{ card.card_id }})">Delete
  164. </button>
  165. </td>
  166. </tr>
  167. {% endfor %}
  168. </tbody>
  169. </table>
  170. </div>
  171. <!-- Modal -->
  172. <div class="modal fade" id="addActionCard" tabindex="-1" role="dialog"
  173. aria-labelledby="addActionCardTitle"
  174. aria-hidden="true">
  175. <div class="modal-dialog modal-dialog-centered" role="document">
  176. <div class="modal-content">
  177. <div class="modal-header">
  178. <h5 class="modal-title" id="addActionCardTitle">Add an Action Card</h5>
  179. <button type="button" class="close" data-dismiss="modal" aria-label="Close">
  180. <span aria-hidden="true">&times;</span>
  181. </button>
  182. </div>
  183. <form id="submit-action-card">
  184. <div class="modal-body">
  185. <div class="form-group">
  186. <label for="cardid_action" class="col-form-label">Card ID: </label>
  187. <input type="number" class="form-control" id="cardid_action" required>
  188. <label for="action_action" class="col-form-label">Action: </label>
  189. <input type="text" class="form-control" id="action_action" required>
  190. <label for="img_action" class="col-form-label">Image: </label>
  191. <input type="file" accept="image/png" class="form-control" id="img_action" required>
  192. </div>
  193. <div id="response-msg-action">
  194. </div>
  195. </div>
  196. <div class="modal-footer">
  197. <button type="button" class="btn btn-secondary" data-dismiss="modal">Cancel</button>
  198. <button type="submit" class="btn btn-primary">Submit & Add</button>
  199. </div>
  200. </form>
  201. </div>
  202. </div>
  203. </div>
  204. <div id="char-card-table">
  205. <div class="row-align"><h2> Character Cards </h2>
  206. <button style="margin-left: 10px" type="button" class="btn btn-primary" data-toggle="modal"
  207. data-target="#addCharCard">
  208. Add Card
  209. </button>
  210. </div>
  211. <table class="table {#table-striped#} table-responsive table-hover">
  212. <thead class="table-header">
  213. <tr>
  214. <th>Card ID</th>
  215. <th>Name</th>
  216. <th>Age</th>
  217. <th>Role</th>
  218. <th>Interest</th>
  219. <th>Quote</th>
  220. <th>Image</th>
  221. </tr>
  222. </thead>
  223. <tbody>
  224. {% for card in char_cards %}
  225. <tr>
  226. <td>{{ card.card_id }}</td>
  227. <td>{{ card.name }}</td>
  228. <td>{{ card.age }}</td>
  229. <td>{{ card.role }}</td>
  230. <td>{{ card.interest }}</td>
  231. <td>{{ card.quote }}</td>
  232. <td>{% set filename = card.img_path.split('/')[-1] %}
  233. <a target="_blank" rel="noopener noreferrer"
  234. href="/api/get_card_image/{{ filename }}">{{ filename }}</a>
  235. </td>
  236. <td>
  237. <button type="button" class="btn btn-danger" onclick="delete_card('char', {{ card.card_id }})">
  238. Delete
  239. </button>
  240. </td>
  241. </tr>
  242. {% endfor %}
  243. </tbody>
  244. </table>
  245. </div>
  246. <!-- Modal -->
  247. <div class="modal fade" id="addCharCard" tabindex="-1" role="dialog"
  248. aria-labelledby="addCharCardTitle"
  249. aria-hidden="true">
  250. <div class="modal-dialog modal-dialog-centered" role="document">
  251. <div class="modal-content">
  252. <div class="modal-header">
  253. <h5 class="modal-title" id="addCharCardTitle">Add a Character Card</h5>
  254. <button type="button" class="close" data-dismiss="modal" aria-label="Close">
  255. <span aria-hidden="true">&times;</span>
  256. </button>
  257. </div>
  258. <form id="submit-char-card">
  259. <div class="modal-body">
  260. <div class="form-group">
  261. <label for="cardid_char" class="col-form-label">Card ID: </label>
  262. <input type="number" class="form-control" id="cardid_char" required>
  263. <label for="name_char" class="col-form-label">Name: </label>
  264. <input type="text" class="form-control" id="name_char" required>
  265. <label for="age_char" class="col-form-label">Age: </label>
  266. <input type="number" accept="image/png" class="form-control" id="age_char" required>
  267. <label for="role_char" class="col-form-label">Role: </label>
  268. <input type="text" class="form-control" id="role_char" required>
  269. <label for="interest_char" class="col-form-label">Interest: </label>
  270. <input type="text" class="form-control" id="interest_char" required>
  271. <label for="quote_char" class="col-form-label">Quote: </label>
  272. <input type="text" class="form-control" id="quote_char" required>
  273. <label for="img_char" class="col-form-label">Image: </label>
  274. <input type="file" accept="image/png" class="form-control" id="img_char" required>
  275. </div>
  276. <div id="response-msg-char">
  277. </div>
  278. </div>
  279. <div class="modal-footer">
  280. <button type="button" class="btn btn-secondary" data-dismiss="modal">Cancel</button>
  281. <button type="submit" class="btn btn-primary">Submit & Add</button>
  282. </div>
  283. </form>
  284. </div>
  285. </div>
  286. </div>
  287. <div id="goal-card-table">
  288. <div class="row-align"><h2> Goal Cards </h2>
  289. <button style="margin-left: 10px" type="button" class="btn btn-primary" data-toggle="modal"
  290. data-target="#addGoalCard">
  291. Add Card
  292. </button>
  293. </div>
  294. <table class="table {#table-striped#} table-responsive table-hover">
  295. <thead class="table-header">
  296. <tr>
  297. <th>Card ID</th>
  298. <th>Goal</th>
  299. <th>Image</th>
  300. </tr>
  301. </thead>
  302. <tbody>
  303. {% for card in goal_cards %}
  304. <tr>
  305. <td>{{ card.card_id }}</td>
  306. <td>{{ card.goal }}</td>
  307. <td>{% set filename = card.img_path.split('/')[-1] %}
  308. <a target="_blank" rel="noopener noreferrer"
  309. href="/api/get_card_image/{{ filename }}">{{ filename }}</a>
  310. </td>
  311. <td>
  312. <button type="button" class="btn btn-danger" onclick="delete_card('goal', {{ card.card_id }})">
  313. Delete
  314. </button>
  315. </td>
  316. </tr>
  317. {% endfor %}
  318. </tbody>
  319. </table>
  320. </div>
  321. <!-- Modal -->
  322. <div class="modal fade" id="addGoalCard" tabindex="-1" role="dialog"
  323. aria-labelledby="addGoalCardTitle"
  324. aria-hidden="true">
  325. <div class="modal-dialog modal-dialog-centered" role="document">
  326. <div class="modal-content">
  327. <div class="modal-header">
  328. <h5 class="modal-title" id="addGoalCardTitle">Add a Goal Card</h5>
  329. <button type="button" class="close" data-dismiss="modal" aria-label="Close">
  330. <span aria-hidden="true">&times;</span>
  331. </button>
  332. </div>
  333. <form id="submit-goal-card">
  334. <div class="modal-body">
  335. <div class="form-group">
  336. <label for="cardid_goal" class="col-form-label">Card ID: </label>
  337. <input type="number" class="form-control" id="cardid_goal" required>
  338. <label for="goal_goal" class="col-form-label">Goal: </label>
  339. <input type="text" class="form-control" id="goal_goal" required>
  340. <label for="img_goal" class="col-form-label">Image: </label>
  341. <input type="file" accept="image/png" class="form-control" id="img_goal" required>
  342. </div>
  343. <div id="response-msg-goal">
  344. </div>
  345. </div>
  346. <div class="modal-footer">
  347. <button type="button" class="btn btn-secondary" data-dismiss="modal">Cancel</button>
  348. <button type="submit" class="btn btn-primary">Submit & Add</button>
  349. </div>
  350. </form>
  351. </div>
  352. </div>
  353. </div>
  354. <!-- Button trigger modal -->
  355. <button type="button" class="btn btn-danger" data-toggle="modal" data-target="#exampleModalCenter">
  356. Reset Database
  357. </button>
  358. <!-- Modal -->
  359. <div class="modal fade" id="exampleModalCenter" tabindex="-1" role="dialog"
  360. aria-labelledby="exampleModalCenterTitle"
  361. aria-hidden="true">
  362. <div class="modal-dialog modal-dialog-centered" role="document">
  363. <div class="modal-content">
  364. <div class="modal-header">
  365. <h5 class="modal-title" id="exampleModalLongTitle">Are you Sure you want to reset the Database?</h5>
  366. <button type="button" class="close" data-dismiss="modal" aria-label="Close">
  367. <span aria-hidden="true">&times;</span>
  368. </button>
  369. </div>
  370. <form id="confirm-pass">
  371. <div class="modal-body">
  372. <p>WARNING: All games and images will be permanently deleted!</p>
  373. <p>Retype your password to continue.</p>
  374. <div class="form-group">
  375. <label for="pass" class="col-form-label">Password:</label>
  376. <input type="password" class="form-control" id="pass" required>
  377. </div>
  378. <div id="response-msg">
  379. </div>
  380. </div>
  381. <div class="modal-footer">
  382. <button type="button" class="btn btn-secondary" data-dismiss="modal">Cancel</button>
  383. <button type="submit" class="btn btn-primary">Reset Database and Reload Page</button>
  384. </div>
  385. </form>
  386. </div>
  387. </div>
  388. </div>
  389. </div>
  390. <script type="text/javascript">
  391. // Handle action card submission
  392. document.getElementById('submit-action-card').addEventListener('submit', function (event) {
  393. event.preventDefault(); // Prevent default form submission
  394. // Create a new FormData object
  395. const formData = new FormData();
  396. // Retrieve the values from the inputs
  397. const cardid = document.getElementById('cardid_action').value;
  398. const action = document.getElementById('action_action').value;
  399. const image = document.getElementById('img_action').files[0];
  400. // Add the values to the FormData object
  401. formData.append('cardid', cardid);
  402. formData.append('action', action);
  403. formData.append('img', image);
  404. // Perform AJAX POST request
  405. $.ajax({
  406. type: "POST",
  407. url: "/api/add_action_card",
  408. data: formData,
  409. processData: false,
  410. contentType: false,
  411. success: function (result) {
  412. console.log(result)
  413. location.reload()
  414. },
  415. error: function (result, status) {
  416. console.log(result)
  417. document.getElementById('response-msg-action').innerText = result.responseText
  418. }
  419. });
  420. });
  421. // Handle char card submission
  422. document.getElementById('submit-char-card').addEventListener('submit', function (event) {
  423. event.preventDefault(); // Prevent default form submission
  424. // Create a new FormData object
  425. const formData = new FormData();
  426. // Retrieve the values from the inputs
  427. const cardid = document.getElementById('cardid_char').value;
  428. const name = document.getElementById('name_char').value;
  429. const age = document.getElementById('age_char').value;
  430. const role = document.getElementById('role_char').value;
  431. const interest = document.getElementById('interest_char').value;
  432. const quote = document.getElementById('quote_char').value;
  433. const image = document.getElementById('img_char').files[0];
  434. // Add the values to the FormData object
  435. formData.append('cardid', cardid);
  436. formData.append('name', name);
  437. formData.append('age', age);
  438. formData.append('role', role);
  439. formData.append('interest', interest);
  440. formData.append('quote', quote);
  441. formData.append('img', image);
  442. // Perform AJAX POST request
  443. $.ajax({
  444. type: "POST",
  445. url: "/api/add_char_card",
  446. data: formData,
  447. processData: false,
  448. contentType: false,
  449. success: function (result) {
  450. console.log(result)
  451. location.reload()
  452. },
  453. error: function (result, status) {
  454. console.log(result)
  455. document.getElementById('response-msg-char').innerText = result.responseText
  456. }
  457. });
  458. });
  459. // Handle goal card submission
  460. document.getElementById('submit-goal-card').addEventListener('submit', function (event) {
  461. event.preventDefault(); // Prevent default form submission
  462. // Create a new FormData object
  463. const formData = new FormData();
  464. // Retrieve the values from the inputs
  465. const cardid = document.getElementById('cardid_goal').value;
  466. const goal = document.getElementById('goal_goal').value;
  467. const image = document.getElementById('img_goal').files[0];
  468. // Add the values to the FormData object
  469. formData.append('cardid', cardid);
  470. formData.append('goal', goal);
  471. formData.append('img', image);
  472. // Perform AJAX POST request
  473. $.ajax({
  474. type: "POST",
  475. url: "/api/add_goal_card",
  476. data: formData,
  477. processData: false,
  478. contentType: false,
  479. success: function (result) {
  480. console.log(result)
  481. location.reload()
  482. },
  483. error: function (result, status) {
  484. console.log(result)
  485. document.getElementById('response-msg-goal').innerText = result.responseText
  486. }
  487. });
  488. });
  489. $("#confirm-pass").submit(function (event) {
  490. event.preventDefault(); //stop a full postback
  491. let password = $("#pass").val(); //get the entered value from the password box
  492. $.ajax({
  493. type: "POST",
  494. url: "/api/reset_database",
  495. data: JSON.stringify({"password": password}),
  496. contentType: "application/json",
  497. success: function (result) {
  498. location.reload()
  499. },
  500. error: function (result, status) {
  501. document.getElementById('response-msg').innerText = result.responseText
  502. }
  503. });
  504. });
  505. function delete_card(type, id) {
  506. let url;
  507. switch (type) {
  508. case "action":
  509. url = "/api/delete_action_card";
  510. break;
  511. case "char":
  512. url = "/api/delete_char_card";
  513. break;
  514. case "goal":
  515. url = "/api/delete_goal_card";
  516. break;
  517. }
  518. let fd = new FormData();
  519. fd.append('cardid', id);
  520. $.ajax({
  521. type: "POST",
  522. url: url,
  523. data: fd,
  524. processData: false,
  525. contentType: false,
  526. success: function (result) {
  527. location.reload()
  528. },
  529. error: function (result, status) {
  530. document.getElementById('response-msg').innerText = result.responseText
  531. }
  532. });
  533. }
  534. </script>
  535. </body>
  536. </html>