From 1d0fd5e55530c493e6149111fb28e0be08ee92bc Mon Sep 17 00:00:00 2001 From: Olivier Biot Date: Wed, 14 Feb 2024 09:03:40 +0800 Subject: [PATCH] small code refactoring to increase readability and "isolate" sprite sheet and/or packed texture parsing --- src/video/texture/atlas.js | 125 ++-------------------- src/video/texture/parser/spritesheet.js | 68 ++++++++++++ src/video/texture/parser/texturepacker.js | 55 ++++++++++ 3 files changed, 129 insertions(+), 119 deletions(-) create mode 100644 src/video/texture/parser/spritesheet.js create mode 100644 src/video/texture/parser/texturepacker.js diff --git a/src/video/texture/atlas.js b/src/video/texture/atlas.js index 34d090248..5a7b43b60 100755 --- a/src/video/texture/atlas.js +++ b/src/video/texture/atlas.js @@ -3,7 +3,8 @@ import Sprite from "./../../renderable/sprite.js"; import { renderer } from "./../video.js"; import pool from "./../../system/pooling.js"; import { getImage } from "./../../loader/loader.js"; -import { ETA } from "./../../math/math.js"; +import { parseTexturePacker } from "./parser/texturepacker.js"; +import { parseSpriteSheet } from "./parser/spritesheet.js"; /** * create a simple 1 frame texture atlas based on the given parameters @@ -107,9 +108,7 @@ export class TextureAtlas { this.sources.set(atlas.meta.image || "default", typeof src === "string" ? getImage(src) : src); } this.repeat = "no-repeat"; - } - // ShoeBox - else if (atlas.meta.app.includes("ShoeBox")) { + } else if (atlas.meta.app.includes("ShoeBox")) { if (!atlas.meta.exporter || !atlas.meta.exporter.includes("melonJS")) { throw new Error( "ShoeBox requires the JSON exporter : " + @@ -126,8 +125,9 @@ export class TextureAtlas { this.repeat = atlas.meta.repeat || "no-repeat"; this.sources.set("default", typeof src === "string" ? getImage(src) : src); } + // initialize the atlas - this.atlases.set(atlas.meta.image || "default", this.parse(atlas)); + this.atlases.set(atlas.meta.image || "default", parseTexturePacker(atlas, this)); } else { // a regular spritesheet @@ -141,7 +141,7 @@ export class TextureAtlas { atlas.image = typeof src === "string" ? getImage(src) : src; } // initialize the atlas - this.atlases.set("default", this.parseFromSpriteSheet(atlas)); + this.atlases.set("default", parseSpriteSheet(atlas, this)); this.sources.set("default", atlas.image); } @@ -162,119 +162,6 @@ export class TextureAtlas { } } - /** - * build an atlas from the given data - * @ignore - */ - parse(data) { - let atlas = {}; - - data.frames.forEach((frame) => { - // fix wrongly formatted JSON (e.g. last dummy object in ShoeBox) - if (frame.hasOwnProperty("filename")) { - // Source coordinates - let s = frame.frame; - let trimmed = !!frame.trimmed; - - let trim; - - if (trimmed) { - trim = { - x : frame.spriteSourceSize.x, - y : frame.spriteSourceSize.y, - w : frame.spriteSourceSize.w, - h : frame.spriteSourceSize.h - }; - } - - let originX, originY; - // Pixel-based offset origin from the top-left of the source frame - let hasTextureAnchorPoint = (frame.sourceSize && frame.pivot); - if (hasTextureAnchorPoint) { - originX = (frame.sourceSize.w * frame.pivot.x) - ((trimmed) ? trim.x : 0); - originY = (frame.sourceSize.h * frame.pivot.y) - ((trimmed) ? trim.y : 0); - } - - atlas[frame.filename] = { - name : frame.filename, // frame name - texture : data.meta.image || "default", // the source texture - offset : new Vector2d(s.x, s.y), - anchorPoint : (hasTextureAnchorPoint) ? new Vector2d(originX / s.w, originY / s.h) : null, - trimmed : trimmed, - trim : trim, - width : s.w, - height : s.h, - angle : (frame.rotated === true) ? -ETA : 0 - }; - this.addUVs(atlas, frame.filename, data.meta.size.w, data.meta.size.h); - } - }); - return atlas; - } - - /** - * build an atlas from the given spritesheet - * @ignore - */ - parseFromSpriteSheet(data) { - let atlas = {}; - let image = data.image; - let spacing = data.spacing || 0; - let margin = data.margin || 0; - - let width = image.width; - let height = image.height; - - // calculate the sprite count (line, col) - let spritecount = pool.pull("Vector2d", - ~~((width - margin + spacing) / (data.framewidth + spacing)), - ~~((height - margin + spacing) / (data.frameheight + spacing)) - ); - - // verifying the texture size - if ((width % (data.framewidth + spacing)) !== 0 || - (height % (data.frameheight + spacing)) !== 0) { - let computed_width = spritecount.x * (data.framewidth + spacing); - let computed_height = spritecount.y * (data.frameheight + spacing); - if (computed_width - width !== spacing && computed_height - height !== spacing) { - // "truncate size" if delta is different from the spacing size - width = computed_width; - height = computed_height; - // warning message - console.warn( - "Spritesheet Texture for image: " + image.src + - " is not divisible by " + (data.framewidth + spacing) + - "x" + (data.frameheight + spacing) + - ", truncating effective size to " + width + "x" + height - ); - } - } - - // build the local atlas - for (let frame = 0, count = spritecount.x * spritecount.y; frame < count; frame++) { - let name = "" + frame; - atlas[name] = { - name : name, - texture : "default", // the source texture - offset : new Vector2d( - margin + (spacing + data.framewidth) * (frame % spritecount.x), - margin + (spacing + data.frameheight) * ~~(frame / spritecount.x) - ), - anchorPoint : (data.anchorPoint || null), - trimmed : false, - trim : undefined, - width : data.framewidth, - height : data.frameheight, - angle : 0 - }; - this.addUVs(atlas, name, width, height); - } - - pool.push(spritecount); - - return atlas; - } - /** * return the default or specified atlas dictionnary * @param {string} [name] - atlas name in case of multipack textures diff --git a/src/video/texture/parser/spritesheet.js b/src/video/texture/parser/spritesheet.js new file mode 100644 index 000000000..17e38d92a --- /dev/null +++ b/src/video/texture/parser/spritesheet.js @@ -0,0 +1,68 @@ +import pool from "./../../../system/pooling.js"; +import Vector2d from "./../../../math/vector2.js"; + +/** + * parse the given data and return a corresponding atlas + * @param {Object} data - atlas data information. See {@link loader.getJSON} + * @param {TextureAtlas} textureAtlas - the texture atlas class calling the parser + * @returns {Object} the corresponding Atlas + * @ignore + */ +export function parseSpriteSheet(data, textureAtlas) { + let atlas = {}; + let image = data.image; + let spacing = data.spacing || 0; + let margin = data.margin || 0; + + let width = image.width; + let height = image.height; + + // calculate the sprite count (line, col) + let spritecount = pool.pull("Vector2d", + ~~((width - margin + spacing) / (data.framewidth + spacing)), + ~~((height - margin + spacing) / (data.frameheight + spacing)) + ); + + // verifying the texture size + if ((width % (data.framewidth + spacing)) !== 0 || + (height % (data.frameheight + spacing)) !== 0) { + let computed_width = spritecount.x * (data.framewidth + spacing); + let computed_height = spritecount.y * (data.frameheight + spacing); + if (computed_width - width !== spacing && computed_height - height !== spacing) { + // "truncate size" if delta is different from the spacing size + width = computed_width; + height = computed_height; + // warning message + console.warn( + "Spritesheet Texture for image: " + image.src + + " is not divisible by " + (data.framewidth + spacing) + + "x" + (data.frameheight + spacing) + + ", truncating effective size to " + width + "x" + height + ); + } + } + + // build the local atlas + for (let frame = 0, count = spritecount.x * spritecount.y; frame < count; frame++) { + let name = "" + frame; + atlas[name] = { + name : name, + texture : "default", // the source texture + offset : new Vector2d( + margin + (spacing + data.framewidth) * (frame % spritecount.x), + margin + (spacing + data.frameheight) * ~~(frame / spritecount.x) + ), + anchorPoint : (data.anchorPoint || null), + trimmed : false, + trim : undefined, + width : data.framewidth, + height : data.frameheight, + angle : 0 + }; + textureAtlas.addUVs(atlas, name, width, height); + } + + pool.push(spritecount); + + return atlas; +} diff --git a/src/video/texture/parser/texturepacker.js b/src/video/texture/parser/texturepacker.js new file mode 100644 index 000000000..3942b1ffd --- /dev/null +++ b/src/video/texture/parser/texturepacker.js @@ -0,0 +1,55 @@ +import { ETA } from "./../../../math/math.js"; +import Vector2d from "./../../../math/vector2.js"; + +/** + * parse the given data and return a corresponding atlas + * @param {Object} data - atlas data information. See {@link loader.getJSON} + * @param {TextureAtlas} textureAtlas - the texture atlas class calling the parser + * @returns {Object} the corresponding Atlas + * @ignore + */ +export function parseTexturePacker(data, textureAtlas) { + let atlas = {}; + + data.frames.forEach((frame) => { + // fix wrongly formatted JSON (e.g. last dummy object in ShoeBox) + if (frame.hasOwnProperty("filename")) { + // Source coordinates + let s = frame.frame; + let trimmed = !!frame.trimmed; + + let trim; + + if (trimmed) { + trim = { + x : frame.spriteSourceSize.x, + y : frame.spriteSourceSize.y, + w : frame.spriteSourceSize.w, + h : frame.spriteSourceSize.h + }; + } + + let originX, originY; + // Pixel-based offset origin from the top-left of the source frame + let hasTextureAnchorPoint = (frame.sourceSize && frame.pivot); + if (hasTextureAnchorPoint) { + originX = (frame.sourceSize.w * frame.pivot.x) - ((trimmed) ? trim.x : 0); + originY = (frame.sourceSize.h * frame.pivot.y) - ((trimmed) ? trim.y : 0); + } + + atlas[frame.filename] = { + name : frame.filename, // frame name + texture : data.meta.image || "default", // the source texture + offset : new Vector2d(s.x, s.y), + anchorPoint : (hasTextureAnchorPoint) ? new Vector2d(originX / s.w, originY / s.h) : null, + trimmed : trimmed, + trim : trim, + width : s.w, + height : s.h, + angle : (frame.rotated === true) ? -ETA : 0 + }; + textureAtlas.addUVs(atlas, frame.filename, data.meta.size.w, data.meta.size.h); + } + }); + return atlas; +}