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

CTRLPalettesStore.js 7.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. import GlobalEvents from "/app/js/common/EventCaller.js";
  2. import Utils from "/app/js/common/Utils.js";
  3. import EditableText from "/app/js/ui/EditableText.js";
  4. import NESPalette from "/app/js/models/NESPalette.js";
  5. const PLI_TEMPLATE = "palette-list-item-template";
  6. const PLI_TITLE = "title";
  7. const PLI_SELECTED = "list-item-selected";
  8. const PLI_BG_COLOR = "pal-bg-color";
  9. const PLI_FG_BASE = "pal-fg-";
  10. const PLI_BG_BASE = "pal-bg-";
  11. const PLI_COLOR_BASE = "nes-color-bg-";
  12. var Palettes = [];
  13. var CurrentPaletteIndex = 0;
  14. var BlockEmits = false;
  15. function HANDLE_PaletteClick(e){
  16. if (!this.hasAttribute("palname")){return;}
  17. var pname = this.getAttribute("palname");
  18. if (Palettes.length > 0){
  19. if (Palettes[CurrentPaletteIndex][0].value !== pname || !Palettes[CurrentPaletteIndex][2].classList.contains(PLI_SELECTED)){
  20. var oel = Palettes[CurrentPaletteIndex][2];
  21. oel.classList.remove(PLI_SELECTED);
  22. for (let i=0; i < Palettes.length; i++){
  23. if (Palettes[i][0].value === pname){
  24. Palettes[i][2].classList.add(PLI_SELECTED);
  25. CurrentPaletteIndex = i;
  26. GlobalEvents.emit("set_app_palette", Palettes[i][1]);
  27. break;
  28. }
  29. }
  30. }
  31. }
  32. }
  33. function SetElPaletteName(el, pname){
  34. el.setAttribute("palname", pname);
  35. var et = new EditableText(el, "title");
  36. et.value = pname;
  37. et.listen("value_change", (v) => {el.setAttribute("palname", v);});
  38. return et;
  39. /*var sel = el.querySelectorAll("." + PLI_TITLE);
  40. if (sel.length === 1){
  41. sel = sel[0];
  42. sel.innerHTML = pname;
  43. }*/
  44. }
  45. function SetElToColor(el, mode, pi, ci, hex){
  46. var cel = null;
  47. if (ci === 0){
  48. cel = el.querySelectorAll("." + PLI_BG_COLOR);
  49. } else {
  50. cel = el.querySelectorAll("." + ((mode == 0) ? PLI_FG_BASE : PLI_BG_BASE) + pi + "-" + ci);
  51. }
  52. if (cel !== null && cel.length === 1){
  53. cel = cel[0];
  54. var clist = cel.getAttribute("class").split(" ");
  55. for (let i=0; i < clist.length; i++){
  56. if (clist[i].startsWith(PLI_COLOR_BASE)){
  57. cel.classList.remove(clist[i]);
  58. break;
  59. }
  60. }
  61. cel.classList.add(PLI_COLOR_BASE + hex.toUpperCase());
  62. }
  63. }
  64. function ColorElementToPalette(el, palette){
  65. SetElToColor(
  66. el, 0,
  67. 0, 0,
  68. palette.get_palette_syscolor_index(0,0,true)
  69. );
  70. for (let p=0; p < 8; p++){
  71. for (let c=1; c < 4; c++){
  72. SetElToColor(
  73. el, (p >= 4) ? 0 : 1,
  74. p%4, c,
  75. palette.get_palette_syscolor_index(p,c,true)
  76. );
  77. }
  78. }
  79. }
  80. function ConnectElementToPalette(el, palette){
  81. palette.listen("palettes_changed", (e) => {
  82. if (e.type == "ALL"){
  83. if (e.hasOwnProperty("cindex")){
  84. SetElToColor(
  85. el, 0,
  86. 0, 0,
  87. palette.get_palette_syscolor_index(0,0,true)
  88. );
  89. } else {
  90. ColorElementToPalette(el, palette);
  91. }
  92. } else if (e.type == "SPRITE"){
  93. SetElToColor(
  94. el, 0,
  95. e.pindex%4, e.cindex,
  96. palette.get_palette_syscolor_index(e.pindex, e.cindex, true)
  97. );
  98. } else if (e.type == "TILE"){
  99. SetElToColor(
  100. el, 1,
  101. e.pindex, e.cindex,
  102. palette.get_palette_syscolor_index(e.pindex, e.cindex, true)
  103. );
  104. }
  105. });
  106. }
  107. function CreatePaletteDOMEntry(pname, palette){
  108. var oel = document.querySelectorAll("." + PLI_TEMPLATE);
  109. if (oel.length == 1){
  110. var el = oel[0].cloneNode(true);
  111. el.classList.remove(PLI_TEMPLATE);
  112. el.classList.remove("hidden");
  113. ConnectElementToPalette(el, palette);
  114. ColorElementToPalette(el, palette);
  115. //SetElPaletteName(el, pname);
  116. el.addEventListener("click", HANDLE_PaletteClick);
  117. oel[0].parentNode.appendChild(el);
  118. return el;
  119. } else {
  120. console.log("WARNING: Multiple templates found. Ambigous state.");
  121. }
  122. return null;
  123. }
  124. class CTRLPalettesStore{
  125. constructor(){
  126. GlobalEvents.listen("palstore-add", (function(e){
  127. if (e.hasOwnProperty("palname")){
  128. this.createPalette(e.palname);
  129. this.activatePalette(e.palname);
  130. }
  131. }).bind(this));
  132. GlobalEvents.listen("palstore-remove", (function(e){
  133. this.removePalette(Palettes[CurrentPaletteIndex][0]);
  134. }).bind(this));
  135. }
  136. get json(){
  137. var d = {
  138. cpi: CurrentPaletteIndex,
  139. pals: []
  140. };
  141. for (let i=0; i < Palettes.length; i++){
  142. d.pals.push([Palettes[i][0], Palettes[i][1].json]);
  143. }
  144. return JSON.stringify(d);
  145. }
  146. set json(j){
  147. try {
  148. var d = JSON.parse(j);
  149. } catch (e) {
  150. throw e;
  151. }
  152. if (d.hasOwnProperty("cpi") && d.hasOwnProperty("pals")){
  153. if (Utils.isInt(d.cpi) && d.pals instanceof Array){
  154. var newPalettes = []
  155. for (let i=0; i < d.pals.length; i++){
  156. if (d.pals[i] instanceof Array){
  157. if (this.getPalette(d.pals[i][0]) === null){
  158. this.createPalette(d.pals[i][0], d.pals[i][1]);
  159. }
  160. }
  161. }
  162. CurrentPaletteIndex = 0
  163. if (Palettes.length > 0){
  164. if (d.cpi >= 0 && d.cpi < Palettes.length){
  165. CurrentPaletteIndex = d.cpi;
  166. }
  167. GlobalEvents.emit("set_app_palette", Palettes[CurrentPaletteIndex][1]);
  168. }
  169. } else {
  170. throw new TypeError("JSON Property Value types invalid.");
  171. }
  172. } else {
  173. throw new TypeError("JSON missing expected properties.");
  174. }
  175. }
  176. initialize(){
  177. if (Palettes.length <= 0)
  178. this.createPalette("Palette");
  179. return this;
  180. }
  181. paletteIndexFromName(name){
  182. for (let i=1; i < Palettes.length; i++){
  183. if (Palettes[i][0].value == name){
  184. return i;
  185. }
  186. }
  187. return -1;
  188. }
  189. getPalette(name){
  190. var i = this.paletteIndexFromName(name);
  191. return (i >= 0) ? Palettes[i][1] : null;
  192. }
  193. createPalette(name, pjson){
  194. var palette = this.getPalette(name);
  195. if (palette === null){
  196. palette = new NESPalette();
  197. if (typeof(pjson) === "string"){
  198. try {
  199. palette.json = pjson;
  200. } catch (e) {
  201. console.log("Failed to create palette.", e.toString());
  202. palette = null;
  203. }
  204. } else {
  205. palette.set_palette([
  206. "0F",
  207. "05","06","07",
  208. "09","0A","0B",
  209. "01","02","03",
  210. "0D","00","20",
  211. "15","16","17",
  212. "19","1A","1B",
  213. "11","21","31",
  214. "1D","10","30"
  215. ]);
  216. }
  217. if (palette !== null){
  218. var el = CreatePaletteDOMEntry(name, palette);
  219. var eltitle = SetElPaletteName(el, name);
  220. Palettes.push([eltitle, palette, el]);
  221. if (Palettes.length <= 1 && !BlockEmits){
  222. el.click();
  223. }
  224. }
  225. }
  226. return this;
  227. }
  228. removePalette(name){
  229. for (let i=0; i < Palettes.length; i++){
  230. if (Palettes[i][0].value === name){
  231. if (CurrentPaletteIndex === i){
  232. CurrentPaletteIndex = 0;
  233. this.activatePalette(Palettes[0][0].value);
  234. }
  235. Palettes[i][2].parentNode.removeChild(Palettes[i][2]);
  236. Palettes.splice(i, 1);
  237. }
  238. }
  239. if (Palettes.length <= 0){
  240. this.createPalette("Palette");
  241. } else {
  242. Palettes[CurrentPaletteIndex][2].click();
  243. }
  244. return this;
  245. }
  246. renamePalette(oldname, newname){
  247. var i = this.paletteIndexFromName(oldname);
  248. if (i < 0)
  249. throw new ValueError("Failed to find palette named '" + oldname +"'. Cannot rename.");
  250. Palettes[i][0].value = newname;
  251. //SetElPaletteName(Palettes[i][2], newname);
  252. return this;
  253. }
  254. activatePalette(name){
  255. var i = this.paletteIndexFromName(name);
  256. if (i >= 0 && CurrentPaletteIndex !== i){
  257. Palettes[CurrentPaletteIndex][2].click();
  258. }
  259. return this;
  260. }
  261. clear(){
  262. for (let i=0; i < Palettes.length; i++){
  263. Palettes[i][2].parentNode.removeChild(Palettes[i][2]);
  264. }
  265. CurrentPaletteIndex = 0;
  266. }
  267. }
  268. const instance = new CTRLPalettesStore();
  269. export default instance;