<html lang="en">
<head>
<title>Extract and draw sprite</title>
<meta name="description" content="Extract and draw sprite">
<style>
canvas {
border: 1px solid black;
}
</style>
</head>
<body>
<p>Sprite width: 48, height: 92, rows: 8, sprites per posture: 13</p>
<label for="x">x: </label><input id="x" type="number" min="0"><br/>
<label for="y">y: </label><input id="y" type="number" min="0"><br/>
<label for="width">width: </label><input id="width" type="number" min="0"><br/>
<label for="height">height: </label><input id="height" type="number" min="0">
<p>Select current sprite: <input type="range" id="spriteSelect" value="0">
<output id="spriteNumber"></output>
</p>
<p><canvas id="canvas" width="48" height="92" /></p>
<p><canvas id="spritesheet"></canvas></p>
</body>
</html>
var SPRITESHEET_URL = "https://i.imgur.com/3VesWqx.png";
var SPRITE_WIDTH = 48;
var SPRITE_HEIGHT = 92;
var NB_POSTURES=8;
var NB_FRAMES_PER_POSTURE = 13;
var xField, yField, wField, hField, spriteSelect, spriteNumber;
var canvas, canvasSpriteSheet, ctx1, ctx2;
window.onload = function() {
canvas = document.getElementById("canvas");
ctx1 = canvas.getContext("2d");
canvasSpriteSheet = document.getElementById("spritesheet");
ctx2 = canvasSpriteSheet.getContext("2d");
xField = document.querySelector("#x");
yField = document.querySelector("#y");
wField = document.querySelector("#width");
hField = document.querySelector("#height");
spriteSelect = document.querySelector("#spriteSelect");
spriteNumber = document.querySelector("#spriteNumber");
wField.value = SPRITE_WIDTH;
hField.value = SPRITE_HEIGHT;
xField.value = 0;
yField.value = 0;
spriteSelect.min = 0;
spriteSelect.max=NB_POSTURES*NB_FRAMES_PER_POSTURE - 1;
spriteSelect.disabled = true;
spriteNumber.innerHTML=0;
// load the spritesheet
spritesheet = new Image();
spritesheet.src = SPRITESHEET_URL;
// Called when the spritesheet has been loaded
spritesheet.onload = function() {
// enable slider
spriteSelect.disabled = false;
// Resize big canvas to the size of the spritesheet image
canvasSpriteSheet.width = spritesheet.width;
canvasSpriteSheet.height = spritesheet.height;
// Resize small canvas to the size of the spritesheet image
canvas.width = SPRITE_WIDTH;
canvas.height = SPRITE_HEIGHT;
// Draw the whole spritesheet
ctx2.drawImage(spritesheet, 0, 0);
// Draw the first sprite in the small canvas, corresponding to sprite 0
// wireframe rectangle in the sprite sheet
drawWireFrameRect(ctx2, 0 , 0, SPRITE_WIDTH, SPRITE_HEIGHT, 'red', 3);
// small canvas, draw subimage corresponding to sprite 0
ctx1.drawImage(spritesheet, 0, 0, SPRITE_WIDTH, SPRITE_HEIGHT, 0, 0, SPRITE_WIDTH, SPRITE_HEIGHT);
};
// input listener on the
spriteSelect.oninput = function(evt) {
// Current sprite number from 0 to NB_FRAMES_PER_POSTURE * NB_ROWS
var index = spriteSelect.value;
var nbSpritesPerRow = Math.floor(spritesheet.width / SPRITE_WIDTH);
// Computation of the x and y position that corresponds to the sprite
// number index
// x is the rest of index/nbSpritesPerRow * with of a sprite
var x = (index % nbSpritesPerRow) * SPRITE_WIDTH;
// y is the divisor of index/nbSpritePerRow * height of a sprite
var y = Math.floor(index / nbSpritesPerRow) * SPRITE_HEIGHT;
// Did we go further than the last posture on a single row ?
if(x+SPRITE_WIDTH > spritesheet.width) {
// yes, go back at the beginning of the "line"
x = 0;
}
// update fields
xField.value = x;
yField.value = y;
// Clear big canvas, draw wireframe rect at x, y, redraw stylesheet
ctx2.clearRect(0, 0, canvasSpriteSheet.width, canvasSpriteSheet.height);
ctx2.drawImage(spritesheet, 0, 0);
drawWireFrameRect(ctx2, x , y, SPRITE_WIDTH, SPRITE_HEIGHT, 'red', 3);
// Draw current sprite in the small canvas
ctx1.clearRect(0, 0, SPRITE_WIDTH, SPRITE_HEIGHT);
ctx1.drawImage(spritesheet, x, y, SPRITE_WIDTH, SPRITE_HEIGHT,
0, 0, SPRITE_WIDTH, SPRITE_HEIGHT);
// Update output elem on the right of the slider
spriteNumber.innerHTML = index;
};
};
function drawWireFrameRect(ctx, x, y, w, h, color, lineWidth) {
ctx.save();
ctx.strokeStyle = color;
ctx.lineWidth = lineWidth;
ctx.strokeRect(x , y, w, h);
ctx.restore();
}
Output
300px
You can jump to the latest bin by adding /latest
to your URL
Keyboard Shortcuts
Shortcut | Action |
---|---|
ctrl + [num] | Toggle nth panel |
ctrl + 0 | Close focused panel |
ctrl + enter | Re-render output. If console visible: run JS in console |
Ctrl + l | Clear the console |
ctrl + / | Toggle comment on selected lines |
ctrl + ] | Indents selected lines |
ctrl + [ | Unindents selected lines |
tab | Code complete & Emmet expand |
ctrl + shift + L | Beautify code in active panel |
ctrl + s | Save & lock current Bin from further changes |
ctrl + shift + s | Open the share options |
ctrl + y | Archive Bin |
Complete list of JS Bin shortcuts |
JS Bin URLs
URL | Action |
---|---|
/ | Show the full rendered output. This content will update in real time as it's updated from the /edit url. |
/edit | Edit the current bin |
/watch | Follow a Code Casting session |
/embed | Create an embeddable version of the bin |
/latest | Load the very latest bin (/latest goes in place of the revision) |
/[username]/last | View the last edited bin for this user |
/[username]/last/edit | Edit the last edited bin for this user |
/[username]/last/watch | Follow the Code Casting session for the latest bin for this user |
/quiet | Remove analytics and edit button from rendered output |
.js | Load only the JavaScript for a bin |
.css | Load only the CSS for a bin |
Except for username prefixed urls, the url may start with http://jsbin.com/abc and the url fragments can be added to the url to view it differently. |