forked from: 4K Dungeon Romp Game
forked from 4K Dungeon Romp Game (diff: 1)
ActionScript3 source code
/**
* Copyright michael.tessin ( http://wonderfl.net/user/michael.tessin )
* MIT License ( http://www.opensource.org/licenses/mit-license.php )
* Downloaded from: http://wonderfl.net/c/l1ah
*/
// forked from Zahurdias.Calimero's 4K Dungeon Romp Game
package {
import flash.events.KeyboardEvent;
import flash.events.Event;
import flash.geom.Matrix;
import flash.geom.ColorTransform;
import flash.text.TextFormat;
import flash.text.TextField;
import flash.geom.Rectangle;
import flash.display.BitmapData;
import flash.display.Bitmap;
import flash.display.Sprite;
/**
* @author Cory Petosky
*/
[SWF(width="800", height="600", backgroundColor="#000000", frameRate="30")]
dynamic public class g extends Sprite {
public function g() {
this.dir = 0x01;
this.fpv = new Sprite();
this.map = new BitmapData(320, 320);
this.bchance = 0.07;
this.battle = this.defending = false;
this.bgc = 0;
this.tl = [];
var tf:TextFormat = new TextFormat("Courier New", 20, 0xFFFFFFFF);
this.newChar = function ():void {
this.tl.push("Gain levels, find stairs down, how deep can you get?");
this.mhp = this.hp = 10 + int(Math.random() * 10);
this.str = 8 + int(Math.random() * 8);
this.def = 8 + int(Math.random() * 8);
this.level = 1;
this.xp = 0;
this.xpn = 100;
this.floor = 0;
this.newFloor();
};
this.newFloor = function():void {
this.floor++;
randomFloor();
this.wa = new BitmapData(32, 16);
this.wa.perlinNoise(32, 16, 16, Math.random() * 1000000, true, true, 1 + Math.random() * 7);
do {
this.px = int(Math.random() * 20);
this.py = int(Math.random() * 20);
} while (this.maze[this.px][this.py] == 0);
this.maze[this.px][this.py] |= 0x10;
do {
this.sx = int(Math.random() * 20);
this.sy = int(Math.random() * 20);
} while (this.maze[this.sx][this.sy] == 0);
this.sk = false;
};
// setup dirs right->down->left
this.dirs = [];
this.dirs[0x01] = [0x02, 0x04, 0x08];
this.dirs[0x02] = [0x04, 0x08, 0x01];
this.dirs[0x04] = [0x08, 0x01, 0x02];
this.dirs[0x08] = [0x01, 0x02, 0x04];
this.d_stats = new TextField();
this.d_stats.defaultTextFormat = tf;
this.d_stats.x = 10;
this.d_stats.y = 325;
this.d_stats.width = 780;
this.d_stats.selectable = false;
addChild(this.d_stats);
this.d_text = new TextField();
this.d_text.defaultTextFormat = tf;
this.d_text.x = 10;
this.d_text.width = 780;
this.d_text.y = 395;
this.d_text.height = 205;
this.d_text.selectable = false;
this.d_text.background = true;
this.d_text.border = true;
this.d_text.borderColor = 0xFFFFFFFF;
addChild(this.d_text);
this.newChar();
this.fpv.scrollRect = new Rectangle(0, 0, 480, 320);
addChild(this.fpv);
this.maph = new Bitmap(this.map);
this.maph.x = 480;
addChild(this.maph);
addEventListener("enterFrame", efl);
}
private function kl(event:KeyboardEvent):void {
if (this.battle) {
if (event.keyCode == 65) {
var dmg:int = (1 + Math.random() * 10) + this.str - this.edef - (1 + Math.random() * 10);
if (dmg < 1)
dmg = Math.random() * 2;
if (dmg < 1)
this.tl.push("You miss.");
else {
this.tl.push("You attack for " + dmg + ".");
this.ehp -= dmg;
}
if (this.ehp < 1) {
this.tl.push("You win!");
this.battle = false;
this.tl.push("Gained " + 10 * this.floor + " XP.");
this.xp += 10 * this.floor;
while (this.xp >= this.xpn) {
this.level++;
this.tl.push("Welcome to level " + this.level + "!");
var gain:int = 1 + Math.random() * 5;
this.tl.push("Gained " + gain + " HP.");
this.mhp += gain;
this.hp += gain;
gain = 1 + Math.random() * 4;
this.tl.push("Gained " + gain + " STR.");
this.str += gain;
gain = 1 + Math.random() * 4;
this.tl.push("Gained " + gain + " DEF.");
this.def += gain;
this.xpn = int(this.xpn * 2.1);
}
} else {
enemyTurn();
}
} else if (event.keyCode == 72) {
this.tl.push("You begin mending...");
this.defending = true;
enemyTurn();
} else if (event.keyCode == 82) {
if (Math.random() < this.rchance) {
this.tl.push("You run away!");
this.battle = false;
} else {
this.rchance *= 1.25; // make it easier on successive attempts
this.tl.push("You couldn't run away...");
enemyTurn();
}
}
} else {
if (event.keyCode == 38) {
if (this.maze[this.px][this.py] & this.dir) {
move(this.dir);
this.maze[this.px][this.py] |= 0x10;
if (this.sx == this.px && this.sy == this.py) {
this.tl.push("Stairs down. You can (d)escend.");
this.sk = true;
} else {
checkBattle();
}
}
} else if (event.keyCode == 39) {
this.dir = this.dirs[this.dir][0];
} else if (event.keyCode == 40) {
if (this.maze[this.px][this.py] & this.dirs[this.dir][1]) {
move(this.dirs[this.dir][1]);
this.maze[this.px][this.py] |= 0x10;
if (this.sx == this.px && this.sy == this.py) {
this.tl.push("Stairs down. You can (d)escend.");
this.sk = true;
} else {
checkBattle();
}
}
} else if (event.keyCode == 37) {
this.dir = this.dirs[this.dir][2];
} else if (event.keyCode == 68) {
if (this.sx == this.px && this.sy == this.py) {
this.newFloor();
this.tl.push("Welcome to floor " + this.floor + ".");
} else {
this.tl.push("No stairs to (d)escend.");
}
}
}
}
private function enemyTurn():void {
var dmg:int = (1 + Math.random() * 10) + this.estr - this.def - (1 + Math.random() * 10);
// 1.5x when defending!
if (this.defending) dmg -= (this.def + (1 + Math.random() * 10) >> 1);
if (dmg < 1 && Math.random() * 2 < 1) dmg = 1;
if (dmg < 1) {
if (this.defending) {
if (dmg < this.hp - this.mhp) dmg = this.hp - this.mhp;
this.tl.push("You defend yourself and mend for " + -dmg + ".");
this.hp -= dmg;
} else {
this.tl.push("Enemy misses.");
}
} else {
this.tl.push("Enemy hits you for " + dmg);
this.hp -= dmg;
}
this.defending = false;
if (this.hp < 1) {
this.d_text.text = "";
this.tl.push("You lose...");
this.tl.push("You got to floor " + this.floor + " as a level " + this.level + " hero.");
this.tl.push("Restarting game...");
this.battle = false;
this.newChar();
} else {
this.tl.push("(A)ttack, (H)eal, (R)un?");
}
}
private function checkBattle():void {
if (Math.random() < this.bchance) {
// battle start
this.battle = true;
this.bchance = 0.07;
this.rchance = 0.25;
this.bgc = 0xFFFFFFFF;
this.ehp = (3 + Math.random() * 10) * this.floor;
this.estr = (3 + Math.random() * 8) * this.floor;
this.edef = (3 + Math.random() * 8) * this.floor;
this.tl.push("Enemy!");
this.tl.push("(A)ttack, (H)eal, (R)un?");
} else {
this.bchance *= 1.01;
}
}
private function move(dir:uint):void {
if (dir == 0x01) this.py--;
else if (dir == 0x02) this.px++;
else if (dir == 0x04) this.py++;
else this.px--;
} // end function move
private function efl(event:Event):void {
stage.addEventListener("keyDown", kl);
if (this.tl[0]) {
this.d_text.appendText(this.tl[0] + "\n");
while (this.d_text.numLines > 9) this.d_text.text = this.d_text.text.substr(this.d_text.text.indexOf("\r") + 1);
this.tl.shift();
}
var i:uint;
var j:uint;
this.fpv.graphics.clear();
this.fpv.graphics.beginFill(0x111111);
this.fpv.graphics.drawRect(0, 0, 480, 320);
this.fpv.graphics.lineStyle(2, 0xFFFFFFFF);
move(this.dir);
move(this.dir);
move(this.dir);
move(this.dir);
move(this.dir);
var p:Array = [];
for (var d:int = 5; d > 0; d--) {
move(this.dirs[this.dir][1]);
var z:int = 200 * d;
// draw middle frontal wall
if (this.maze[this.px] && this.maze[this.px][this.py] && !(this.maze[this.px][this.py] & this.dir)) {
p.push([-100, 100, z]);
p.push([100, 100, z]);
p.push([100, -100, z]);
p.push([-100, -100, z]);
}
var x:int = -100;
var sw:Boolean;
var m:int = 0;
while ((240 * (x / z)) + 240 > 0) {
sw = false;
// find left side wall
if (this.maze[this.px] && this.maze[this.px][this.py] && !(this.maze[this.px][this.py] & this.dirs[this.dir][2])) sw = true;
move(this.dirs[this.dir][2]);
++m;
// draw left back wall
if (this.maze[this.px] && this.maze[this.px][this.py] && !(this.maze[this.px][this.py] & this.dir)) {
p.push([x - 200, 100, z]);
p.push([x, 100, z]);
p.push([x, -100, z]);
p.push([x - 200, -100, z]);
}
// draw left side wall
if (sw) {
p.push([x, 100, z]);
p.push([x, -100, z]);
p.push([x, -100, (z - 200) > 50 ? (z - 200) : 50]);
p.push([x, 100, (z - 200) > 50 ? (z - 200) : 50]);
}
x -= 200;
}
while (m-- > 0) move(this.dirs[this.dir][0]);
x = 100;
m = 0;
while ((240 * (x / z)) + 240 < 480) {
sw = false;
// find right side wall
if (this.maze[this.px] && this.maze[this.px][this.py] && !(this.maze[this.px][this.py] & this.dirs[this.dir][0])) {
sw = true;
}
move(this.dirs[this.dir][0]);
++m;
// draw right back wall
if (this.maze[this.px] && this.maze[this.px][this.py] && !(this.maze[this.px][this.py] & this.dir)) {
p.push([x + 200, 100, z]);
p.push([x, 100, z]);
p.push([x, -100, z]);
p.push([x + 200, -100, z]);
}
// draw right side wall
if (sw) {
p.push([x, 100, z]);
p.push([x, -100, z]);
p.push([x, -100,(z - 200) > 50 ? (z - 200) : 50]);
p.push([x, 100, (z - 200) > 50 ? (z - 200) : 50]);
}
x += 200;
}
while (m-- > 0) move(this.dirs[this.dir][2]);
}
for (i = 0; i < p.length; i += 4) {
this.fpv.graphics.beginBitmapFill(this.wa);
this.fpv.graphics.moveTo(240 * p[i][0] / p[i][2] + 240,240 * p[i][1] / p[i][2] + 160);
for (j = i + 1; j < i + 4; ++j) this.fpv.graphics.lineTo( 240 * p[j][0] / p[j][2] + 240,240 * p[j][1] / p[j][2] + 160);
}
var rect:Rectangle = new Rectangle();
// draw minimap
this.map.fillRect(new Rectangle(0, 0, 320, 320), 0xFFFFFFFF);
for (i = 0; i < 20; i++) {
for (j = 0; j < 20; j++) {
var cell:uint = this.maze[i][j];
if ((cell & 0x10) && cell) {
if ((cell & 0x01) == 0) {
rect.x = i * 16;
rect.y = j * 16;
rect.width = 16;
rect.height = 3;
this.map.fillRect(rect, 0xFF000000);
}
if ((cell & 0x08) == 0) {
rect.x = i * 16;
rect.y = j * 16;
rect.width = 3;
rect.height = 16;
this.map.fillRect(rect, 0xFF000000);
}
if ((cell & 0x02) == 0) {
rect.x = i * 16 + 13;
rect.y = j * 16;
rect.width = 3;
rect.height = 16;
this.map.fillRect(rect, 0xFF000000);
}
if ((cell & 0x04) == 0) {
rect.x = i * 16;
rect.y = j * 16 + 13;
rect.width = 16;
rect.height = 3;
this.map.fillRect(rect, 0xFF000000);
}
} else {
rect.x = i * 16;
rect.y = j * 16;
rect.width = 16;
rect.height = 16;
this.map.fillRect(rect, 0xFF666666);
}
}
}
// draw position on minimap
rect.x = this.px * 16 + 4;
rect.y = this.py * 16 + 4;
rect.width = 8;
rect.height = 8;
this.map.fillRect(rect, 0xFF0000FF);
// draw head
rect.width >>= 1;
rect.height >>= 1;
if (this.dir == 0x01) {
rect.x += 2;
} else if (this.dir == 0x02) {
rect.x += 4;
rect.y += 2;
} else if (this.dir == 0x04) {
rect.x += 2;
rect.y += 4;
} else {
rect.y += 2;
}
this.map.fillRect(rect, 0xFFFFFFFF);
// draw stairs (if applicable)
if (this.sk) {
rect.x = this.sx * 16 + 4;
rect.y = this.sy * 16 + 4;
rect.width = 8;
rect.height = 8;
this.map.fillRect(rect, 0xFF00AA55);
}
// display stats
this.d_stats.text = "HP: " + this.hp + "/" + this.mhp + "\n" +
"STR: " + this.str + " " + "DEF: " + this.def + "\n" +
"Level " + this.level + " XP: " + this.xp + "/" + this.xpn;
this.bgc *= 0.35;
this.d_text.backgroundColor = this.bgc;
}
public function randomFloor():void {
var i:uint;
var j:uint;
this.maze = [
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],
[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0]
];
this.ccx = int(Math.random() * 20);
this.ccy = int(Math.random() * 20);
this.lss = 0;
this.ld = 0;
var remaining:uint = 399;
while (remaining > 0) {
while (!drawPassage(true)) {
do {
this.ccx = int(Math.random() * 20);
this.ccy = int(Math.random() * 20);
} while(this.maze[this.ccx][this.ccy] == 0);
}
--remaining;
}
// Sparsify
for (i = 0; i < 20; i++) {
for (j = 0; j < 20; j++) {
if (this.maze[i][j] == 0x01) {
this.maze[i][j] = 0;
this.maze[i][j - 1] ^= 0x04;
} else if (this.maze[i][j] == 0x02) {
this.maze[i][j] = 0;
this.maze[i + 1][j] ^= 0x08;
} else if (this.maze[i][j] == 0x04) {
this.maze[i][j] = 0;
this.maze[i][j + 1] ^= 0x01;
} else if (this.maze[i][j] == 0x08) {
this.maze[i][j] = 0;
this.maze[i - 1][j] ^= 0x02;
}
} // for j
}
for (i = 0; i < 20; i++) {
for (j = 0; j < 20; j++) {
if (this.maze[i][j] == 0x01) {
this.maze[i][j] = 0;
this.maze[i][j - 1] ^= 0x04;
} else if (this.maze[i][j] == 0x02) {
this.maze[i][j] = 0;
this.maze[i + 1][j] ^= 0x08;
} else if (this.maze[i][j] == 0x04) {
this.maze[i][j] = 0;
this.maze[i][j + 1] ^= 0x01;
} else if (this.maze[i][j] == 0x08) {
this.maze[i][j] = 0;
this.maze[i - 1][j] ^= 0x02;
}
} // for j
}
// Remove deadends
for (i = 0; i < 20; i++) {
for (j = 0; j < 20; j++) {
if (iss(this.maze[i][j]) && Math.random() * 100 < 75) {
this.ccx = i;
this.ccy = j;
this.lss = 0;
this.ld = 0;
while (iss(this.maze[this.ccx][this.ccy])) {
while (!drawPassage(false)) {
do {
this.ccx = int(Math.random() * 20);
this.ccy = int(Math.random() * 20);
} while(this.maze[this.ccx][this.ccy] == 0);
}
}
}
}
}
}
/**
* Attempts to add a single new passage to a cells array using a
* Hunt-and-Kill algorithm.
*
* @param cells The cells array to modify.
* @param currentCell The cell to work from.
* @param randomness Percentage (0-100) chance to change direction
* last successful passage.
* @param lastPassage Struct containing the last direction moved and
* the number of times the passage has gone in a single direction.
* @param noLoops If true, will never draw a passage that connects to
* an already-visted cell.
*/
private function drawPassage(nol:Boolean):Boolean {
var directions:uint = this.maze[this.ccx][this.ccy];
if (directions == 0x0F)
return false;
var direction:uint;
if (this.ccx < 1)
directions |= 0x08;
if (this.ccx + 1 >= 20)
directions |= 0x02;
if (this.ccy < 1)
directions |= 0x01;
if (this.ccy + 1 >= 20)
directions |= 0x04;
if (int(Math.random() * 8) < 4 || !cgs(nol) ) {
this.lss = 0;
direction = srd(directions, nol);
} else {
++this.lss;
direction = this.ld;
}
if (direction == 0)
return false;
this.ld = direction;
this.maze[this.ccx][this.ccy] |= direction;
if (direction == 0x01) {
--this.ccy;
direction = 0x04;
} else if (direction == 0x02) {
++this.ccx;
direction = 0x08;
} else if (direction == 0x04) {
++this.ccy;
direction = 0x01;
} else if (direction == 0x08) {
--this.ccx;
direction = 0x02;
}
this.maze[this.ccx][this.ccy] |= direction;
return true;
}
/**
* Returns true if the next cell in the current direction is unvisited.
*/
private function cgs(nl:Boolean):Boolean {
if (this.ld == 0x01) {
return (
(this.lss < (this.ccy >> 1)) &&
(this.ccy > 0) &&
!(nl && this.maze[this.ccx][this.ccy - 1])
);
} else if (this.ld == 0x02) {
return (
(this.lss < (this.ccx >> 1)) &&
(this.ccx + 1 < 20) &&
!(nl && this.maze[this.ccx + 1][this.ccy])
);
} else if (this.ld == 0x04) {
return (
(this.lss < (this.ccy >> 1)) &&
(this.ccy + 1 < 20) &&
!(nl && this.maze[this.ccx][this.ccy + 1])
);
} else if (this.ld == 0x08) {
return (
(this.lss < (this.ccx >> 1)) &&
(this.ccx > 0) &&
!(nl && this.maze[this.ccx - 1][this.ccy])
);
} else {
return false;
}
}
private function srd(ds:uint, ar:Boolean):uint {
var d:uint;
while ((d == 0) || ((ds & d) != 0)) {
var tx:uint = this.ccx;
var ty:uint = this.ccy;
var r:int = int(Math.random() * 4);
if (r == 0) {
if (this.ccy > 0) {
d = 0x01;
--ty;
} else {
ds |= 0x01;
}
} else if (r == 1) {
if (this.ccy + 1 < 20) {
d = 0x04;
++ty;
} else {
ds |= 0x04;
}
} else if (r == 2) {
if (this.ccx > 0) {
d = 0x08;
--tx;
} else {
ds |= 0x08;
}
} else {
if (this.ccx + 1 < 20) {
d = 0x02;
++tx;
} else {
ds |= 0x02;
}
}
if (ar && this.maze[tx][ty] != 0) {
ds |= d;
if (ds == 0x0F)
return 0;
d = 0;
}
}
return d;
}
public function iss(d:uint):Boolean {
return ( d == 0x01 || d == 0x02 || d == 0x04 || d == 0x08 );
}
}
}