sandboxels/mods/pressure.js

223 lines
8.7 KiB
JavaScript

runAfterLoad(function(){
tryMove = function(pixel,nx,ny,leaveBehind,force) {
if (pixel.drag && !force) { return true; }
var info = elements[pixel.element];
var oob = outOfBounds(nx,ny);
if (isEmpty(nx,ny,false,oob)) { // If coords is empty, move to coords
movePixel(pixel,nx,ny,leaveBehind);
return true;
}
else if (!oob) {
// Reactions
newPixel = pixelMap[nx][ny];
if (info.density !== undefined) {
newPixel.pressure = info.density/5000 + (pixel.pressure ? pixel.pressure : 0);
}
var rr1 = false;
if (info.reactions !== undefined && info.reactions[newPixel.element] !== undefined) {
rr1 = reactPixels(pixel,newPixel)
if (rr1) {
return true;
}
}
if (!rr1 && elements[newPixel.element].reactions !== undefined && elements[newPixel.element].reactions[pixel.element] !== undefined && !elements[newPixel.element].reactions[pixel.element].oneway) {
if (reactPixels(newPixel,pixel)) {
return true;
}
}
// Density
if (elements[pixel.element].id !== elements[newPixel.element].id) {
if (info.density !== undefined && elements[newPixel.element].density !== undefined) {
// if the pixel's state + ">" + newPixel's state is in validDensitySwaps, and the pixel's density is larger than the newPixel's density, swap the pixels
if (validDensitySwaps[info.state][elements[newPixel.element].state] && info.density >= elements[newPixel.element].density) {
// chance depending on the difference in density
if (Math.random() < (info.density - elements[newPixel.element].density)/(info.density + elements[newPixel.element].density)) {
swapPixels(pixel,newPixel);
return true;
}
}
}
}
// else { // same-element density swapping
// if (info.density !== undefined) {
// if (validDensitySwaps[info.state][info.state]) {
// if (Math.random() < 0.01) {
// swapPixels(pixel,newPixel);
// return true;
// }
// }
// }
// }
}
return false;
}
doVelocity = function(pixel) {
if ((pixel.vx||pixel.vy) && elements[pixel.element].movable) {
if (pixel.vx) {
// move the pixel vx times
for (var i = 0; i < Math.floor(Math.abs(pixel.vx)); i++) {
var x = pixel.x+Math.sign(pixel.vx);
var y = pixel.y;
if (!tryMove(pixel,x,y)) {
// if (!isEmpty(x,y,true)) {
// var newPixel = pixelMap[x][y];
// if (elements[newPixel.element].movable) {
// newPixel.vx = (newPixel.vx||0) + pixel.vx - Math.sign(pixel.vx);
// if (elements[pixel.element].breakInto && Math.random()<elements[pixel.element].breakIntoChance) {
// changePixel(pixel,elements[pixel.element].breakInto);
// }
// }
// }
pixel.vx = -pixel.vx/2;
return;
}
else
{
pixel.pressure = 0;
}
if(pixel.del)
{
return;
}
}
if (pixel.vx) { pixel.vx = Math.abs(pixel.vx)/1.25*Math.sign(pixel.vx) }
}
if (pixel.vy) {
// move the pixel vy times
for (var i = 0; i < Math.floor(Math.abs(pixel.vy)); i++) {
var x = pixel.x;
var y = pixel.y+Math.sign(pixel.vy);
if (!tryMove(pixel,x,y)) {
// if (!isEmpty(x,y,true)) {
// var newPixel = pixelMap[x][y];
// if (elements[newPixel.element].movable) {
// newPixel.vy = (newPixel.vy||0) + pixel.vy - Math.sign(pixel.vy);
// if (elements[pixel.element].breakInto && Math.random()<elements[pixel.element].breakIntoChance) {
// changePixel(pixel,elements[pixel.element].breakInto);
// }
// }
// }
pixel.vy = -pixel.vy/2;
return;
}
else
{
pixel.pressure = 0;
}
if(pixel.del)
{
return;
}
}
if (pixel.vy) { pixel.vy = Math.abs(pixel.vy)/1.25*Math.sign(pixel.vy) }
}
}
}
runPerPixel(function(pixel) {
doVelocity(pixel);
pixel.prevX = pixel.x;
pixel.prevY = pixel.y;
})
runEveryTick(function() {
for (var i = 0; i < currentPixels.length; i++) {
pixel = currentPixels[i];
let density = elements[pixel.element].density ? elements[pixel.element].density : 1;
let accelx = (pixel.x - pixel.prevX)/2 * (1+(pixel.pressure ? pixel.pressure : 0)/density);
let accely = (pixel.y - pixel.prevY)/2 * (1+(pixel.pressure ? pixel.pressure : 0)/density);
if(!pixel.vx)
pixel.vx = accelx;
else
pixel.vx += accelx;
if(!pixel.vy)
pixel.vy = accely;
else
pixel.vy += accely;
}
})
})
explodeAt = function(x,y,radius,fire="fire") {
// if fire contains , split it into an array
if (fire.indexOf(",") !== -1) {
fire = fire.split(",");
}
var coords = circleCoords(x,y,radius);
var power = radius/10;
//for (var p = 0; p < Math.round(radius/10+1); p++) {
for (var i = 0; i < coords.length; i++) {
// damage value is based on distance from x and y
var damage = Math.random() + (Math.floor(Math.sqrt(Math.pow(coords[i].x-x,2) + Math.pow(coords[i].y-y,2)))) / radius;
// invert
damage = 1 - damage;
if (damage < 0) { damage = 0; }
damage *= power;
if (isEmpty(coords[i].x,coords[i].y)) {
// create smoke or fire depending on the damage if empty
if (damage < 0.02) { } // do nothing
else if (damage < 0.2) {
createPixel("smoke",coords[i].x,coords[i].y);
}
else {
// if fire is an array, choose a random item
if (Array.isArray(fire)) {
createPixel(fire[Math.floor(Math.random() * fire.length)],coords[i].x,coords[i].y);
}
else {
createPixel(fire,coords[i].x,coords[i].y);
}
}
}
else if (!outOfBounds(coords[i].x,coords[i].y)) {
// damage the pixel
var pixel = pixelMap[coords[i].x][coords[i].y];
var info = elements[pixel.element];
if (info.hardness) { // lower damage depending on hardness(0-1)
if (info.hardness < 1) {
damage = damage * ((1 - info.hardness)*10);
}
else { damage = 0; }
}
if (damage > 0.9) {
if (Array.isArray(fire)) {
var newfire = fire[Math.floor(Math.random() * fire.length)];
}
else {
var newfire = fire;
}
changePixel(pixel,newfire);
continue;
}
else if (damage > 0.25) {
if (info.breakInto !== undefined) {
breakPixel(pixel);
continue;
}
else {
if (Array.isArray(fire)) {
var newfire = fire[Math.floor(Math.random() * fire.length)];
}
else {
var newfire = fire;
}
changePixel(pixel,newfire);
continue;
}
}
if (damage > 0.75 && info.burn) {
pixel.burning = true;
pixel.burnStart = pixelTicks;
}
pixel.temp += damage*radius*power;
pixelTempCheck(pixel);
// set the pixel.vx and pixel.vy depending on the angle and power
if (!elements[pixel.element].excludeRandom) {
var angle = Math.atan2(pixel.y-y,pixel.x-x);
pixel.vx = Math.round((pixel.vx|0) + Math.cos(angle) * (radius * power));
pixel.vy = Math.round((pixel.vy|0) + Math.sin(angle) * (radius * power));
}
}
}
}