Quantcast
Channel: Forest - A Simulated Ecosystem - Code Golf Stack Exchange
Viewing all articles
Browse latest Browse all 5

Answer by Kevin L for Forest - A Simulated Ecosystem

$
0
0

Javascript

I think this mostly works. There's some wonky behavior where I spawn all the new bears/lumberjacks in sync and right next to each other because laziness in insertions.

This implementation does not allow lumberjacks to stand on saplings, cause you know, trampling saplings is bad. Fiddle art uses colored rectangles by default, change the second line to false to use letters to draw.

fiddle

HTML:

<canvas id="c" width="1" height="1"></canvas><div id="p1"></div><div id="p2"></div><div id="p3"></div><div id="p4"></div>

Js:

var n = 10; // Size of the gridvar drawUsingColor = true; // If true, draws colors for each entity instead :Dvar intervalTime = 1000; // how often each tick happens, in millisecondsvar jackRatio = 0.1;var treeRatio = 0.5;var bearRatio = 0.02;var size = 48; // Pixels allocated (in size x size) for each entityvar font = "30px Lucida Console"; // if drawUsingColor is falsevar bearColor = '#8B4513'; // Saddlebrownvar elderColor = '#556B2F'; // DarkOliveGreenvar lumberjackColor = '#B22222'; // Firebrickvar treeColor = '#008000'; // Greenvar saplingColor = '#ADFF2F'; // GreenYellow// Game rules:var spawnSaplingChance = 0.1;var elderTreeAge = 120;var elderSaplingChance = 0.2;var treeAge = 12;var lumberjackRange = 3;var bearRange = 5;var zooPeriod = 12; // If a maul happens within this periodvar lumberPeriod = 12; // New lumberjacks hired in this periodvar time = 1;var world;var n2 = n * n; //because one saved keystrokevar zooqueue = [];var lumberqueue = [];var canvas = document.getElementById('c'); // Needs more jqueryvar context = canvas.getContext('2d');context.font = font;// various statisticsvar treesAlive = 0;var jacksAlive = 0;var bearsAlive = 0;var currentLumber = 0;var lumberjacksMauled = 0;var recentEvents = '';// Entity is a bear, eldertree, lumberjack, tree, sapling, with age. aka belts.function Entity(belts, birthday) {    this.type = belts;    this.age = 0;    this.birthday = birthday;}function initWorld() {    canvas.height = size * n;    canvas.width = size * n;    world = new Array(n2);    // One pass spawning algorithm: numEntity = number of entity left to spawn    // If rand() in range [0,numtrees), spawn tree    // if rand() in range [numtrees, numtrees+numjacks), spawn lumberjack    // if rand() in range [numtrees+numjacks, numtrees+numjacks+numbears), spawn bear    var numTrees = treeRatio * n2;    var numJacks = jackRatio * n2;    var numBears = bearRatio * n2;    var godseed = new Array(n2);    for (var i = 0; i < n2; i++) {        godseed[i] = i;    }    shuffle(godseed);    for (var i = 0; i < n2; i++) {        var god = godseed.pop();        if (god < numTrees) {            world[i] = new Entity('T', 0);            treesAlive++;        } else if (god < numTrees + numJacks) {            world[i] = new Entity('L', 0);            jacksAlive++;        } else if (god < numTrees + numJacks + numBears) {            world[i] = new Entity('B', 0);            bearsAlive++;        }        // console.log(world, i);    }    // populate zoo array, lumber array    for (var i = 0; i < zooPeriod; i++) {        zooqueue.push(0);    }    for (var i = 0; i < lumberPeriod; i++) {        lumberqueue.push(0);    }}animateWorld = function () {    recentEvents = '';    computeWorld();    drawWorld();    time++;    $('#p1').text(treesAlive +' trees alive');    $('#p2').text(bearsAlive +' bears alive');    $('#p3').text(jacksAlive +' lumberjacks alive');    $('#p4').text(recentEvents);};function computeWorld() {    zooqueue.push(lumberjacksMauled);    lumberqueue.push(currentLumber);    // Calculate entity positions    for (var i = 0; i < n2; i++) {        if (world[i]) {            switch (world[i].type) {                case 'B':                    bearStuff(i);                    break;                case 'E':                    elderStuff(i);                    break;                case 'L':                    lumberjackStuff(i);                    break;                case 'T':                    treeStuff(i);                    break;                case 'S':                    saplingStuff(i);                    break;            }        }    }    // Pop the # mauls from zooPeriod's ago, if lumberjacksMauled > oldmauls, then someone was eaten.    var oldmauls = zooqueue.shift();    if (time % zooPeriod === 0) {        if (lumberjacksMauled > oldmauls) {            if (remove('B') == 1) {                bearsAlive--;                recentEvents += 'Bear sent to zoo! ';            }        } else {            bearsAlive++;            spawn('B');            recentEvents += 'New bear appeared! ';        }    }    var oldLumber = lumberqueue.shift();    if (time % lumberPeriod === 0) {        // # lumberjack to hire        var hire = Math.floor((currentLumber - oldLumber) / jacksAlive);        if (hire > 0) {            recentEvents += 'Lumber jack hired! ('+ hire +') ';            while (hire > 0) {                jacksAlive++;                spawn('L');                hire--;            }        } else {            if (remove('L') == 1) {                recentEvents += 'Lumber jack fired!  ';                jacksAlive--;            }            else {            }        }    }    // Ensure > 1 lumberjack    if (jacksAlive === 0) {        jacksAlive++;        spawn('L');        recentEvent += 'Lumberjack spontaneously appeared';    }}// Not the job of spawn/remove to keep track of whatever was spawned/removedfunction spawn(type) {    var index = findEmpty(type);    if (index != -1) {        world[index] = new Entity(type, time);    }    // recentEvents += 'Spawned a '+ type +'\n';}function remove(type) {    var index = findByType(type);    if (index != -1) {        world[index] = null;        return 1;    }    return -1;    // recentEvents += 'Removed a '+ type +'\n';}// Searches in world for an entity with type=type. Currently implemented as// linear scan, which isn't very randomfunction findByType(type) {    for (var i = 0; i < n2; i++) {        if (world[i] && world[i].type == type) return i;    }    return -1;}// Also linear scan function findEmpty(type) {    for (var i = 0; i < n2; i++) {        if (!world[i]) {            return i;        }    }    return -1;}function bearStuff(index) {    if (world[index].birthday == time) {        return;    }    // Wander around    var tindex = index;    for (var i = 0; i < lumberjackRange; i++) {        var neighbors = get8Neighbor(tindex);        var mov = neighbors[Math.floor(Math.random() * neighbors.length)];        if (world[mov] && world[mov].type == 'L') {            recentEvents += 'Bear ('+ index % 10 +','+ Math.floor(index / 10) +') mauled a Lumberjack ('+ mov % 10 +','+ Math.floor(mov / 10) +') !';            lumberjacksMauled++;            jacksAlive--;            world[mov] = new Entity('B', time);            world[mov].age = ++world[index].age;            world[index] = null;            return;        }        tindex = mov;    }    if (!world[tindex]) {        world[tindex] = new Entity('B', time);        world[tindex].age = ++world[index].age;        world[index] = null;    }}function elderStuff(index) {    if (world[index].birthday == time) {        return;    }    neighbors = get8Neighbor(index);    // spawn saplings    for (var i = 0; i < neighbors.length; i++) {        if (!world[neighbors[i]]) {            if (Math.random() < elderSaplingChance) {                world[neighbors[i]] = new Entity('S', time);                treesAlive++;            }        }    }    // become older    world[index].age++;}function lumberjackStuff(index) {    if (world[index].birthday == time) {        return;    }    // Wander around    var tindex = index;    for (var i = 0; i < lumberjackRange; i++) {        var neighbors = get8Neighbor(tindex);        var mov = neighbors[Math.floor(Math.random() * neighbors.length)];        if (world[mov] && (world[mov].type == 'T' || world[mov].type == 'E')) {            world[mov].type == 'T' ? currentLumber++ : currentLumber += 2;            treesAlive--;            world[mov] = new Entity('L', time);            world[mov].age = ++world[index].age;            world[index] = null;            return;        }        tindex = mov;    }    if (!world[tindex]) {        world[tindex] = new Entity('L', time);        world[tindex].age = ++world[index].age;        world[index] = null;    }}function treeStuff(index) {    if (world[index].birthday == time) {        return;    }    neighbors = get8Neighbor(index);    // spawn saplings    for (var i = 0; i < neighbors.length; i++) {        if (!world[neighbors[i]]) {            if (Math.random() < spawnSaplingChance) {                world[neighbors[i]] = new Entity('S', time);                treesAlive++;            }        }    }    // promote to elder tree?    if (world[index].age >= elderTreeAge) {        world[index] = new Entity('E', time);        return;    }    // become older    world[index].age++;}function saplingStuff(index) {    if (world[index].birthday == time) {        return;    }    // promote to tree?    if (world[index].age > treeAge) {        world[index] = new Entity('T', time);        return;    }    world[index].age++;}// Returns array containing up to 8 valid neighbors.// Prolly gonna break for n < 3 but oh wellfunction get8Neighbor(index) {    neighbors = [];    if (index % n != 0) {        neighbors.push(index - n - 1);        neighbors.push(index - 1);        neighbors.push(index + n - 1);    }    if (index % n != n - 1) {        neighbors.push(index - n + 1);        neighbors.push(index + 1);        neighbors.push(index + n + 1);    }    neighbors.push(index - n);    neighbors.push(index + n);    return neighbors.filter(function (val, ind, arr) {        return (0 <= val && val < n2)    });}// Each entity allocated 5x5px for their artfunction drawWorld() {    context.clearRect(0, 0, canvas.width, canvas.height);    for (var i = 0; i < n2; i++) {        if (world[i]) {            var x = i % n;            var y = Math.floor(i / n);            switch (world[i].type) {                case 'B':                    drawBear(x, y);                    break;                case 'E':                    drawElder(x, y);                    break;                case 'L':                    drawJack(x, y);                    break;                case 'T':                    drawTree(x, y);                    break;                case 'S':                    drawSapling(x, y);                    break;            }        }    }}function drawBear(x, y) {    if (drawUsingColor) {        drawRect(x * size, y * size, size, size, bearColor);    } else {        drawLetter(x * size, y * size, 'B');    }}function drawElder(x, y) {    if (drawUsingColor) {        drawRect(x * size, y * size, size, size, elderColor);    } else {        drawLetter(x * size, y * size, 'E');    }}function drawJack(x, y) {    if (drawUsingColor) {        drawRect(x * size, y * size, size, size, lumberjackColor);    } else {        drawLetter(x * size, y * size, 'J');    }}function drawTree(x, y) {    if (drawUsingColor) {        drawRect(x * size, y * size, size, size, treeColor);    } else {        drawLetter(x * size, y * size, 'T');    }}function drawSapling(x, y) {    if (drawUsingColor) {        drawRect(x * size, y * size, size, size, saplingColor);    } else {        drawLetter(x * size, y * size, 'S');    }}function drawLine(x1, y1, x2, y2, c) {    context.beginPath();    context.moveTo(x1, y1);    context.lineTo(x2, y2);    context.lineWidth = 3;    context.strokeStyle = c;    context.stroke();}function drawRect(x, y, w, h, c) {    context.fillStyle = c;    context.fillRect(x, y, w, h);}function drawLetter(x, y, l) {    context.fillText(l, x, y);}$(document).ready(function () {    initWorld();    intervalID = window.setInterval(animateWorld, intervalTime);    /*$('#s').click(function() {        animateWorld();    })*/});// http://stackoverflow.com/questions/2450954/how-to-randomize-shuffle-a-javascript-arrayfunction shuffle(array) {    var currentIndex = array.length,        temporaryValue, randomIndex;    // While there remain elements to shuffle...    while (0 !== currentIndex) {        // Pick a remaining element...        randomIndex = Math.floor(Math.random() * currentIndex);        currentIndex -= 1;        // And swap it with the current element.         temporaryValue = array[currentIndex];        array[currentIndex] = array[randomIndex];        array[randomIndex] = temporaryValue;    }    return array;}

Viewing all articles
Browse latest Browse all 5

Trending Articles



<script src="https://jsc.adskeeper.com/r/s/rssing.com.1596347.js" async> </script>