mirror of
https://github.com/scratchfoundation/scratch-html5.git
synced 2025-01-07 04:31:56 -05:00
263 lines
8.7 KiB
JavaScript
263 lines
8.7 KiB
JavaScript
// Copyright (C) 2013 Massachusetts Institute of Technology
|
|
//
|
|
// This program is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU General Public License version 2,
|
|
// as published by the Free Software Foundation.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program; if not, write to the Free Software
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
'use strict';
|
|
|
|
var SensingPrims = function() {};
|
|
|
|
SensingPrims.prototype.addPrimsTo = function(primTable) {
|
|
primTable['touching:'] = this.primTouching;
|
|
primTable['touchingColor:'] = this.primTouchingColor;
|
|
primTable['color:sees:'] = this.primColorTouchingColor;
|
|
|
|
primTable['doAsk'] = this.primDoAsk;
|
|
primTable['answer'] = this.primAnswer;
|
|
|
|
primTable['keyPressed:'] = this.primKeyPressed;
|
|
primTable['mousePressed'] = function(b) { return runtime.mouseDown; };
|
|
primTable['mouseX'] = function(b) { return runtime.mousePos[0]; };
|
|
primTable['mouseY'] = function(b) { return runtime.mousePos[1]; };
|
|
primTable['distanceTo:'] = this.primDistanceTo;
|
|
|
|
primTable['getAttribute:of:'] = this.primGetAttribute;
|
|
|
|
primTable['timeAndDate'] = function(b){ return runtime.getTimeString(interp.arg(b, 0)); };
|
|
primTable['timestamp'] = this.primTimestamp;
|
|
};
|
|
|
|
SensingPrims.prototype.primTouching = function(b) {
|
|
var s = interp.targetSprite();
|
|
if (s == null || !s.visible) return false;
|
|
|
|
var arg = interp.arg(b, 0);
|
|
if (arg == '_edge_') {
|
|
return false; // TODO
|
|
}
|
|
|
|
if (arg == '_mouse_') {
|
|
return false; // TODO
|
|
}
|
|
|
|
var s2 = runtime.spriteNamed(arg);
|
|
if (s2 == null || !s2.visible) return false;
|
|
|
|
return spriteHitTest(s, s2);
|
|
};
|
|
|
|
SensingPrims.prototype.primTouchingColor = function(b) {
|
|
var s = interp.targetSprite();
|
|
if (s == null || !s.visible) return false;
|
|
|
|
var color = interp.arg(b, 0);
|
|
|
|
return stageColorHitTest(s, color);
|
|
};
|
|
|
|
SensingPrims.prototype.primColorTouchingColor = function(b) {
|
|
var s = interp.targetSprite();
|
|
if (s == null || !s.visible) return false;
|
|
|
|
var myColor = interp.arg(b, 0);
|
|
var stageColor = interp.arg(b, 1);
|
|
|
|
return stageColorByColorHitTest(s, myColor, stageColor);
|
|
};
|
|
|
|
var spriteHitTest = function(a, b) {
|
|
var hitCanvas = document.createElement('canvas');
|
|
hitCanvas.width = 480;
|
|
hitCanvas.height = 360;
|
|
var hitTester = hitCanvas.getContext('2d');
|
|
hitTester.globalCompositeOperation = 'source-over';
|
|
a.stamp(hitTester, 100);
|
|
hitTester.globalCompositeOperation = 'source-in';
|
|
b.stamp(hitTester, 100);
|
|
|
|
var aData = hitTester.getImageData(0, 0, 480, 360).data;
|
|
|
|
var pxCount = aData.length;
|
|
for (var i = 0; i < pxCount; i += 4) {
|
|
if (aData[i+3] > 0) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
|
|
var stageColorHitTest = function(target, color) {
|
|
var r, g, b;
|
|
r = (color >> 16);
|
|
g = (color >> 8 & 255);
|
|
b = (color & 255);
|
|
|
|
var targetCanvas = document.createElement('canvas');
|
|
targetCanvas.width = 480;
|
|
targetCanvas.height = 360;
|
|
var targetTester = targetCanvas.getContext('2d');
|
|
target.stamp(targetTester, 100);
|
|
|
|
var stageCanvas = document.createElement('canvas');
|
|
stageCanvas.width = 480;
|
|
stageCanvas.height = 360;
|
|
var stageContext = stageCanvas.getContext('2d');
|
|
|
|
$.each(runtime.sprites, function(i, sprite) {
|
|
if (sprite != target)
|
|
sprite.stamp(stageContext, 100);
|
|
});
|
|
|
|
var hitData = stageContext.getImageData(0, 0, stageCanvas.width, stageCanvas.height).data;
|
|
var meshData = targetTester.getImageData(0, 0, targetCanvas.width, targetCanvas.height).data;
|
|
var pxCount = meshData.length;
|
|
for (var i = 0; i < pxCount; i += 4) {
|
|
if (meshData[i+3] > 0 && hitData[i] == r && hitData[i+1] == g && hitData[i+2] == b)
|
|
return true;
|
|
}
|
|
return false;
|
|
};
|
|
|
|
var stageColorByColorHitTest = function(target, myColor, otherColor) {
|
|
var threshold_acceptable = function(a, b, c, x, y, z) {
|
|
var diff_a = Math.abs(a-x);
|
|
var diff_b = Math.abs(b-y);
|
|
var diff_c = Math.abs(c-z);
|
|
if (diff_a + diff_b + diff_c < 100) {
|
|
return true;
|
|
}
|
|
return false;
|
|
};
|
|
var targetCanvas = document.createElement('canvas');
|
|
targetCanvas.width = 480;
|
|
targetCanvas.height = 360;
|
|
var targetTester = targetCanvas.getContext('2d');
|
|
target.stamp(targetTester, 100);
|
|
var targetData = targetTester.getImageData(0, 0, targetCanvas.width, targetCanvas.height).data;
|
|
|
|
// Calculate RGB values of the colors - TODO thresholding
|
|
//myColor = Math.abs(myColor);
|
|
//otherColor = Math.abs(otherColor);
|
|
var mr, mg, mb, or, og, ob;
|
|
mr = (myColor >> 16);
|
|
mg = (myColor >> 8 & 255);
|
|
mb = (myColor & 255);
|
|
or = (otherColor >> 16);
|
|
og = (otherColor >> 8 & 255);
|
|
ob = (otherColor & 255);
|
|
|
|
// Create the hit canvas for comparison
|
|
var hitCanvas = document.createElement('canvas');
|
|
hitCanvas.width = 480;
|
|
hitCanvas.height = 360;
|
|
var hitCtx = hitCanvas.getContext('2d');
|
|
$.each(runtime.sprites, function(i, sprite) {
|
|
if (sprite != target) {
|
|
sprite.stamp(hitCtx, 100);
|
|
}
|
|
});
|
|
|
|
var hitData = hitCtx.getImageData(0, 0, hitCanvas.width, hitCanvas.height).data;
|
|
var pxCount = targetData.length;
|
|
for (var i = 0; i < pxCount; i += 4) {
|
|
if (threshold_acceptable(targetData[i], targetData[i+1], targetData[i+2], mr, mg, mb) && threshold_acceptable(hitData[i], hitData[i+1], hitData[i+2], or, og, ob)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
|
|
SensingPrims.prototype.primDoAsk= function(b) {
|
|
showBubble(b, "doAsk");
|
|
var s = interp.targetSprite();
|
|
if (s !== null) {
|
|
interp.activeThread.paused = true;
|
|
s.showAsk();
|
|
}
|
|
};
|
|
|
|
SensingPrims.prototype.primAnswer = function(b) {
|
|
var s = interp.targetStage();
|
|
return (s !== null ? s.askAnswer : undefined);
|
|
};
|
|
|
|
|
|
SensingPrims.prototype.primKeyPressed = function(b) {
|
|
var key = interp.arg(b, 0);
|
|
var ch = key.charCodeAt(0);
|
|
if (ch > 127) return false;
|
|
if (key == "left arrow") ch = 37;
|
|
if (key == "right arrow") ch = 39;
|
|
if (key == "up arrow") ch = 38;
|
|
if (key == "down arrow") ch = 40;
|
|
if (key == "space") ch = 32;
|
|
return (typeof(runtime.keysDown[ch]) != 'undefined');
|
|
};
|
|
|
|
SensingPrims.prototype.primDistanceTo = function(b) {
|
|
var s = interp.targetSprite();
|
|
var p = mouseOrSpritePosition(interp.arg(b, 0));
|
|
if (s == null || p == null) return 0;
|
|
var dx = p.x - s.scratchX;
|
|
var dy = p.y - s.scratchY;
|
|
return Math.sqrt((dx * dx) + (dy * dy));
|
|
};
|
|
|
|
SensingPrims.prototype.primGetAttribute = function(b) {
|
|
var attr = interp.arg(b, 0);
|
|
var targetSprite = runtime.spriteNamed(interp.arg(b, 1));
|
|
if (targetSprite == null) return 0;
|
|
if (attr == 'x position') return targetSprite.scratchX;
|
|
if (attr == 'y position') return targetSprite.scratchY;
|
|
if (attr == 'direction') return targetSprite.direction;
|
|
if (attr == 'costume #') return targetSprite.currentCostumeIndex + 1;
|
|
if (attr == 'costume name') return targetSprite.costumes[targetSprite.currentCostumeIndex]['costumeName'];
|
|
if (attr == 'size') return targetSprite.getSize();
|
|
if (attr == 'volume') return targetSprite.volume;
|
|
return 0;
|
|
};
|
|
|
|
SensingPrims.prototype.primTimeDate = function(b) {
|
|
var dt = interp.arg(b, 0);
|
|
var now = new Date();
|
|
if (dt == 'year') return now.getFullYear();
|
|
if (dt == 'month') return now.getMonth()+1;
|
|
if (dt == 'date') return now.getDate();
|
|
if (dt == 'day of week') return now.getDay()+1;
|
|
if (dt == 'hour') return now.getHours();
|
|
if (dt == 'minute') return now.getMinutes();
|
|
if (dt == 'second') return now.getSeconds();
|
|
return 0;
|
|
};
|
|
|
|
SensingPrims.prototype.primTimestamp = function(b) {
|
|
var now = new Date();
|
|
var epoch = new Date(2000, 0, 1);
|
|
var dst = now.getTimezoneOffset() - epoch.getTimezoneOffset();
|
|
var msSince = now.getTime() - epoch.getTime();
|
|
msSince += (now.getTimezoneOffset() - dst) * 60000;
|
|
return msSince / 86400000;
|
|
};
|
|
|
|
// Helpers
|
|
SensingPrims.prototype.mouseOrSpritePosition = function(arg) {
|
|
if (arg == "_mouse_") {
|
|
var w = runtime.stage;
|
|
return new Point(runtime.mousePos[0], runtime.mousePos[1]);
|
|
} else {
|
|
var s = runtime.spriteNamed(arg);
|
|
if (s == null) return null;
|
|
return new Point(s.scratchX, s.scratchY);
|
|
}
|
|
return null;
|
|
};
|