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

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. import Utils from "/app/js/common/Utils.js";
  2. import {EventCaller} from "/app/js/common/EventCaller.js";
  3. import NESPalette from "/app/js/models/NESPalette.js";
  4. function BitMask(offset, inv){
  5. switch(offset){
  6. case 0:
  7. return parseInt((inv === true) ? '01111111' : '10000000', 2);
  8. case 1:
  9. return parseInt((inv === true) ? '10111111' : '01000000', 2);
  10. case 2:
  11. return parseInt((inv === true) ? '11011111' : '00100000', 2);
  12. case 3:
  13. return parseInt((inv === true) ? '11101111' : '00010000', 2);
  14. case 4:
  15. return parseInt((inv === true) ? '11110111' : '00001000', 2);
  16. case 5:
  17. return parseInt((inv === true) ? '11111011' : '00000100', 2);
  18. case 6:
  19. return parseInt((inv === true) ? '11111101' : '00000010', 2);
  20. }
  21. return parseInt((inv === true) ? '11111110' : '00000001', 2);
  22. }
  23. var BLOCK_CHANGE_EMIT = false; // This will block the "data_changed" event when class is processing
  24. // lots of changes.
  25. export default class NESTile extends EventCaller{
  26. constructor(){
  27. super();
  28. this.__paletteIndex = 0;
  29. this.__data = new Uint8Array(16);
  30. this.__pixels = new Proxy(this, {
  31. get: function(obj, prop){
  32. if (prop === "length")
  33. return 64;
  34. if (!Utils.isInt(prop))
  35. throw new TypeError("Expected integer index.");
  36. if (prop < 0 || prop >= 64)
  37. throw new RangeError("Index out of bounds.");
  38. var dindex = Math.floor(prop*0.125);
  39. var bitoffset = 7 - (prop%8);
  40. var v = (obj.__data[dindex] & (1 << bitoffset)) >> bitoffset;
  41. v += 2*((obj.__data[8+dindex] & (1 << bitoffset)) >> bitoffset);
  42. return v;
  43. },
  44. set: function(obj, prop, value){
  45. if (!Utils.isInt(prop))
  46. throw new TypeError("Expected integer index.");
  47. prop = parseInt(prop);
  48. if (!Utils.isInt(value))
  49. throw new TypeError("Color index expected to be integer.");
  50. if (prop < 0 || prop >= 64)
  51. throw new RangeError("Index out of bounds.");
  52. if (value < 0 || value >= 4)
  53. throw new RangeError("Color index out of bounds.");
  54. var dindex = Math.floor(prop*0.125);
  55. var bitoffset = (prop % 8);
  56. if (value == 1 || value == 3){
  57. obj.__data[dindex] |= BitMask(bitoffset);
  58. } else {
  59. obj.__data[dindex] &= BitMask(bitoffset, true);
  60. }
  61. if (value == 2 || value == 3){
  62. obj.__data[8+dindex] |= BitMask(bitoffset);
  63. } else {
  64. obj.__data[8+dindex] &= BitMask(bitoffset, true);
  65. }
  66. if (!BLOCK_CHANGE_EMIT)
  67. obj.emit("data_changed");
  68. return true;
  69. }
  70. });
  71. }
  72. get width(){return 8;}
  73. get height(){return 8;}
  74. get pixels(){
  75. return this.__pixels;
  76. }
  77. get dataArray(){
  78. var d = [];
  79. for (var y = 0; y < 8; y++){
  80. for (var x = 0; x < 8; x++){
  81. d.push(this.getPixelIndex(x, y));
  82. }
  83. }
  84. return d;
  85. }
  86. get chr(){
  87. return new Uint8Array(this.__data);
  88. }
  89. set chr(buff){
  90. if (!(buff instanceof Uint8Array))
  91. throw new TypeError("Expected Uint8Array buffer");
  92. if (buff.length !== 16)
  93. throw new RangeError("Buffer contains invalid byte length.");
  94. this.__data = new Uint8Array(buff);
  95. this.emit("data_changed");
  96. }
  97. get base64(){
  98. var b = ""
  99. for (var i = 0; i < this.__data.length; i++) {
  100. b += String.fromCharCode(this.__data[i]);
  101. }
  102. b += String.fromCharCode(this.__paletteIndex);
  103. return window.btoa(b);
  104. }
  105. set base64(s){
  106. var b = window.atob(s);
  107. var len = b.length;
  108. if (b.length !== 17){
  109. throw new Error("Base64 string contains invalid byte count.");
  110. }
  111. var bytes = new Uint8Array(b.length-1);
  112. for (var i=0; i < b.length-1; i++){
  113. bytes[i] = b.charCodeAt(i);
  114. }
  115. this.__data = bytes;
  116. this.__paletteIndex = b.charCodeAt(b.length-1);
  117. this.emit("data_changed");
  118. }
  119. get paletteIndex(){return this.__paletteIndex;}
  120. set paletteIndex(pi){
  121. if (!Utils.isInt(pi))
  122. throw new TypeError("Palette index expected to be an integer.");
  123. if (pi < 0 || pi >= 4){
  124. throw new RangeError("Palette index out of bounds.");
  125. }
  126. this.__paletteIndex = pi;
  127. this.emit("data_changed");
  128. }
  129. setPixelIndex(x, y, ci){
  130. if (x < 0 || x >= 8 || y < 0 || y >= 8){
  131. throw new ValueError("Coordinates out of bounds.");
  132. }
  133. if (ci < 0 || ci >= 4){
  134. throw new ValueError("Color index out of bounds.");
  135. }
  136. this.__pixels[(y*8)+x] = ci;
  137. return this;
  138. }
  139. getPixelIndex(x, y){
  140. if (x < 0 || x >= 8 || y < 0 || y >= 8){
  141. throw new ValueError("Coordinates out of bounds.");
  142. }
  143. return this.__pixels[(8*y) + x];
  144. }
  145. flip(flag){
  146. if (flag >= 1 && flag <= 3){
  147. var oldData = this.__data;
  148. var newData = new Uint8Array(16);
  149. BLOCK_CHANGE_EMIT = true;
  150. for (var x = 0; x < 8; x++){
  151. for (var y = 0; y < 8; y++){
  152. this.__data = oldData;
  153. var ci = this.getPixelIndex(x, y);
  154. this.__data = newData;
  155. this.setPixelIndex(
  156. (flag == 1 || flag == 3) ? 7 - x : x,
  157. (flag == 2 || flag == 3) ? 7 - y : y,
  158. ci
  159. );
  160. }
  161. }
  162. BLOCK_CHANGE_EMIT = false;
  163. this.emit("data_changed");
  164. }
  165. return this;
  166. }
  167. clone(){
  168. return (new NESTile()).copy(this);
  169. }
  170. copy(t){
  171. if (!(t instanceof NESTile))
  172. throw new TypeError("Expected NESTile object.");
  173. this.__data.set(t.__data);
  174. this.emit("data_changed");
  175. return this;
  176. }
  177. clear(){
  178. this.__data = new Uint8Array(16);
  179. this.emit("data_changed");
  180. return this;
  181. }
  182. isEmpty(){
  183. for (let i=0; i < this.__data.length; i++){
  184. if (this.__data[i] !== 0)
  185. return false;
  186. }
  187. return true;
  188. }
  189. isEq(tile, sameOrientation){
  190. if (!(tile instanceof NESTile)){
  191. throw new TypeError("Expected NESTile instance.");
  192. }
  193. sameOrientation = (sameOrientation === true);
  194. var b64 = this.base64;
  195. if (tile.base64 === b64){
  196. return 0;
  197. }
  198. if (!sameOrientation){
  199. var tc = tile.clone().flip(1); // Flip horizontal.
  200. if (tc.base64 === b64){
  201. return 1;
  202. }
  203. tc.flip(3); // Flip horizontal AND verticle. Net effect is the same as tile.clone().flip(2) ... Flip Verticle
  204. if (tc.base64 === b64){
  205. return 2;
  206. }
  207. tc.flip(1); // Flip horizontal again. Net effect is the same as tile.clone().flip(3) ... flip H & V
  208. if (tc.base64 === b64){
  209. return 3;
  210. }
  211. }
  212. return -1;
  213. }
  214. }