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文字以内のものにしてください。

NESBank.js 24KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847
  1. import Utils from "/app/js/common/Utils.js";
  2. import ISurface from "/app/js/ifaces/ISurface.js";
  3. import NESTile from "/app/js/models/NESTile.js";
  4. import NESPalette from "/app/js/models/NESPalette.js";
  5. function CnvIdx(x, y, am, off){
  6. var res = {
  7. side: 0,
  8. tileidx: 0,
  9. x: 0,
  10. y: 0
  11. }
  12. switch(am){
  13. case NESBank.ACCESSMODE_8K:
  14. res.side = (x >= 128) ? 1 : 0;
  15. x -= (res.side === 1) ? 128 : 0;
  16. res.tileidx = (Math.floor(y/8) * 16) + Math.floor(x / 8);
  17. break;
  18. case NESBank.ACCESSMODE_4K:
  19. res.side = off;
  20. res.tileidx = (Math.floor(y/8) * 16) + Math.floor(x / 8);
  21. break;
  22. case NESBank.ACCESSMODE_2K:
  23. res.side = Math.floor(off * 0.5);
  24. off -= (off > 1) ? 2 : 0;
  25. res.tileidx = (off*128) + ((Math.floor(y/8) * 16) + Math.floor(x / 8));
  26. break;
  27. case NESBank.ACCESSMODE_1K:
  28. res.side = Math.floor(off * 0.25);
  29. off -= (off > 3) ? 4 : 0;
  30. res.tileidx = (off * 64) + ((Math.floor(y/8) * 16) + Math.floor(x / 8));
  31. break;
  32. case NESBank.ACCESSMODE_1T:
  33. res.side = (off >= 256) ? 1 : 0;
  34. res.tileidx = (off >= 256) ? off - 256 : off;
  35. break;
  36. case NESBank.ACCESSMODE_2T:
  37. res.side = (off >= 128) ? 1 : 0;
  38. off -= (off >= 128) ? 128 : 0;
  39. //res.tileidx = (Math.floor(off % 8)*32) + Math.floor(off / 16) + ((y >= 8) ? 16 : 0);
  40. res.tileidx = ((Math.floor(off/16)*32) + (off % 16)) + ((y >= 8) ? 16 : 0);
  41. }
  42. res.x = x%8;
  43. res.y = y%8;
  44. return res;
  45. }
  46. function AdjOffsetToNewMode(nmode, omode, ooff){
  47. // NOTE: 8K never shows up because it will ALWAYS return an offset of 0, so it's easier to just let it all fall through
  48. // to the default return value.
  49. switch(nmode){
  50. case NESBank.ACCESSMODE_4K:
  51. if (ooff > 1){
  52. switch(omode){
  53. case NESBank.ACCESSMODE_2K:
  54. return Math.floor(ooff * 0.5);
  55. case NESBank.ACCESSMODE_1K:
  56. return Math.floor(ooff * 0.25);
  57. case NESBank.ACCESSMODE_1T:
  58. return Math.floor(ooff / 256);
  59. case NESBank.ACCESSMODE_2T:
  60. return Math.floor(ooff / 128);
  61. }
  62. }
  63. return ooff;
  64. case NESBank.ACCESSMODE_2K:
  65. switch(omode){
  66. case NESBank.ACCESSMODE_4K:
  67. return ooff * 2;
  68. case NESBank.ACCESSMODE_1K:
  69. return Math.floor(ooff * 0.5);
  70. case NESBank.ACCESSMODE_1T:
  71. return Math.floor(ooff * 0.0078125); // divide by 128
  72. case NESBank.ACCESSMODE_2T:
  73. return Math.floor(ooff * 0.015625); // divide by 64
  74. }
  75. break;
  76. case NESBank.ACCESSMODE_1K:
  77. switch(omode){
  78. case NESBank.ACCESSMODE_4K:
  79. return ooff * 4;
  80. case NESBank.ACCESSMODE_2K:
  81. return ooff * 2;
  82. case NESBank.ACCESSMODE_1T:
  83. return Math.floor(ooff * 0.015625); // divide by 64
  84. case NESBank.ACCESSMODE_2T:
  85. return Math.floor(ooff * 0.03125); // divide by 32
  86. }
  87. break;
  88. case NESBank.ACCESSMODE_1T:
  89. switch(omode){
  90. case NESBank.ACCESSMODE_4K:
  91. return ooff * 256;
  92. case NESBank.ACCESSMODE_2K:
  93. return ooff * 128;
  94. case NESBank.ACCESSMODE_1K:
  95. return ooff * 64;
  96. case NESBank.ACCESSMODE_2T:
  97. let _off = (ooff >= 128) ? ooff - 128 : ooff;
  98. return ((ooff >= 128) ? 256 : 0) + ((Math.floor(_off / 16)*32) + (_off % 16));
  99. }
  100. break;
  101. case NESBank.ACCESSMODE_2T:
  102. switch(omode){
  103. case NESBank.ACCESSMODE_4K:
  104. return ooff * 128;
  105. case NESBank.ACCESSMODE_2K:
  106. return ooff * 64;
  107. case NESBank.ACCESSMODE_1K:
  108. return ooff * 32;
  109. case NESBank.ACCESSMODE_1T:
  110. let _off = (ooff >= 256) ? ooff - 256 : ooff;
  111. return ((ooff >= 256) ? 128 : 0) + ((Math.floor(_off / 32)*16) + (_off % 16));
  112. }
  113. break;
  114. }
  115. return 0;
  116. }
  117. export default class NESBank extends ISurface{
  118. constructor(){
  119. super();
  120. this.__LP = []; // Left Patterns (Sprites)
  121. this.__RP = []; // Right Patterns (Backgrounds)
  122. this.__View = [];
  123. this.__AccessMode = NESBank.ACCESSMODE_8K;
  124. this.__AccessOffset = 0;
  125. this.__undos = []; // Holds Base64 snapshots of the bank.
  126. this.__redos = []; // Holds Base64 snapshots of work undone.
  127. this.__emitsEnabled = true;
  128. var handle_datachanged = Utils.debounce((function(side, idx){
  129. var sendEmit = false;
  130. switch(this.__AccessMode){
  131. case NESBank.ACCESSMODE_2T:
  132. if (side === Math.floor(this.__AccessOffset / 128)){
  133. if (idx === this.__AccessOffset)
  134. sendEmit = true;
  135. }
  136. case NESBank.ACCESSMODE_1T:
  137. if (side === Math.floor(this.__AccessOffset / 256)){
  138. if (idx === this.__AccessOffset)
  139. sendEmit = true;
  140. }
  141. case NESBank.ACCESSMODE_1K:
  142. if (side === Math.floor(this.__AccessOffset / 4)){
  143. if (Math.floor(idx / 64) === Math.floor(this.__AccessOffset/4))
  144. sendEmit = true;
  145. }
  146. break;
  147. case NESBank.ACCESSMODE_2K:
  148. if (side === Math.floor(this.__AccessOffset / 2)){
  149. if (Math.floor(idx / 128) === Math.floor(this.__AccessOffset/2))
  150. sendEmit = true;
  151. }
  152. break;
  153. case NESBank.ACCESSMODE_4K:
  154. if (side === this.__AccessOffset)
  155. sendEmit = true;
  156. break;
  157. case NESBank.ACCESSMODE_8K:
  158. sendEmit = true;
  159. }
  160. if (sendEmit && this.__emitsEnabled){
  161. this.emit("data_changed");
  162. }
  163. }).bind(this), 250);
  164. for (var i=0; i < 256; i++){
  165. this.__LP.push(new NESTile());
  166. this.__LP[i].listen("data_changed", handle_datachanged.bind(this, 0, i));
  167. this.__RP.push(new NESTile());
  168. this.__RP[i].listen("data_changed", handle_datachanged.bind(this, 1, i));
  169. }
  170. this.__palette = null;
  171. }
  172. get access_mode(){return this.__AccessMode;}
  173. set access_mode(m){
  174. if (!Utils.isInt(m))
  175. throw new TypeError("Access mode expected to be integer.");
  176. var oam = this.__AccessMode;
  177. switch(m){
  178. case NESBank.ACCESSMODE_8K:
  179. this.__AccessMode = NESBank.ACCESSMODE_8K;
  180. break;
  181. case NESBank.ACCESSMODE_4K:
  182. this.__AccessMode = NESBank.ACCESSMODE_4K
  183. break;
  184. case NESBank.ACCESSMODE_2K:
  185. this.__AccessMode = NESBank.ACCESSMODE_2K;
  186. break;
  187. case NESBank.ACCESSMODE_1K:
  188. this.__AccessMode = NESBank.ACCESSMODE_1K;
  189. break;
  190. case NESBank.ACCESSMODE_1T:
  191. this.__AccessMode = NESBank.ACCESSMODE_1T;
  192. break;
  193. case NESBank.ACCESSMODE_2T:
  194. this.__AccessMode = NESBank.ACCESSMODE_2T;
  195. break;
  196. default:
  197. throw new Error("Unknown Access Mode.");
  198. }
  199. this.__AccessOffset = AdjOffsetToNewMode(m, oam, this.__AccessOffset);
  200. if (this.__emitsEnabled)
  201. this.emit("data_changed");
  202. }
  203. get access_offset(){return this.__AccessOffset;}
  204. set access_offset(o){
  205. if (!Utils.isInt(o))
  206. throw new TypeError("Access offset expected to be integer.");
  207. switch (this.__AccessMode){
  208. case NESBank.ACCESSMODE_8K:
  209. if (o !== 0)
  210. throw new RangeError("Access Offset is out of bounds based on current Access Mode.");
  211. break;
  212. case NESBank.ACCESSMODE_4K:
  213. if (o !== 0 && o !== 1)
  214. throw new RangeError("Access Offset is out of bounds based on current Access Mode.");
  215. break;
  216. case NESBank.ACCESSMODE_2K:
  217. if (o < 0 || o >= 4)
  218. throw new RangeError("Access Offset is out of bounds based on current Access Mode.");
  219. break;
  220. case NESBank.ACCESSMODE_1K:
  221. if (o < 0 || o >= 8)
  222. throw new RangeError("Access Offset is out of bounds based on current Access Mode.");
  223. break;
  224. case NESBank.ACCESSMODE_1T:
  225. if (o < 0 || o >= 512)
  226. throw new RangeError("Access Offset is out of bounds based on current Access Mode.");
  227. break;
  228. case NESBank.ACCESSMODE_2T:
  229. if (o < 0 || o >= 256)
  230. throw new RangeError("Access Offset is out of bounds based on current Access Mode.");
  231. break;
  232. }
  233. this.__AccessOffset = o;
  234. if (this.__emitsEnabled)
  235. this.emit("data_changed");
  236. }
  237. get access_offset_length(){
  238. switch(this.__AccessMode){
  239. case NESBank.ACCESSMODE_4K:
  240. return 2;
  241. case NESBank.ACCESSMODE_2K:
  242. return 4;
  243. case NESBank.ACCESSMODE_1K:
  244. return 8;
  245. case NESBank.ACCESSMODE_1T:
  246. return 512;
  247. case NESBank.ACCESSMODE_2T:
  248. return 256;
  249. }
  250. return 0;
  251. }
  252. get json(){
  253. JSON.stringify({
  254. LP: this.__LP.map(x=>x.base64),
  255. RP: this.__RP.map(x=>x.base64)
  256. });
  257. }
  258. get chr(){
  259. var buff = null;
  260. var offset = 0;
  261. switch (this.__AccessMode){
  262. case NESBank.ACCESSMODE_8K:
  263. buff = new Uint8Array(8192);
  264. this.__LP.forEach((i) => {
  265. buff.set(i.chr, offset);
  266. offset += 16;
  267. });
  268. this.__RP.forEach((i) => {
  269. buff.set(i.chr, offset);
  270. offset += 16;
  271. });
  272. break;
  273. case NESBank.ACCESSMODE_4K:
  274. buff = new Uint8Array(4096);
  275. var list = (this.__AccessOffset === 0) ? this.__LP : this.__RP;
  276. list.forEach((i) => {
  277. buff.set(i.chr, offset);
  278. offset += 16;
  279. });
  280. break;
  281. case NESBank.ACCESSMODE_2K:
  282. buff = new Uint8Array(2048);
  283. var list = (this.__AccessOffset < 2) ? this.__LP : this.__RP;
  284. var s = Math.floor(this.__AccessOffset * 0.5) * 128;
  285. var e = s + 128;
  286. for (let i=s; i < e; i++){
  287. buff.set(list[i].chr, offset);
  288. offset += 16;
  289. }
  290. break;
  291. case NESBank.ACCESSMODE_1K:
  292. buff = new Uint8Array(1024);
  293. var list = (this.__AccessOffset < 4) ? this.__LP : this.__RP;
  294. var s = Math.floor(this.__AccessOffset * 0.25) * 64;
  295. var e = s + 64;
  296. for (let i=s; i < e; i++){
  297. buff.set(list[i].chr, offset);
  298. offset += 16;
  299. }
  300. break;
  301. case NESBank.ACCESSMODE_1T:
  302. var list = (this.__AccessOffset < 256) ? this.__LP : this.__RP;
  303. var idx = this.__AccessOffset % 256;
  304. buff = list[idx].chr;
  305. break;
  306. case NESBank.ACCESSMODE_2T:
  307. var list = (this.__AccessOffset < 128) ? this.__LP : this.__RP;
  308. var off = this.__AccessOffset % 128;
  309. var idx = ((Math.floor(off / 16)*32) + (off % 16));
  310. buff = new Uint8Array(32);
  311. buff.set(list[idx].chr, 0);
  312. buff.set(list[idx+16].chr, 16);
  313. break;
  314. }
  315. return buff;
  316. }
  317. set chr(buff){
  318. if (!(buff instanceof Uint8Array))
  319. throw new TypeError("Expected Uint8Array buffer.");
  320. this.setCHR(buff);
  321. }
  322. get base64(){
  323. return this.getBase64(this.__AccessMode, this.__AccessOffset);
  324. //var b = "";
  325. //var data = this.chr;
  326. //for (var i = 0; i < data.length; i++) {
  327. // b += String.fromCharCode(data[i]);
  328. //}
  329. //return window.btoa(b);
  330. }
  331. set base64(s){
  332. var b = window.atob(s);
  333. var len = b.length;
  334. if (b.length !== 8192){
  335. throw new Error("Base64 string contains invalid byte count.");
  336. }
  337. b = new Uint8Array(b.split("").map(function(c){
  338. return c.charCodeAt(0);
  339. }));
  340. this.chr = b;
  341. }
  342. get palette(){return this.__palette;}
  343. set palette(p){
  344. if (p !== null && !(p instanceof NESPalette))
  345. throw new TypeError("Expected null or NESPalette object.");
  346. if (p !== this.__palette){
  347. this.__palette = p;
  348. }
  349. }
  350. get width(){
  351. if (this.__AccessMode == NESBank.ACCESSMODE_8K)
  352. return 256;
  353. if (this.__AccessMode == NESBank.ACCESSMODE_1T || this.__AccessMode == NESBank.ACCESSMODE_2T)
  354. return 8;
  355. return 128;
  356. }
  357. get height(){
  358. switch(this.__AccessMode){
  359. case NESBank.ACCESSMODE_2K:
  360. return 64;
  361. case NESBank.ACCESSMODE_1K:
  362. return 32;
  363. case NESBank.ACCESSMODE_1T:
  364. return 8;
  365. case NESBank.ACCESSMODE_2T:
  366. return 16;
  367. }
  368. return 128;
  369. }
  370. get length(){return this.width * this.height;}
  371. get undos(){return this.__undos.length;}
  372. get redos(){return this.__redos.length;}
  373. get coloridx(){
  374. return new Proxy(this, {
  375. get:function(obj, prop){
  376. var len = obj.length * 8;
  377. if (prop === "length")
  378. return len;
  379. if (!Utils.isInt(prop))
  380. throw new TypeError("Expected integer index.");
  381. prop = parseInt(prop);
  382. if (prop < 0 || prop >= len)
  383. return NESPalette.Default[4];
  384. var x = Math.floor(prop % this.width);
  385. var y = Math.floor(prop / this.width);
  386. var res = CnvIdx(x, y, this.__AccessMode, this.__AccessOffset);
  387. var list = (res.side === 0) ? obj.__LP : obj.__RP;
  388. return list[res.tileidx].getPixelIndex(res.x, res.y);
  389. },
  390. set:function(obj, prop, value){
  391. if (!Utils.isInt(prop))
  392. throw new TypeError("Expected integer index.");
  393. if (!Utils.isInt(value))
  394. throw new TypeError("Color expected to be integer.");
  395. prop = parseInt(prop);
  396. value = parseInt(value);
  397. if (prop < 0 || prop >= len)
  398. throw new RangeError("Index out of bounds.");
  399. if (value < 0 || value >= 4)
  400. throw new RangeError("Color index out of bounds.");
  401. var x = Math.floor(prop % this.width);
  402. var y = Math.floor(prop / this.width);
  403. var res = CnvIdx(x, y, this.__AccessMode, this.__AccessOffset);
  404. var list = (res.side === 0) ? obj.__LP : obj.__RP;
  405. list[res.tileidx].setPixelIndex(res.x, res.y, value);
  406. return true;
  407. }
  408. });
  409. }
  410. get lp(){
  411. return new Proxy(this, {
  412. get: function(obj, prop){
  413. if (prop === "length")
  414. return obj.__LP.length;
  415. if (!Utils.isInt(prop))
  416. throw new TypeError("Expected integer index.");
  417. prop = parseInt(prop);
  418. if (prop < 0 || prop >= 256)
  419. throw new RangeError("Index out of bounds.");
  420. return obj.__LP[prop];
  421. },
  422. set: function(obj, prop, value){
  423. if (!Utils.isInt(prop))
  424. throw new TypeError("Expected integer index.");
  425. if (!(value instanceof NESTile))
  426. throw new TypeError("Can only assign NESTile objects.");
  427. prop = parseInt(prop);
  428. if (prop < 0 || prop >= 256)
  429. throw new RangeError("Index out of bounds.");
  430. obj.__LP[prop].copy(value);
  431. return true;
  432. }
  433. });
  434. }
  435. get rp(){
  436. return new Proxy(this, {
  437. get: function(obj, prop){
  438. if (prop === "length")
  439. return obj.__RP.length;
  440. if (!Utils.isInt(prop))
  441. throw new TypeError("Expected integer index.");
  442. prop = parseInt(prop);
  443. if (prop < 0 || prop >= 256)
  444. throw new RangeError("Index out of bounds.");
  445. return obj.__RP[prop];
  446. },
  447. set: function(obj, prop, value){
  448. if (!Utils.isInt(prop))
  449. throw new TypeError("Expected integer index.");
  450. if (!(value instanceof NESTile))
  451. throw new TypeError("Can only assign NESTile objects.");
  452. prop = parseInt(prop);
  453. if (prop < 0 || prop >= 256)
  454. throw new RangeError("Index out of bounds.");
  455. obj.__RP[prop].copy(value);
  456. return true;
  457. }
  458. });
  459. }
  460. copy(b){
  461. if (!(b instanceof NESBank))
  462. throw new TypeError("Expected NESBank object.");
  463. for (var i=0; i < 256; i++){
  464. this.lp[i] = b.lp[i];
  465. this.rp[i] = b.rp[i];
  466. }
  467. return this;
  468. }
  469. clone(){
  470. return (new NESBank()).copy(this);
  471. }
  472. getBase64(mode, offset){
  473. var b = "";
  474. var data = this.getCHR(mode, offset);
  475. for (var i = 0; i < data.length; i++) {
  476. b += String.fromCharCode(data[i]);
  477. }
  478. return window.btoa(b);
  479. }
  480. getCHR(mode, offset){
  481. this.__emitsEnabled = false;
  482. var oam = this.access_mode;
  483. var oao = this.access_offset;
  484. try{
  485. this.access_mode = mode;
  486. this.access_offset = offset;
  487. } catch (e){
  488. this.access_mode = oam;
  489. this.access_offset = oao;
  490. this.__emitsEnabled = true;
  491. throw e;
  492. }
  493. var chr = this.chr;
  494. this.access_mode = oam;
  495. this.access_offset = oao;
  496. this.__emitsEnabled = true;
  497. return chr;
  498. }
  499. setCHR(buff, offset){
  500. if (!Utils.isInt(offset))
  501. offset = -1;
  502. var idx = 0;
  503. switch(buff.length){
  504. case 8192:
  505. if (offset < 0)
  506. offset = AdjOffsetToNewMode(NESBank.ACCESSMODE_8K, this.__AccessMode, this.__AccessOffset);
  507. this.__LP.forEach((i) => {
  508. i.chr = buff.slice(idx, idx+16);
  509. idx += 16;
  510. });
  511. this.__RP.forEach((i) => {
  512. i.chr = buff.slice(idx, idx+16);
  513. idx += 16;
  514. });
  515. break;
  516. case 4096:
  517. if (offset < 0)
  518. offset = AdjOffsetToNewMode(NESBank.ACCESSMODE_4K, this.__AccessMode, this.__AccessOffset);
  519. if (offset >= 2)
  520. throw new RangeError("Offset mismatch based on Buffer length.");
  521. var list = (offset === 0) ? this.__LP : this.__RP;
  522. list.forEach((i) => {
  523. i.chr = buff.slice(idx, idx+16);
  524. idx += 16;
  525. });
  526. break;
  527. case 2048:
  528. if (offset < 0)
  529. offset = AdjOffsetToNewMode(NESBank.ACCESSMODE_2K, this.__AccessMode, this.__AccessOffset);
  530. if (offset >= 4)
  531. throw new RangeError("Offset mismatch based on Buffer length.");
  532. var list = (offset < 2) ? this.__LP : this.__RP;
  533. var s = Math.floor(offset * 0.5) * 128;
  534. var e = s + 128;
  535. for (let i=s; i < e; i++){
  536. list[i].chr = buff.slice(idx, idx+16);
  537. idx += 16;
  538. }
  539. break;
  540. case 1024:
  541. if (offset < 0)
  542. offset = AdjOffsetToNewMode(NESBank.ACCESSMODE_1K, this.__AccessMode, this.__AccessOffset);
  543. if (offset >= 8)
  544. throw new RangeError("Offset mismatch based on Buffer length.");
  545. var list = (offset < 4) ? this.__LP : this.__RP;
  546. var s = Math.floor(this.__AccessOffset * 0.25) * 64;
  547. var e = s + 64;
  548. for (let i=s; i < e; i++){
  549. list[i].chr = buff.slice(idx, idx+16);
  550. idx += 16;
  551. }
  552. break;
  553. case 32:
  554. if (offset < 0)
  555. offset = AdjOffsetToNewMode(NESBank.ACCESSMODE_2T, this.__AccessMode, this.__AccessOffset);
  556. if (offset >= 256)
  557. throw new RangeError("Offset mismatch based on Buffer length.");
  558. var list = (offset < 128) ? this.__LP : this.__RP;
  559. var off = offset % 128;
  560. var idx = ((Math.floor(off / 32)*16) + (off % 16));
  561. list[idx].chr = buff.slice(0, 16);
  562. list[idx+16].char = buff.slice(16, 32);
  563. break;
  564. case 16:
  565. if (offset < 0)
  566. offset = AdjOffsetToNewMode(NESBank.ACCESSMODE_1T, this.__AccessMode, this.__AccessOffset);
  567. if (offset >= 512)
  568. throw new RangeError("Offset mismatch based on Buffer length.");
  569. var list = (offset < 256) ? this.__LP : this.__RP;
  570. var idx = offset % 256;
  571. list[idx].chr = buff;
  572. break;
  573. default:
  574. throw new RangeError("Buffer length does not match any of the supported bank sizes.");
  575. }
  576. return this;
  577. }
  578. getColor(x,y){
  579. if (x < 0 || x >= this.width || y < 0 || y >= this.height)
  580. return this.__default_pi[4];
  581. var res = CnvIdx(x, y, this.__AccessMode, this.__AccessOffset);
  582. var list = (res.side === 0) ? this.__LP : this.__RP;
  583. var pi = list[res.tileidx].paletteIndex + ((res.side === 0) ? 4 : 0);
  584. var ci = list[res.tileidx].getPixelIndex(res.x, res.y);
  585. if (this.__palette !== null){
  586. return this.__palette.get_palette_color(pi, ci);
  587. }
  588. return NESPalette.Default[ci];
  589. }
  590. getColorIndex(x, y){
  591. if (x < 0 || x >= this.width || y < 0 || y >= this.height)
  592. return {pi: -1, ci:-1};
  593. var res = CnvIdx(x, y, this.__AccessMode, this.__AccessOffset);
  594. var list = (res.side === 0) ? this.__LP : this.__RP;
  595. return {
  596. pi: list[res.tileidx].paletteIndex,
  597. ci: list[res.tileidx].getPixelIndex(res.x, res.y)
  598. };
  599. }
  600. getRegion(x, y, w, h){
  601. if (w <= 0 || h <= 0)
  602. throw new RangeError("Width and/or Height must be greater than zero.");
  603. var region = [];
  604. for (let j=y; j<y+h; j++){
  605. if (j === this.height){
  606. h = j - y;
  607. break;
  608. }
  609. for (let i=x; i<x+w; i++){
  610. if (i === this.width){
  611. w = i - x;
  612. break;
  613. }
  614. region.push(this.getColorIndex(i, j));
  615. }
  616. }
  617. return {w:w, h:h, r:region};
  618. }
  619. setColorIndex(x, y, ci, pi){
  620. if (x < 0 || x >= this.width || y < 0 || y > this.height)
  621. throw new RangeError("Coordinates out of bounds.");
  622. if (!Utils.isInt(pi))
  623. pi = -1;
  624. if (!Utils.isInt(ci))
  625. ci = 0;
  626. if (pi < 0){
  627. this.coloridx[(y*this.width)+x] = ci;
  628. } else {
  629. var res = CnvIdx(x, y, this.__AccessMode, this.__AccessOffset);
  630. var list = (res.side === 0) ? this.__LP : this.__RP;
  631. list[res.tileidx].paletteIndex = pi;
  632. list[res.tileidx].setPixelIndex(res.x, res.y, ci);
  633. }
  634. return this;
  635. }
  636. setRegion(x, y, w, h, r){
  637. if (w <= 0 || h <= 0)
  638. throw new RangeError("Width and/or Height must be greater than zero.");
  639. if (!(r instanceof Array)){
  640. throw new TypeError("Region expected to be an array.");
  641. }
  642. if (r.length !== w*h)
  643. throw new RangeError("Region length does not match given width/height values.");
  644. for (let j=0; j < h; j++){
  645. for (let i=0; i < w; i++){
  646. var index = (j*w) + i;
  647. if ("pi" in r[index] && "ci" in r[index] && r[index].ci >= 0){
  648. var _x = i+x;
  649. var _y = j+y;
  650. if (_x >= 0 && _x < this.width && _y >= 0 && _y < this.height)
  651. this.setColorIndex(_x, _y, r[index].ci, r[index].pi);
  652. }
  653. }
  654. }
  655. return this;
  656. }
  657. snapshot(){
  658. if (this.__redos.length > 0) // Remove the redo history. We're adding a new snapshot.
  659. this.__redos = [];
  660. var snap = this.base64;
  661. if (this.__undos.length === this.__historyLength){
  662. this.__undos.pop();
  663. }
  664. this.__undos.splice(0,0,snap);
  665. return this;
  666. }
  667. undo(){
  668. if (this.__undos.length > 0){
  669. var usnap = this.__undos.splice(0, 1)[0];
  670. var rsnap = this.base64;
  671. this.base64 = usnap;
  672. if (this.__redos.length === this.__historyLength){
  673. this.__redos.pop();
  674. }
  675. this.__redos.splice(0,0,rsnap);
  676. }
  677. return this;
  678. }
  679. redo(){
  680. if (this.__redos.length > 0){
  681. var rsnap = this.__redos.splice(0,1)[0];
  682. var usnap = this.base64;
  683. this.base64 = rsnap;
  684. if (this.__undos.length === this.__historyLength){
  685. this.__undos.pop();
  686. }
  687. this.__undos.splice(0,0,usnap);
  688. }
  689. return this;
  690. }
  691. clearUndos(){
  692. this.__undos = [];
  693. return this;
  694. }
  695. clearRedos(){
  696. this.__redos = [];
  697. return this;
  698. }
  699. removeDuplicates(sameOrientation){
  700. sameOrientation = (sameOrientation === true);
  701. var remdup = (list, idxoff) => {
  702. for (let i=1; i < list.length; i++){
  703. for (let j=0; j < i; j++){
  704. var res = list[i].isEq(list[j], sameOrientation);
  705. if (res >= 0){
  706. list[i].clear();
  707. this.emit("tile_drop", {
  708. oIdx: idxoff + i,
  709. nIdx: idxoff + j,
  710. flag: res
  711. });
  712. }
  713. }
  714. }
  715. };
  716. this.clearUndos();
  717. this.clearRedos();
  718. remdup(this.__LP, 0);
  719. remdup(this.__RP, 256);
  720. return this;
  721. }
  722. compact(ignoreTileZero){
  723. ignoreTileZero = (ignoreTileZero === true);
  724. var comp = (list, idxoff, it0) => {
  725. var etile = (it0) ? 1 : 0;
  726. while (etile < list.length && !list[etile].isEmpty()){
  727. etile++;
  728. }
  729. var ctile = etile + 1;
  730. for (ctile; ctile < list.length; ctile++){
  731. if (!list[ctile].isEmpty()){
  732. list[etile].copy(list[ctile]);
  733. list[ctile].clear();
  734. this.emit("tile_move", {
  735. oIdx: idxoff + ctile,
  736. nIdx: idxoff + etile
  737. });
  738. etile++;
  739. }
  740. }
  741. };
  742. this.clearUndos();
  743. this.clearRedos();
  744. comp(this.__LP, 0, false);
  745. comp(this.__RP, 256, ignoreTileZero);
  746. return this;
  747. }
  748. eq(b){
  749. if (b instanceof NESBank){
  750. if (this.getBase64(NESBank.ACCESSMODE_8K, 0) === b.getBase64(NESBank.ACCESSMODE_8K, 0))
  751. return true;
  752. }
  753. return false;
  754. }
  755. }
  756. NESBank.ACCESSMODE_8K = 0;
  757. NESBank.ACCESSMODE_1K = 1;
  758. NESBank.ACCESSMODE_2K = 2;
  759. NESBank.ACCESSMODE_4K = 3;
  760. NESBank.ACCESSMODE_1T = 4; // 8x8 Tile size
  761. NESBank.ACCESSMODE_2T = 5; // 8x16 Tile size