Skip to content

Instantly share code, notes, and snippets.

@pixelrevision
Last active January 23, 2018 19:28
Show Gist options
  • Star 1 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save pixelrevision/d685a38128047139aee0 to your computer and use it in GitHub Desktop.
Save pixelrevision/d685a38128047139aee0 to your computer and use it in GitHub Desktop.
PixelLevelCollision.js
/*
Works with the alpha channel of an image to determine if there is collision with alpha
@parameter image - the image to check coordinate with.
@parameter sampling - (0.1-1.0) The percentage to downsample the image. Helps with performance and memory. Default 0.5
@parameter threshhold - (0.0-1.0) the threshhold to set the alpha for a hit. default 0.2
*/
var PixelLevelCollision = function(image, sampling, threshhold){
var self = this;
if(sampling === undefined){
sampling = 0.5;
}
if(sampling > 1.0){
sampling = 1.0;
}
if(sampling < 0.01){
sampling = 0.1;
}
if(threshhold === undefined){
threshhold = 0.2;
}
this.image = image;
this.sampling = sampling;
this.threshhold = threshhold;
this.canCreateMap = true;
this.image.onload = function(){
self.setupCanvas();
}
this.setupCanvas();
}
PixelLevelCollision.prototype.setupCanvas = function(){
this.canCreateMap = true;
this.baseWidth = Math.floor(this.image.width * this.sampling);
this.baseHeight = Math.floor(this.image.height * this.sampling);
var canvas = document.createElement("canvas");
canvas.width = this.baseWidth;
canvas.height = this.baseHeight;
var ctx = canvas.getContext("2d");
ctx.clearRect(0, 0, this.baseWidth, this.baseHeight);
try{
ctx.drawImage(this.image, 0, 0, this.baseWidth, this.baseHeight);
var pixelData = ctx.getImageData(0, 0, this.baseWidth, this.baseHeight);
}catch(e){
if(typeof console !== "undefined"){
console.log("!Warning could not get image data defaulting to always hit");
}
this.canCreateMap = false;
return;
}
this.alphaData = [];
var total = pixelData.data.length/4;
var inc = 255.0/total;
var start = 0.0;
for(var i=0; i<pixelData.data.length; i+=4){
this.alphaData.push(pixelData.data[i + 3]);
start += inc;
}
ctx.putImageData(pixelData, 0, 0);
}
/**
Checks if there is a collision with pixel at pointX and pointY from top left corner of the original image
@parameter pointX - (0-image width) The x point from the top left of the image to check.
@parameter pointY - (0-image height) The y point from the top left of the image to check.
*/
PixelLevelCollision.prototype.isCollided = function(pointX, pointY) {
if(!this.canCreateMap){
return true;
}
pointX = Math.floor(pointX * this.sampling);
pointY = Math.floor(pointY * this.sampling);
var pixel = (pointY * this.baseWidth) + pointX;
return this.alphaData[pixel] > this.threshhold;
};
<!DOCTYPE html>
<html>
<head>
<title></title>
<script src="http://code.jquery.com/jquery-1.11.2.min.js"></script>
<script src="PixelLevelCollision.js" type="text/javascript"></script>
<script type="text/javascript">
var plc;
$(document).ready(function(){
plc = new PixelLevelCollision(document.getElementById("sample_image"), 0.5);
$("#sample_image").mousemove(function(e){
var posX = e.pageX - $(this).offset().left;
var posY = e.pageY - $(this).offset().top;
if(plc.isCollided(posX, posY)){
$(".output").html("collided");
}else{
$(".output").html("not collided");
}
});
});
</script>
</head>
<body>
<div id="container">
<div class="output">info</div>
<img id="sample_image" width="150" height="187" src=""/>
</div>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment