A pixel art painter geared specifically at NES pixel art. Includes export for .chr binary file as well as palette and namespace data.
您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

269 行
7.4KB

  1. import {EventCaller} from "/app/js/common/EventCaller.js"
  2. /**
  3. * Object for manipulating the eight NES palettes.
  4. * @extends EventCaller
  5. */
  6. export default class NESPalette extends EventCaller{
  7. constructor(){
  8. super();
  9. this.__BGColor = 63; // Index to the background color ALL palettes MUST share.
  10. this.__palette = [
  11. // Tile/Background Palettes
  12. 0,0,0,
  13. 0,0,0,
  14. 0,0,0,
  15. 0,0,0,
  16. // Sprite Palettes
  17. 0,0,0,
  18. 0,0,0,
  19. 0,0,0,
  20. 0,0,0
  21. ];
  22. }
  23. /**
  24. * Sets one or all of the eight color palettes to the values given. By default, function
  25. * assumes the given array is for all eight palettes (or 25 total color indexes, 3 per palette
  26. * and 1 background/transparency color used by ALL palettes).
  27. * If a single palette is being set, the array must only contain 3 entries.
  28. * @param {Array} apci - Array of color indexes to store into the palette(s)
  29. * @param {number} [p=8] - Zero-based index of the palette being set. Any value outside the range of 0 - 7 will set ALL palettes.
  30. * @returns {this}
  31. */
  32. set_palette(apci, p=8){
  33. var StoreColorValue = (i, v) => {
  34. if (typeof(v) == 'number'){
  35. if (i >= 0)
  36. this.__palette[i] = v
  37. else
  38. this.__BGColor = v;
  39. } else if (typeof(v) == 'string' && v.length == 2){
  40. var c = parseInt(v, 16);
  41. if (!isNaN(c)){
  42. if (i >= 0)
  43. this.__palette[i] = c;
  44. else
  45. this.__BGColor = c;
  46. }
  47. }
  48. };
  49. if (typeof(p) != 'number')
  50. throw new TypeError("First argument expected to be a number.");
  51. if (!(apci instanceof Array))
  52. throw new TypeError("Expected an array of color index values.");
  53. if (p < 0 || p >= 8){ // Setting ALL palettes!
  54. if (apci.length != 25)
  55. throw new RangeError("Color array must contain 25 color values to fill all palettes.");
  56. StoreColorValue(-1, apci[0]);
  57. for (var i=0; i < 24; i++){
  58. StoreColorValue(i, apci[i+1]);
  59. }
  60. } else { // Setting a specific palette.
  61. if (apci.length != 3)
  62. throw new RangeError("Color array must contain three color values.");
  63. p *= 3;
  64. for (var i=0; i < 4; i++){
  65. StoreColorValue(p+i, apci[i]);
  66. }
  67. }
  68. this.emit("palettes_changed", {type:"ALL"});
  69. return this;
  70. }
  71. /**
  72. * Sets a palette's color index value to a given system color index.
  73. * NOTE: Setting palette color index 0 for ANY palette changes that index for ALL palettes.
  74. * @param {number} p - The index of the palette being set.
  75. * @param {number} pci - The palette color index (0 - 3) to set.
  76. * @param {number} sci - The system color index (0 - 63) value to set to.
  77. * @returns {this}
  78. */
  79. set_palette_syscolor_index(p, pci, sci){
  80. if (typeof(p) != 'number' || typeof(pci) != 'number' || typeof(sci) != 'number')
  81. throw new TypeError("Palette, palette color, and system color index expected to be numbers.");
  82. if (p < 0 || p >= 8){
  83. throw new RangeError("Palette index is out of bounds.");
  84. }
  85. if (pci < 0 || pci >= 4){
  86. throw new RangeError("Palette color index is out of bounds.");
  87. }
  88. if (typeof(sci) == "string" && sci.length == 2)
  89. sci = parseInt(sci, 16);
  90. if (isNaN(sci))
  91. throw new TypeError("System Color Index expected to be a number of hex value string.");
  92. if (sci < 0 || sci >= 64){
  93. throw new RangeError("System color index is out of bounds.");
  94. }
  95. if (pci == 0){
  96. this.__BGColor = sci;
  97. this.emit("palettes_changed", {type:"ALL", cindex:0});
  98. } else {
  99. this.__palette[(p*3) + (pci-1)] = sci;
  100. this.emit("palettes_changed", {type:(p < 4) ? "TILE" : "SPRITE", pindex:p, cindex:pci});
  101. }
  102. return this;
  103. }
  104. /**
  105. * Returns the system color index at the given palette color index.
  106. * @param {number} p - The index (0 - 7) of the palette.
  107. * @param {number} pci - The palette color index (0 - 3).
  108. * @param {boolean} [ashex=false] - If true, will return the index as a two character hex string.
  109. * @returns {number} - The index of the system color used.
  110. */
  111. get_palette_syscolor_index(p, pci, ashex=false){
  112. if (typeof(p) != 'number' || typeof(pci) != 'number')
  113. throw new TypeError("Palette and color index expected to be numbers.");
  114. if (p < 0 || p >= 8){
  115. throw new RangeError("Palette index is out of bounds.");
  116. }
  117. if (pci < 0 || pci >= 4){
  118. throw new RangeError("Palette color index is out of bounds.");
  119. }
  120. var i = (pci === 0) ? this.__BGColor : this.__palette[(p*3)+(pci-1)];
  121. if (ashex){
  122. i = i.toString(16);
  123. i = ((i.length < 2) ? "0" : "") + i;
  124. }
  125. return i;
  126. }
  127. /**
  128. * Returns a hex string color value used by the NES system at the index stored at the given
  129. * palette color index.
  130. * @param {number} p - The index (0 - 7) of the palette.
  131. * @param {number} pci - The palette color index (0 - 3).
  132. * @returns {string}
  133. */
  134. get_palette_color(p, pci){
  135. if (typeof(p) != 'number' || typeof(pci) != 'number')
  136. throw new TypeError("Palette and color index expected to be numbers.");
  137. if (p < 0 || p >= 8){
  138. throw new RangeError("Palette index is out of bounds.");
  139. }
  140. if (pci < 0 || pci >= 4){
  141. throw new RangeError("Palette color index is out of bounds.");
  142. }
  143. return NESPalette.SystemColor[this.get_palette_syscolor_index(p, pci)];
  144. }
  145. /**
  146. * Generates a small 6502 assembly block string containing the current palette data.
  147. * @param {string} [memname="PaletteData"] The label named under which to store the data.
  148. * @returns {string}
  149. */
  150. to_asm(memname="PaletteData"){
  151. var NumToHex=function(n){
  152. var h = n.toString(16);
  153. if (h.length %2)
  154. h = '0' + h;
  155. return '$' + h;
  156. };
  157. var BGHex = NumToHex(this.__BGColor);
  158. var s = memname + ":\n\t.db ";
  159. // Storing background palette data.
  160. for (var i=0; i < 12; i++){
  161. if (i % 3 == 0)
  162. s += ((i == 0) ? "" : " ") + BGHex;
  163. s += " " + NumToHex(this.__palette[i]);
  164. }
  165. s += "\t; Background palette data.\n\t.db ";
  166. // Storing foreground palette data.
  167. for (var i=12; i < 24; i++){
  168. if (i % 3 == 0)
  169. s += ((i == 12) ? "" : " ") + BGHex;
  170. s += " " + NumToHex(this.__palette[i]);
  171. }
  172. s += "\t; Foreground palette data.";
  173. return s;
  174. }
  175. }
  176. // NES Palette color information comes from the following site...
  177. // http://www.thealmightyguru.com/Games/Hacking/Wiki/index.php/NES_Palette
  178. /**
  179. * Hex string color values representing the NES system palette.
  180. */
  181. NESPalette.SystemColor = [
  182. "#7C7C7C",
  183. "#0000FC",
  184. "#0000BC",
  185. "#4428BC",
  186. "#940084",
  187. "#A80020",
  188. "#A81000",
  189. "#881400",
  190. "#503000",
  191. "#007800",
  192. "#006800",
  193. "#005800",
  194. "#004058",
  195. "#000000",
  196. "#000000",
  197. "#000000",
  198. "#BCBCBC",
  199. "#0078F8",
  200. "#0058F8",
  201. "#6844FC",
  202. "#D800CC",
  203. "#E40058",
  204. "#F83800",
  205. "#E45C10",
  206. "#AC7C00",
  207. "#00B800",
  208. "#00A800",
  209. "#00A844",
  210. "#008888",
  211. "#000000",
  212. "#000000",
  213. "#000000",
  214. "#F8F8F8",
  215. "#3CBCFC",
  216. "#6888FC",
  217. "#9878F8",
  218. "#F878F8",
  219. "#F85898",
  220. "#F87858",
  221. "#FCA044",
  222. "#F8B800",
  223. "#B8F818",
  224. "#58D854",
  225. "#58F898",
  226. "#00E8D8",
  227. "#787878",
  228. "#000000",
  229. "#000000",
  230. "#FCFCFC",
  231. "#A4E4FC",
  232. "#B8B8F8",
  233. "#D8B8F8",
  234. "#F8B8F8",
  235. "#F8A4C0",
  236. "#F0D0B0",
  237. "#FCE0A8",
  238. "#F8D878",
  239. "#D8F878",
  240. "#B8F8B8",
  241. "#B8F8D8",
  242. "#00FCFC",
  243. "#F8D8F8",
  244. "#000000",
  245. "#000000"
  246. ];
  247. NESPalette.Default = [
  248. "#080808",
  249. "#343434",
  250. "#a2a2a2",
  251. "#efefef",
  252. "#666666" // Out of bounds color.
  253. ];