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

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561
  1. import {EventCaller} from "/app/js/common/EventCaller.js";
  2. import Utils from "/app/js/common/Utils.js";
  3. // Keycode list based on...
  4. // https://keycode.info/
  5. var KEYBYCODE = {
  6. 3:"break",
  7. 8:"backspace",
  8. 9:"tab",
  9. 13:"enter",
  10. 16:"shift",
  11. 17:"ctrl",
  12. 18:"alt",
  13. 19:"pause",
  14. 20:"capslock",
  15. 27:"esc",
  16. 32:"space",
  17. 33:"pageup",
  18. 34:"pagedown",
  19. 35:"end",
  20. 36:"home",
  21. 37:"left",
  22. 38:"up",
  23. 39:"right",
  24. 40:"down",
  25. 41:"select",
  26. 42:"print",
  27. 43:"execute",
  28. 44:"printscreen",
  29. 45:"insert",
  30. 46:"delete",
  31. 47:"help",
  32. 48:"0",
  33. 49:"1",
  34. 50:"2",
  35. 51:"3",
  36. 52:"4",
  37. 53:"5",
  38. 54:"6",
  39. 55:"7",
  40. 56:"8",
  41. 57:"9",
  42. 65:"a",
  43. 66:"b",
  44. 67:"c",
  45. 68:"d",
  46. 69:"e",
  47. 70:"f",
  48. 71:"g",
  49. 72:"h",
  50. 73:"i",
  51. 74:"j",
  52. 75:"k",
  53. 76:"l",
  54. 77:"m",
  55. 78:"n",
  56. 79:"o",
  57. 80:"p",
  58. 81:"q",
  59. 82:"r",
  60. 83:"s",
  61. 84:"t",
  62. 85:"u",
  63. 86:"v",
  64. 87:"w",
  65. 88:"x",
  66. 89:"y",
  67. 90:"z",
  68. 91:"leftmod", // Window key (left)
  69. 92:"rightwin",// Window key (right)
  70. 93:"rightmod",// Window key (right)
  71. 96:"num0",
  72. 97:"num1",
  73. 98:"num2",
  74. 99:"num3",
  75. 100:"num4",
  76. 101:"num5",
  77. 102:"num6",
  78. 103:"num7",
  79. 104:"num8",
  80. 105:"num9",
  81. 112:"f1",
  82. 113:"f2",
  83. 114:"f3",
  84. 115:"f4",
  85. 116:"f5",
  86. 117:"f6",
  87. 118:"f7",
  88. 119:"f8",
  89. 120:"f9",
  90. 121:"f10",
  91. 122:"f11",
  92. 123:"f12",
  93. 144:"numlock",
  94. 145:"scrolllock",
  95. };
  96. var KEYBYNAME = (function(){
  97. var keys = Object.keys(KEYBYCODE);
  98. var o = {};
  99. for (var i=0; i < keys.length; i++){
  100. if (KEYBYCODE.hasOwnProperty(keys[i])){
  101. o[KEYBYCODE[keys[i]]] = keys[i];
  102. }
  103. }
  104. return o;
  105. })();
  106. var KEYTYPE = {
  107. "number":[48,49,50,51,52,53,54,55,56,57,96,97,98,99,100,101,102,103,104,105],
  108. "letter":[65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90],
  109. "mod":[16,17,18],
  110. "arrow":[37,38,39,40],
  111. "wasd":[87,65,83,68],
  112. "fn":[112,113,114,115,116,117,118,119,120,121,122,123],
  113. "n1":[48,96],
  114. "n2":[49,97],
  115. "n3":[50,98],
  116. "n4":[51,99],
  117. "n5":[52,100],
  118. "n6":[53,101],
  119. "n7":[54,102],
  120. "n8":[55,103],
  121. "n9":[56,104],
  122. "n0":[57,105]
  123. };
  124. var KEYMAP = {
  125. "lastcode":null,
  126. "lastaction":"",
  127. "currentcodes":[]
  128. };
  129. // TODO: Reeval this idea.
  130. const KEYPRESS_DELAY = 350; // Time in milliseconds. NOTE: May make this a variable in future.
  131. const MOUSECLICK_DELAY = 350; // Time in milliseconds.
  132. function AssignCodeName(code, name){
  133. name = name.toLowerCase();
  134. var remove = (code in Object.keys(KEYBYCODE));
  135. if (name in Object.keys(KEYBYNAME)){
  136. if (remove && KEYBYCODE[code] === name){
  137. return; // We're being asked to replace the exact same thing. SKIP!
  138. }
  139. throw new ValueError("Key name '" + name + "' already assigned. Cannot use duplicate key names.");
  140. }
  141. if (remove){
  142. delete KEYBYNAME[KEYBYCODE[code]];
  143. }
  144. KEYBYCODE[code] = name;
  145. KEYBYNAME[name] = code;
  146. }
  147. function KeymapContains(code){
  148. return KEYMAP["currentcodes"].findIndex(c=>c[0] == code) >= 0;
  149. }
  150. function AddToKeymap(code, action){
  151. KEYMAP["lastcode"] = code;
  152. KEYMAP["lastaction"] = action;
  153. if (KeymapContains(code) == false){
  154. KEYMAP["currentcodes"].push([code, Math.floor(Date.now())]);
  155. if (KEYMAP["currentcodes"].length > 1){
  156. KEYMAP["currentcodes"].sort(function(a, b){return a[0] - b[0];});
  157. }
  158. return true;
  159. }
  160. return false;
  161. }
  162. function RemoveFromKeymap(code, action){
  163. KEYMAP["lastcode"] = code;
  164. KEYMAP["lastaction"] = action;
  165. var ctime = Math.floor(Date.now());
  166. for (var i=0; i < KEYMAP["currentcodes"].length; i++){
  167. if (KEYMAP["currentcodes"][i][0] === code){
  168. var timediff = ctime - KEYMAP["currentcodes"][i][1];
  169. KEYMAP["currentcodes"].splice(i, 1);
  170. return timediff;
  171. }
  172. }
  173. return -1;
  174. }
  175. function KeyNameToCode(key){
  176. return (key in Object.keys(KEYBYNAME)) ? KEYBYNAME[key] : -1;
  177. }
  178. function CodeToKeyName(code){
  179. return (code in Object.keys(KEYBYCODE)) ? KEYBYCODE[code] : "" + code;
  180. }
  181. function CodesToEventName(codes){
  182. var ename = "";
  183. for (var i=0; i < codes.length; i++){
  184. ename += ((ename !== "") ? "+" : "") + CodeToKeyName(codes[i]);
  185. }
  186. return ename;
  187. }
  188. function KeymapEventName(){
  189. return CodesToEventName(KEYMAP["currentcodes"].map(e=>e[0]));
  190. }
  191. function ReorderEventName(ename){
  192. // This function takes a keyboard event name and reorders it into key-code order.
  193. // This way users can write the event any way they want, but should still result in proper
  194. // event being called.
  195. var elist = ename.split("+");
  196. var ecodes = [];
  197. for (var i=0; i < elist.length; i++){
  198. var key = elist[i].trim().toLowerCase();
  199. if (!(key in Object.keys(KEYBYNAME))){
  200. if (!Number.isNaN(key))
  201. ecodes.push(parseInt(key));
  202. else
  203. return ""; // This event name does not include valid key name!
  204. } else {
  205. ecodes.push(KEYBYNAME[key]);
  206. }
  207. }
  208. if (ecodes.length > 0){
  209. ecodes.sort(function(a, b){return a-b;});
  210. return CodesToEventName(ecodes);
  211. }
  212. return "";
  213. }
  214. export default class Input{
  215. constructor(){
  216. this.__emitter = new EventCaller();
  217. this.__preventDefaults = false;
  218. // Internet Explorer... that fudged up p.o.s. has to make mouse button detection difficult
  219. // with different button values from ALL other browsers. So... if Input has this value set to
  220. // true, then mouseup and mousedown events will base it's detection of IE button values, instead
  221. // of the real values.
  222. this.__ieMouseMode = false;
  223. // If set, this is the element that the mouse will focus on and adjust it's position against.
  224. this.__mouseTarget = null;
  225. this.__mousePosition = null;
  226. this.__mouseLastButton = -1;
  227. this.__mouseLastAction = "";
  228. this.__mouseButtons = [];
  229. this.enableKeyboardInput = (function(){
  230. var handle_keydown = (function(e){
  231. if (AddToKeymap(e.keyCode, "keydown")){
  232. var ename = KeymapEventName();
  233. var edata = {
  234. source: this,
  235. iscombo: (ename.indexOf("+") >= 0),
  236. keys: ename,
  237. keycode:e.keyCode,
  238. keyname:CodeToKeyName(e.keyCode),
  239. action:"keydown"
  240. }
  241. this.__emitter.emit(ename, edata);
  242. this.__emitter.emit("keydown", edata);
  243. }
  244. }).bind(this);
  245. var handle_keyup = (function(e){
  246. var timediff = RemoveFromKeymap(e.keyCode, "keyup");
  247. if (timediff < 0){
  248. console.log("WARNING: Failed to find keycode '" + e.keyCode + "' in the Key Map.");
  249. } else {
  250. var ename = KeymapEventName();
  251. var edata = {
  252. source: this,
  253. iscombo: (ename.indexOf("+") >= 0),
  254. keys: ename,
  255. keycode: e.keyCode,
  256. keyname: CodeToKeyName(e.keyCode),
  257. action:"keyup"
  258. }
  259. if (timediff <= KEYPRESS_DELAY && KEYMAP["currentcodes"].length <= 0){
  260. this.__emitter.emit("keypress", edata);
  261. }
  262. this.__emitter.emit("keyup", edata);
  263. }
  264. }).bind(this);
  265. return (function(enable){
  266. enable = (enable !== false);
  267. if (enable){
  268. window.addEventListener("keydown", handle_keydown, false);
  269. //window.addEventListener("keypress", handle_keypress, false);
  270. window.addEventListener("keyup", handle_keyup, false);
  271. } else {
  272. window.removeEventListener("keydown", handle_keydown);
  273. window.removeEventListener("keyup", handle_keyup);
  274. }
  275. }).bind(this);
  276. }).apply(this);
  277. this.enableMouseInput = (function(){
  278. var mousePosition = (function(e){
  279. var pos = {
  280. lastX: (this.__mousePosition !== null) ? this.__mousePosition.x : null,
  281. lastY: (this.__mousePosition !== null) ? this.__mousePosition.y : null,
  282. x: e.clientX,
  283. y: e.clientY,
  284. inbounds: true
  285. }
  286. if (this.__mouseTarget !== null){
  287. var rect = this.__mouseTarget.getBoundingClientRect();
  288. pos.x -= rect.left;
  289. pos.y -= rect.top;
  290. pos.inbounds = (pos.x >= 0 && pos.x < rect.width && pos.y >= 0 && pos.y < rect.height);
  291. }
  292. pos.x = Math.floor(pos.x);
  293. pos.y = Math.floor(pos.y);
  294. return pos;
  295. }).bind(this);
  296. var buttonID = (function(e){
  297. var btn = e.button;
  298. if ((this.__ieMouseMode && btn === 1) || (!this.__ieMouseMode && btn === 0)){
  299. btn = 0;
  300. } else if ((this.__ieMouseMode && e.button === 4) || (!this.__ieMouseMode && e.button === 1)){
  301. btn = 1;
  302. }
  303. return btn;
  304. }).bind(this);
  305. var addMouseButton = (function(btn){
  306. if (this.__mouseButtons.findIndex(b=>b[0]===btn) < 0){
  307. this.__mouseButtons.push([btn, Math.floor(Date.now())]);
  308. return true;
  309. }
  310. return false;
  311. }).bind(this);
  312. var removeMouseButton = (function(btn){
  313. var idx = this.__mouseButtons.findIndex(b=>b[0]===btn);
  314. var diff = -1;
  315. if (idx >= 0){
  316. diff = Math.floor(Date.now()) - this.__mouseButtons[idx][1];
  317. this.__mouseButtons.splice(idx, 1);
  318. }
  319. return diff;
  320. }).bind(this);
  321. var handle_mousemove = (function(e){
  322. if (this.__preventDefaults)
  323. e.preventDefault();
  324. var pos = mousePosition(e);
  325. if (pos.inbounds){
  326. this.__mousePosition = pos;
  327. this.__mouseLastAction = "mousemove";
  328. this.__emitter.emit("mousemove", {
  329. source: this,
  330. lastX: pos.lastX,
  331. lastY: pos.lastY,
  332. x: pos.x,
  333. y: pos.y,
  334. button: this.__mouseLastButton,
  335. action: "mousemove"
  336. });
  337. }
  338. }).bind(this);
  339. var handle_mousedown = (function(e){
  340. var button = buttonID(e);
  341. var pos = mousePosition(e);
  342. if (pos.inbounds){
  343. if (this.__preventDefaults)
  344. e.preventDefault();
  345. if (addMouseButton(button)){
  346. this.__mousePosition = pos;
  347. this.__mouseLastButton = button;
  348. this.__mouseLastAction = "mousedown";
  349. this.__emitter.emit("mousedown", {
  350. source: this,
  351. lastX: pos.lastX,
  352. lastY: pos.lastY,
  353. x: pos.x,
  354. y: pos.y,
  355. button: button,
  356. action: "mousedown"
  357. });
  358. }
  359. }
  360. }).bind(this);
  361. var handle_mouseup = (function(e){
  362. if (this.__preventDefaults)
  363. e.preventDefault();
  364. var button = buttonID(e);
  365. var pos = mousePosition(e);
  366. // NOTE: I still want to check for button removal, even before testing if an event should
  367. // fire, so that I don't have any phantom buttons listed as "pressed" in the mouseButtons list.
  368. var diff = removeMouseButton(button);
  369. if (pos.inbounds && diff >= 0){
  370. this.__mousePosition = pos;
  371. this.__mouseLastButton = button;
  372. this.__mouseLastAction = "mouseup";
  373. var data = {
  374. source: this,
  375. lastX: pos.lastX,
  376. lastY: pos.lastY,
  377. x: pos.x,
  378. y: pos.y,
  379. button: button,
  380. action: "mouseup"
  381. }
  382. this.__emitter.emit("mouseup", data);
  383. if (diff <= MOUSECLICK_DELAY && this.__mouseButtons.length <= 0){
  384. this.__emitter.emit("mouseclick", data);
  385. }
  386. }
  387. }).bind(this);
  388. var handle_mousewheel = (function(e){
  389. if (this.__preventDefaults)
  390. e.preventDefault();
  391. // TODO: Finish me!
  392. }).bind(this);
  393. // This event is purely for preventing Default behaviors on mouse events we're not using.
  394. var handle_mouseprevdef = (function(e){
  395. var pos = mousePosition(e);
  396. if (this.__preventDefaults && pos.inbounds)
  397. e.preventDefault();
  398. }).bind(this);
  399. return (function(enable){
  400. enable = (enable !== false);
  401. if (enable){
  402. window.addEventListener("mousemove", handle_mousemove);
  403. window.addEventListener("mousedown", handle_mousedown);
  404. window.addEventListener("mouseup", handle_mouseup);
  405. window.addEventListener("mousewheel", handle_mousewheel);
  406. window.addEventListener("click", handle_mouseprevdef);
  407. window.addEventListener("dblclick", handle_mouseprevdef);
  408. } else {
  409. window.removeEventListener("mousemove", handle_mousemove);
  410. window.removeEventListener("mousedown", handle_mousedown);
  411. window.removeEventListener("mouseup", handle_mouseup);
  412. window.removeEventListener("mousewheel", handle_mousewheel);
  413. window.removeEventListener("click", handle_mouseprevdef);
  414. window.removeEventListener("dblclick", handle_mouseprevdef);
  415. }
  416. }).bind(this);
  417. }).apply(this);
  418. this.enableKeyboardInput();
  419. this.enableMouseInput();
  420. }
  421. get lastkey(){
  422. if (KEYMAP["lastcode"] !== null){
  423. if (KEYMAP["lastcode"] in Object.keys(KEYBYCODE)){
  424. return KEYBYCODE[KEYMAP["lastcode"]];
  425. }
  426. return "" + KEYMAP["lastcode"];
  427. }
  428. return "0";
  429. }
  430. get lastkeyaction(){
  431. return KEYMAP["lastaction"];
  432. }
  433. get currentKeys(){
  434. return KeymapEventName();
  435. }
  436. get currentKeyCodes(){
  437. return KEYMAP["currentcodes"].map(e=>e[0]);
  438. }
  439. get lastMouseAction(){
  440. return this.__mouseLastAction;
  441. }
  442. get lastMouseButton(){
  443. return this.__mouseLastButton;
  444. }
  445. get lastMousePosition(){
  446. if (this.__mousePosition === null || this.__mousePosition.lastX === null)
  447. return null;
  448. return {
  449. x: this.__mousePosition.lastX,
  450. y: this.__mousePosition.lastY
  451. };
  452. }
  453. get currentMousePosition(){
  454. if (this.__mousePosition === null)
  455. return null;
  456. return {
  457. x: this.__mousePosition.x,
  458. y: this.__mousePosition.y
  459. };
  460. }
  461. get preventDefaults(){return this.__preventDefaults;}
  462. set preventDefaults(p){
  463. this.__preventDefaults = (p === true);
  464. }
  465. get ieMouseMode(){return this.__ieMouseMode;}
  466. set ieMouseMode(m){this.__ieMouseMode = (m === true);}
  467. get mouseTargetElement(){return this.__mouseTarget;}
  468. set mouseTargetElement(el){
  469. if (el === null || Utils.isElement(el)){
  470. this.__mouseTarget = el;
  471. } else {
  472. throw new TypeError("Expected Mouse Target Element to be null or an HTMLElement object.");
  473. }
  474. }
  475. isKeyDown(key){
  476. if (typeof(key) === 'string'){
  477. key = KeyNameToCode(key);
  478. }
  479. for (var i=0; i < KEYMAP["currentcodes"].length; i++){
  480. if (KEYMAP["currentcodes"][i][0] === key){
  481. return true;
  482. }
  483. }
  484. return false;
  485. }
  486. listen(ename, func, owner=null, once=false){
  487. if ((["keyup", "keydown", "keypress", "mousemove", "mousedown", "mouseup", "mouseclick"]).indexOf(ename) >= 0){
  488. this.__emitter.listen(ename, func, owner, once);
  489. } else {
  490. ename = ReorderEventName(ename);
  491. if (ename === ""){
  492. throw new ValueError("Failed to parse key or key combination.");
  493. }
  494. this.__emitter.listen(ename, func, owner, once);
  495. }
  496. return this;
  497. }
  498. unlisten(ename, func, owner=null){
  499. if ((["keyup", "keydown", "keypress", "mousemove", "mousedown", "mouseup", "mouseclick"]).indexOf(ename) >= 0){
  500. this.__emitter.unlisten(ename, func, owner);
  501. } else {
  502. ename = ReorderEventName(ename);
  503. if (ename !== "")
  504. this.__emitter.unlisten(ename, func, owner);
  505. }
  506. return this;
  507. }
  508. }