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个字符

314 行
8.2KB

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