A pixel art painter geared specifically at NES pixel art. Includes export for .chr binary file as well as palette and namespace data.
No puede seleccionar más de 25 temas Los temas deben comenzar con una letra o número, pueden incluir guiones ('-') y pueden tener hasta 35 caracteres de largo.

CTRLNametableTools.js 5.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. import Utils from "/app/js/common/Utils.js";
  2. import GlobalEvents from "/app/js/common/EventCaller.js";
  3. import Renderer from "/app/js/ui/Renderer.js";
  4. import NESNameTable from "/app/js/models/NESNameTable.js";
  5. import CTRLBanksStore from "/app/js/ctrls/CTRLBanksStore.js";
  6. const TILE_SELECT_CLS = "canvas-item-selected";
  7. var ELCtrl = null;
  8. var SURF = null;
  9. var TileIndex = -1;
  10. function UpdateBankList(curbankname){
  11. var elsel = ELCtrl.querySelector(".nametable-bank-select");
  12. if (elsel){
  13. var child = elsel.firstChild;
  14. // Clear old bank names...
  15. while(child !== null){
  16. let nchild = child.nextSibling;
  17. let drop = true;
  18. if (Utils.isElement(child)){
  19. if (child.hasAttribute("value")){
  20. if (child.getAttribute("value") === "NULL_BANK"){
  21. drop = false;
  22. }
  23. }
  24. }
  25. if (drop){
  26. elsel.removeChild(child);
  27. }
  28. child = nchild;
  29. }
  30. // Get the only remaining option...
  31. var elop = elsel.querySelector("option");
  32. // Get the current list of bank names...
  33. var banknames = CTRLBanksStore.keys;
  34. // Loop through bank names, if there are any, and add new options to the list...
  35. if (elop && banknames.length > 0){
  36. banknames.forEach((name) => {
  37. var newop = elop.cloneNode(true);
  38. newop.setAttribute("value", name);
  39. newop.innerHTML = name;
  40. elsel.appendChild(newop);
  41. });
  42. elsel.value = curbankname;
  43. }
  44. }
  45. }
  46. function UpdateBankTileList(){
  47. var el = ELCtrl.querySelector(".nametable-tile");
  48. if (el){
  49. if (SURF.bank === null){
  50. el.classList.add("hidden");
  51. } else {
  52. var elsel = ELCtrl.querySelector(".nametable-tile-select");
  53. if (elsel){
  54. el.classList.remove("hidden");
  55. let tiles = SURF.bank.rp;
  56. for (let i=0; i < tiles.length; i++){
  57. let cnv = elsel.querySelector('canvas[value="' + i + '"]');
  58. if (cnv){
  59. var psize = Math.floor(cnv.parentNode.clientWidth * 0.9);
  60. if (cnv.clientWidth !== psize){
  61. cnv.width = psize;
  62. cnv.height = psize;
  63. }
  64. let ctx = cnv.getContext("2d");
  65. let tsurf = new Renderer.NESTileSurface(tiles[i], SURF.palette, 0);
  66. Renderer.renderToFit(tsurf, ctx);
  67. }
  68. }
  69. }
  70. }
  71. }
  72. }
  73. function OpenControls(){
  74. if (ELCtrl !== null && SURF !== null){
  75. var curbankname = (SURF.bank !== null) ? CTRLBanksStore.getBankName(SURF.bank) : "NULL_BANK";
  76. if (curbankname === null){
  77. SURF.bank = null;
  78. curbankname = "NULL_BANK";
  79. }
  80. ELCtrl.classList.remove("hidden");
  81. UpdateBankList(curbankname);
  82. UpdateBankTileList();
  83. }
  84. }
  85. function CloseControls(){
  86. if (ELCtrl !== null){
  87. ELCtrl.classList.add("hidden");
  88. }
  89. }
  90. function HANDLE_PaintNametable(x, y){
  91. if (TileIndex >= 0 && TileIndex < 256){
  92. SURF.setTileIndex(x, y, TileIndex);
  93. }
  94. }
  95. function ResetSelectedTile(){
  96. var eltilesel = ELCtrl.querySelector(".nametable-tile-select");
  97. if (eltilesel){
  98. var op = eltilesel.querySelector('canvas[selected="True"]');
  99. if (op){
  100. op.classList.remove(TILE_SELECT_CLS);
  101. op.removeAttribute("selected");
  102. }
  103. }
  104. TileIndex = -1;
  105. }
  106. function HANDLE_SurfChange(surf){
  107. if (surf instanceof NESNameTable){
  108. if (SURF !== null)
  109. SURF.unlisten("paint_nametable", HANDLE_PaintNametable);
  110. SURF = surf;
  111. SURF.listen("paint_nametable", HANDLE_PaintNametable);
  112. ResetSelectedTile();
  113. OpenControls();
  114. } else {
  115. if (SURF !== null)
  116. SURF.unlisten("paint_nametable", HANDLE_PaintNametable);
  117. SURF = null;
  118. CloseControls();
  119. }
  120. }
  121. class CTRLNametableTools{
  122. constructor(){
  123. GlobalEvents.listen("change_surface", HANDLE_SurfChange);
  124. }
  125. initialize(){
  126. ELCtrl = document.querySelector(".toolbar-nametable-control");
  127. if (!ELCtrl)
  128. throw new Error("Failed to find element class 'toolbar-nametable-control'.");
  129. var elbanksel = ELCtrl.querySelector(".nametable-bank-select");
  130. if (!elbanksel)
  131. throw new Error("Failed to find element class 'nametable-bank-select' within toolbar.");
  132. // Building out and setting up the tile selections.
  133. var eltilesel = ELCtrl.querySelector(".nametable-tile-select");
  134. if (!eltilesel)
  135. throw new Error("Failed to find element class 'nametable-tile-select' within toolbar.");
  136. var HANDLE_TileSelect = function(){
  137. var oop = eltilesel.querySelector('canvas[selected="True"]');
  138. if (oop){
  139. oop.classList.remove(TILE_SELECT_CLS);
  140. oop.removeAttribute("selected");
  141. }
  142. this.setAttribute("selected", "True");
  143. this.classList.add(TILE_SELECT_CLS);
  144. TileIndex = parseInt(this.getAttribute("value"));
  145. };
  146. var op0 = eltilesel.querySelector('canvas[value="0"]');
  147. if (!op0)
  148. throw new Error("Failed to find initial canvas element within 'nametable-tile-select'.");
  149. op0.addEventListener("click", HANDLE_TileSelect);
  150. for (let i=1; i < 256; i++){
  151. let op = eltilesel.querySelector('canvas[value="' + i + '"]');
  152. if (!op){
  153. op = op0.cloneNode(true);
  154. op.setAttribute("value", i);
  155. eltilesel.appendChild(op);
  156. op.addEventListener("click", HANDLE_TileSelect);
  157. }
  158. }
  159. elbanksel.addEventListener("change", function(){
  160. if (SURF !== null){
  161. var bankname = this.value;
  162. if (bankname === "NULL_BANK"){
  163. SURF.bank = null;
  164. } else {
  165. var bank = CTRLBanksStore.getBank(bankname);
  166. if (bank !== null){
  167. if (SURF.bank === null || (SURF.bank.eq(bank) === false)){
  168. SURF.bank = bank;
  169. UpdateBankTileList();
  170. }
  171. }
  172. }
  173. }
  174. });
  175. }
  176. }
  177. const instance = new CTRLNametableTools();
  178. export default instance;