HTML preprocessors can make writing HTML more powerful or convenient. For instance, Markdown is designed to be easier to write and read for text documents and you could write a loop in Pug.
In CodePen, whatever you write in the HTML editor is what goes within the <body>
tags in a basic HTML5 template. So you don't have access to higher-up elements like the <html>
tag. If you want to add classes there that can affect the whole document, this is the place to do it.
In CodePen, whatever you write in the HTML editor is what goes within the <body>
tags in a basic HTML5 template. If you need things in the <head>
of the document, put that code here.
The resource you are linking to is using the 'http' protocol, which may not work when the browser is using https.
CSS preprocessors help make authoring CSS easier. All of them offer things like variables and mixins to provide convenient abstractions.
It's a common practice to apply CSS to a page that styles elements such that they are consistent across all browsers. We offer two of the most popular choices: normalize.css and a reset. Or, choose Neither and nothing will be applied.
To get the best cross-browser support, it is a common practice to apply vendor prefixes to CSS properties and values that require them to work. For instance -webkit-
or -moz-
.
We offer two popular choices: Autoprefixer (which processes your CSS server-side) and -prefix-free (which applies prefixes via a script, client-side).
Any URLs added here will be added as <link>
s in order, and before the CSS in the editor. You can use the CSS from another Pen by using its URL and the proper URL extension.
You can apply CSS to your Pen from any stylesheet on the web. Just put a URL to it here and we'll apply it, in the order you have them, before the CSS in the Pen itself.
You can also link to another Pen here (use the .css
URL Extension) and we'll pull the CSS from that Pen and include it. If it's using a matching preprocessor, use the appropriate URL Extension and we'll combine the code before preprocessing, so you can use the linked Pen as a true dependency.
JavaScript preprocessors can help make authoring JavaScript easier and more convenient.
Babel includes JSX processing.
Any URL's added here will be added as <script>
s in order, and run before the JavaScript in the editor. You can use the URL of any other Pen and it will include the JavaScript from that Pen.
You can apply a script from anywhere on the web to your Pen. Just put a URL to it here and we'll add it, in the order you have them, before the JavaScript in the Pen itself.
If the script you link to has the file extension of a preprocessor, we'll attempt to process it before applying.
You can also link to another Pen here, and we'll pull the JavaScript from that Pen and include it. If it's using a matching preprocessor, we'll combine the code before preprocessing, so you can use the linked Pen as a true dependency.
Search for and use JavaScript packages from npm here. By selecting a package, an import
statement will be added to the top of the JavaScript editor for this package.
Using packages here is powered by esm.sh, which makes packages from npm not only available on a CDN, but prepares them for native JavaScript ESM usage.
All packages are different, so refer to their docs for how they work.
If you're using React / ReactDOM, make sure to turn on Babel for the JSX processing.
If active, Pens will autosave every 30 seconds after being saved once.
If enabled, the preview panel updates automatically as you code. If disabled, use the "Run" button to update.
If enabled, your code will be formatted when you actively save your Pen. Note: your code becomes un-folded during formatting.
Visit your global Editor Settings.
<div id="app"><div id="loading"></div></div>
/* https://meyerweb.com/eric/tools/css/reset/
v2.0 | 20110126
License: none (public domain)
*/
html, body, div, span, applet, object, iframe,
h1, h2, h3, h4, h5, h6, p, blockquote, pre,
a, abbr, acronym, address, big, cite, code,
del, dfn, em, img, ins, kbd, q, s, samp,
small, strike, strong, sub, sup, tt, var,
b, u, i, center,
dl, dt, dd, ol, ul, li,
fieldset, form, label, legend,
table, caption, tbody, tfoot, thead, tr, th, td,
article, aside, canvas, details, embed,
figure, figcaption, footer, header, hgroup,
menu, nav, output, ruby, section, summary,
time, mark, audio, video {
margin: 0;
padding: 0;
border: 0;
font-size: 100%;
font: inherit;
vertical-align: baseline;
}
/* HTML5 display-role reset for older browsers */
article, aside, details, figcaption, figure,
footer, header, hgroup, menu, nav, section {
display: block;
}
body {
line-height: 1;
}
ol, ul {
list-style: none;
}
blockquote, q {
quotes: none;
}
blockquote:before, blockquote:after,
q:before, q:after {
content: '';
content: none;
}
table {
border-collapse: collapse;
border-spacing: 0;
}
/// Demo-related styles
/// -------------------
///
/// You can skip this in your own implementation.
///
/// @author Manuel Tancoigne
/// @license MIT
/*
Variables
*/
// Document background
// Fonts
@import url(https://fonts.googleapis.com/css?family=Flamenco:400|Trade+Winds|Overlock:400,900);
$font-header:'Trade Winds';
$font-body: 'Overlock';
$font-quote: 'Flamenco';
$bg-color:#DDD;
// Default
$border-color:darken($bg-color, 10%);
$pane-bg-color:lighten($bg-color, 5%);
$log-bg-color: #FFFFEE;
$green-background:mix(#0F0, $border-color);
$green-border-color:darken($green-background, 15%);
$green-text:darken($green-border-color, 10%);
$blue-background:mix(#00F, $border-color);
$blue-border-color:darken($blue-background, 15%);
$blue-text:darken($blue-border-color, 10%);
$red-background:mix(#F00, $border-color);
$red-border-color:darken($red-background, 15%);
$red-text:darken($red-border-color, 10%);
$orange-background:mix(#F80, $border-color);
$orange-border-color:darken($orange-background, 15%);
$orange-text:darken($orange-border-color, 10%);
$font-size:12px;
$font-color:#444;
$radius-sm:3px;
$radius-lg:5px;
$padding-sm:3px;
$padding-lg:10px;
$margin-sm:5px;
$margin-lg:15px;
$main-width: 1100px; //14*80+2+2*$padding-lg;
*{
box-sizing: border-box;
}
body{
background-color:$bg-color;
font-family: $font-body;
font-size:$font-size;
color:$font-color;
line-height: $font-size+2*$padding-sm;
padding:$padding-lg;
}
h1{
font-family: $font-header;
font-size:3em;
padding:$padding-lg;
padding-top:$padding-lg*2;
margin-bottom:-$margin-lg/2;
margin-left:$margin-lg*2;
font-weight:bold;
background: $pane-bg-color;
border:1px solid $border-color;
border-bottom:none;
display:inline-block;
small{
font-family:Overlock;
font-size:0.5em;
color: $blue-text;
}
}
h2{
margin: $margin-sm 0px;
font-size:1.5em;
font-weight:bold;
}
small{
font-size:0.8em;
}
input{
background-color:$bg-color;
border:1px solid $border-color;
padding:$padding-sm;
border-bottom-width: 3px;
}
input:focus{
background-color: lighten($bg-color, 10%);
}
/* Remove controls from Firefox */
input[type=number] {
-moz-appearance: textfield;
text-align: right;
}
/* Re-applies the controls on :hover and :focus */
input[type="number"]:hover,
input[type="number"]:focus {
-moz-appearance: number-input;
}
// Remove the glow.
textarea, input { outline: none; }
table{
width:100%
}
th{
padding:$padding-sm $padding-lg;
width:75px;
text-align: left;
font-weight: bold;
display:flex;
}
td{
padding-left: $padding-sm $padding-lg;
}
tr:nth-child(odd) {background: rgba(0, 0, 0, 0.1)}
tr:nth-child(even) {background: rgba(255, 255, 255, 0.2)}
#app{
width:($main-width);
margin-left:50%;
transform: translateX(-50%);
min-height: 100%;
@media(max-width:1135px){
margin-left:0px;
transform: translateX(0);
}
}
#main{
display: flex;
margin:$margin-lg 0px;
}
#mapInfos{
display: flex;
p{
margin: $margin-sm $margin-lg;
}
}
#menu{
width:250px;
}
#menu, #game{
margin-right: $margin-lg;
}
#menu, #game, #questLog{
display:block;
}
#questLog{
width:250px;
height:100%;
max-height: 525px;
overflow:hidden;
div{
font-family:monospace;
font-size:0.8em;
&.info{
border-bottom:$padding-sm solid $border-color;
font-size:1.5em;
margin-bottom:$margin-sm;
color:darken($font-color, 10%);
}
}
}
#reset{
padding:$padding-lg;
p{
font-size: 2em;
margin-bottom: 15px;
text-align: center;
}
}
#mapName{
margin-left:$margin-lg;
text-align:center;
font-size:1.5em;
font-weight: bold;
small{
font-size:0.8em;
}
}
#infoMessage{
margin-bottom:$margin-lg*2;
padding:$padding-lg;
font-size: 1.2em;
.btn{
margin-bottom: 0px;
margin-top: $margin-lg;
}
p{
margin-top:$margin-sm;
:first-child{
margin-top: 0px;
}
}
}
.avatar{
border:1px solid $border-color;
border-bottom-width: $radius-sm;
background-color: $pane-bg-color;
padding:$padding-sm;
margin-bottom: $margin-lg;
}
.avatar-header{
display: flex;
margin-bottom: $margin-sm;
}
.avatar-img{
display: inline-flex;
max-width:50px;
max-height:50px;
min-width:50px;
min-height:50px;
background-repeat: no-repeat;
}
.avatar-content{
display: inline-block;
border-left:1px solid $border-color;
margin-left: $margin-sm;
padding-left: $margin-sm;
min-height:50px;
}
.avatar-name{
font-weight:bold;
font-size:1.2em;
margin-bottom:$margin-sm;
}
.avatar-description{
font-family: $font-quote;
font-size: 1.3em;
margin-top:-$margin-sm;
}
.btn{
margin-top:$margin-sm;
cursor: pointer;
font-variant:small-caps;
text-align: center;
font-weight: bold;
text-decoration: none;
color:$font-color;
background-color: lighten($border-color, 5%);
border:1px solid $border-color;
border-radius: $radius-sm;
padding:$padding-sm $padding-lg;
border-bottom-width:$radius-sm;
&:hover{
background-color:lighten($border-color, 15%);
}
.active{
top:$radius-sm;
border-top:$radius-sm;
border-bottom-width: 1px;
background-color:lighten($border-color, 10%);
}
}
.btn-block{
display: block;
width:100%;
margin-bottom: $margin-sm;
}
.disabled{
opacity: 0.5;
cursor:not-allowed;
}
.panel{
border: 1px solid $border-color;
background-color: $pane-bg-color;
padding: $padding-lg;
}
.tooltip {
display:none;
position:absolute;
border:1px solid #333;
background-color:#161616;
border-radius:$radius-sm;
padding:$padding-sm $padding-lg;
color:#fff;
font-size:0.8em;
font-family:monospace;
li{
margin-left:$margin-lg;
}
strong{
text-decoration: underline;
}
}
.message-success{
background-color: transparentize($green-border-color, 0.8);
border-left:3px solid $green-border-color;
color:$font-color;
padding:0 $padding-lg;
}
.message-info{
background-color: transparentize($blue-border-color, 0.8);
border-left:3px solid $blue-border-color;
color:$font-color;
padding:0 $padding-lg;
}
.message-danger{
background-color: transparentize($orange-border-color, 0.8);
border-left:3px solid $orange-border-color;
color:$font-color;
padding:0 $padding-lg;
}
.message-fatal{
background-color: transparentize($red-border-color, 0.8);
border-left:3px solid $red-border-color;
color:$font-color;
padding:0 $padding-lg;
}
.message-system{
background-color: transparentize($border-color, 0.8);
border-left:3px solid $border-color;
color:$font-color;
padding:0 $padding-lg;
}
.message-hr{
text-align: center;
}
.progress{
width:100%;
//margin-top:2px;
.bar-wrap{
//border:1px solid $border-color;
//position: relative;
//transform: translateY(-50%);
.bar{
height:4px;
}
}
span{
position: relative;
float:left;
text-align: center;
display: block;
}
}
.orange{
background-color:$orange-background;
}
.red{
background-color:$red-background;
}
.green{
background-color:$green-background;
}
.btn-blue{
border-color: $blue-border-color;
background-color:$blue-background;
color:$blue-text;
&:hover{
background-color:lighten($blue-background, 10%);
}
}
.btn-green{
border-color: $green-border-color;
background-color:$green-background;
color:$green-text;
&:hover{
background-color:lighten($green-background, 10%);
}
}
#target-player flash, #target-enemy flash{
@extend .red;
transition-duration: .2s;
}
#reset{
position:fixed;
padding:$padding-lg;
top: 50%;
left: 50%;
min-width: 300px;
transform: translateY(-50%);
transform: translateX(-50%);
.btn{
padding:2*$padding-lg;
font-size:2em;
border-bottom-width:5px;
border-color: $green-border-color;
background-color:$green-background;
color:$green-text;
margin-bottom: 0;
&:hover{
background-color:lighten($green-background, 10%);
}
}
}
.overlay {
position: fixed;
min-width: 100%;
min-height: 100%;
top: 0px;
left: 0px;
background-color: rgba(0,0,0, 0.5);
display: block;
}
/*
VARIABLES
*/
/// Number of rooms to be generated. 50 is great...
/// @type Integer
$nb-rooms:50;
/// Prefix for classes
/// @type string
$prefix:'map-';
/// Cell size
/// @type length
$cell-size:35px;
$cell-bg-width:100%;
// Some colors, for cells backgrounds
$floorColor: #c1ac53;
$wallColor: #CCC;
$waterColor: #1d67f9;
$lavaColor: #f9d51d;
// Some cells backgrounds.
// All credit to me. Same license.
// Small sprites for legend
$img-sm-boss: url('');
$img-sm-enemy:url('');
$img-sm-chest:url('');
$img-sm-health:url('');
$img-sm-player: url('');
$img-sm-wall:url('');
$img-sm-floor:url('');
$img-sm-water:url("");
$img-sm-lava:url('');
// Stats
$img-sm-armor:url('');
$img-sm-celerity:url('');
$img-sm-damage:url('');
$img-sm-level:url('');
$img-sm-life:url('');
$img-sm-strength:url('');
$img-sm-xp:url('');
// Sprites for board
$img-boss: url('');
$img-boss-tomb:url('');
$img-enemy:url('');
$img-enemy-tomb:url('');
$img-player:url('');
$img-player-tomb:url('');
// Board: Items
$img-chest:url('');
$img-health:url('');
$img-lava:url('');
$img-wall:url('');
$img-floor:url('');
$img-water:url('');
// Walls variations
$img-wall-1:url('');
$img-wall-2:url('');
$img-wall-3:url('');
$img-wall-4:url('');
// Floor variations
$img-floor-1:url('');
$img-floor-2:url('');
$img-floor-3:url('');
$img-floor-4:url('');
// Avatars
$img-avatar-enemy:url('');
$img-avatar-boss:url('');
$img-avatar-player:url('');
//Loading image
$img-loading:url('');
/*
CLASSES
*/
/// Map row
.#{$prefix}row{
display: table-row;
}
/* Cell */
// Map cell
.#{$prefix}cell{
display: table-cell;
width: $cell-size;
border:none;
}
// Maintains the cell square
.#{$prefix}cell:after{
content: "";
display: block;
padding-bottom: 100%;
border:none;
}
/* Specific cells */
// Walls
.#{$prefix}cell-wall{
background-image: $img-wall;
background-size: 100% auto;
background-color: $wallColor;
}
.#{$prefix}cell-wall-0{background-image: $img-wall-1;}
.#{$prefix}cell-wall-1{background-image: $img-wall-2;}
.#{$prefix}cell-wall-2{background-image: $img-wall-4;}
.#{$prefix}cell-wall-3{background-image: $img-wall-1;}
// Floor
.#{$prefix}cell-floor{
background-image: $img-floor;
background-size: 100% auto;
background-color: $floorColor;
}
.#{$prefix}cell-floor-0{background-image: $img-floor-1 !important;}
.#{$prefix}cell-floor-1{background-image: $img-floor-2 !important;}
.#{$prefix}cell-floor-2{background-image: $img-floor-4 !important;}
.#{$prefix}cell-floor-3{background-image: $img-floor-1 !important;}
// Water
.#{$prefix}cell-water{
/*background-image:url();*/
background-image: $img-water;
background-size: 100% auto;
background-color:$waterColor !important;
}
// Lava
.#{$prefix}cell-lava{
background-image: $img-lava;
background-size: 100% auto;
background-color:$lavaColor !important;
}
// Void
.#{$prefix}cell-void{
background-size: $cell-bg-width auto;
background-color:#000 !important;
}
// Void
.#{$prefix}cell-void-half:after{
background-size: $cell-bg-width auto;
background-color:rgba(0,0,0,.5) !important;
}
// Enemy
.#{$prefix}cell-item-enemy:after{
background-size: 100% auto;
background-image:$img-enemy;
}
// Item
.#{$prefix}cell-item-chest:after{
background-size: 100% auto;
background-image: $img-chest;
}
.#{$prefix}cell-item-health:after{
background-size: 100% auto;
background-image: $img-health;
}
// Boss
.#{$prefix}cell-item-boss:after{
background-size: 100% auto;
background-image: $img-boss;
}
// Boss
.#{$prefix}cell-item-player:after{
background-size: 100% auto;
background-image: $img-player !important;
}
.#{$prefix}cell-item-tomb:after{
background-size: 100% auto;
background-image:$img-enemy-tomb;
}
.#{$prefix}cell-item-tomb-boss:after{
background-size: 100% auto;
background-image:$img-boss-tomb;
}
.#{$prefix}cell-item-tomb-player:after{
background-size: 100% auto;
background-image:$img-player-tomb;
}
.avatar-player{
background-image: $img-avatar-player;
background-size: 100% auto;
}
.avatar-player-dead{
background-image: none;
}
.avatar-enemy{
background-image: $img-avatar-enemy;
background-size: 100% auto;
}
.avatar-boss{
background-image: $img-avatar-boss;
background-size: 100% auto;
}
.avatar-none:after{
content:"?";
line-height:50px;
font-size:35px;
text-align: center;
width:50px;
}
.ico{
display: inline-block;
min-height:14px;
min-width:14px;
width:14px;
height:14px;
background-repeat: no-repeat;
margin:0px $margin-sm;
}
td .ico{
float: left;
display: inline;
}
.ico-stat-armor{background-image: $img-sm-armor;min-width: 15px;min-height:15px;}
.ico-stat-celerity{background-image: $img-sm-celerity;min-width: 15px;min-height:15px;}
.ico-stat-damage{background-image: $img-sm-damage;min-width: 15px;min-height:15px;}
.ico-stat-level{background-image: $img-sm-level;min-width: 15px;min-height:15px;}
.ico-stat-life{background-image: $img-sm-life;min-width: 15px;min-height:15px;}
.ico-stat-strength{background-image: $img-sm-strength;min-width: 15px;min-height:15px;}
.ico-stat-xp{background-image: $img-sm-xp;min-width: 15px;min-height:15px;}
.ico-boss{
background-image:$img-sm-boss;
}
.ico-enemy{
background-image:$img-sm-enemy;
}
.ico-chest{
background-image:$img-sm-chest;
}
.ico-health{
background-image:$img-sm-health;
}
.ico-player{
background-image:$img-sm-player;
}
.ico-wall{
background-image:$img-sm-wall;
}
.ico-floor{
background-image:$img-sm-floor;
}
.ico-water{
background-image:$img-sm-water;
}
.ico-lava{
background-image:$img-sm-lava;
}
#loading{
position:absolute;
top: 0px;
left: 0px;
min-width: 100%;
min-height: 100%;
background-color: $pane-bg-color;
border:1px solid $border-color;
transition: 1s;
}
#loading::after{
width:50px;
height:50px;
min-height:50px;
min-width:50px;
background-image: $img-loading;
background-repeat: no-repeat;
background-size: 100% auto;
display:inline-block;
position:absolute;
top:50%;
left:50%;
margin-left:-25px;
margin-right:25px;
content:"";
}
class Cell extends React.Component{
render(){
var className='map-cell';
if(['visible', 'half'].indexOf(this.props.cellData.type.discovered)>-1){
var tmp='';
// Adding styles from cellData
this.props.cellData.type.classNames.map((c)=>{tmp+=' map-cell-'+c});
className+=tmp;
// Items
for(let i in this.props.items){
if(this.props.items[i].position===this.props.position){
className+=' map-cell-item-'+this.props.items[i].className;
}
}
if(this.props.cellData.type.discovered=='half'){className+=' map-cell-void-half'};
}else{
className+=' map-cell-void';
}
className+=' map-cell-'+this.props.cellData.x+'-'+this.props.cellData.y
return (<div className={className}></div>);
}
}
class EnemyInfos extends React.Component{
constructor(props){
super(props);
}
render(){
var name='Not in combat.';
var description='';
var link='';
var level='';
var damage='';
var strength='';
var celerity='';
var armor='';
var life='';
var pBar='';
if(this.props.name!=null){
name=this.props.name;
description=this.props.description;
link=this.props.link;
level=this.props.level;
damage=this.props.stats.damage;
strength=this.props.stats.strength;
armor=this.props.stats.armor;
celerity=this.props.stats.celerity;
var barColor='green';
var percent=Math.ceil(this.props.stats.life/this.props.stats.totalLife*100);
if(percent>=75){
barColor='green';
}else if(percent>=33){
barColor='orange';
}else{
barColor='red';
}
pBar=(<div className="progress">
<span>{this.props.stats.life}/{this.props.stats.totalLife}</span>
<div className="bar-wrap"> <div className={'bar '+barColor} style={{width:percent+'%'}}></div></div>
</div>
);
}
return(
<div className="avatar" id="target-enemy">
<div className="avatar-header">
<div className={"avatar-img avatar-"+(this.props.name!=null?this.props.className:'none')}>
</div>
<div className="avatar-content">
<p className="avatar-name">
{name}
</p>
<p className="avatar-description">
{description}
</p>
</div>
</div>
{this.props.more!=null?(<a
className={"btn"+(this.props.name==null?' disabled':'')}
href={link}
target="_blank">
More infos...
</a>):''}
<table>
<tbody>
<tr>
<th><span className="ico ico-stat-level"></span>Level:</th>
<td>
{level}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-life"></span>Life:</th>
<td>
{pBar}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-strength"></span>Strength:</th>
<td>
{strength}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-damage"></span>Damage:</th>
<td>
{damage}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-armor"></span>Armor:</th>
<td>
{armor}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-celerity"></span>Celerity:</th>
<td>
{celerity}
</td>
</tr>
</tbody>
</table>
</div>
);
}
}
class Grid extends React.Component{
render(){
var rowId=-1;
var gridRows=[];
for(let i in this.props.grid){
gridRows.push(
<Row
rowData={this.props.grid[i]}
cells={this.props.cells}
key={i}
y={i}
items={this.props.items}
/>
)
}
return(
<div id="grid" className="panel">
{gridRows}
</div>
);
}
}
class MapInfos extends React.Component{
constructor(props){
super(props);
}
render(){
return(
<div className="panel" id="mapInfos">
<h2>Cave:</h2>
<p id="mapName"><small>Level {this.props.mapLevel}: </small>{this.props.mapName}</p>
<p><span className="ico ico-boss"></span>{this.props.items.boss!=undefined?this.props.items.boss:0}/{this.props.orig.boss}</p>
<p><span className="ico ico-enemy"></span>{this.props.items.enemy!=undefined?this.props.items.enemy:0}/{this.props.orig.enemy}</p>
<p><span className="ico ico-chest"></span>{this.props.items.chest!=undefined?this.props.items.chest:0}/{this.props.orig.chest}</p>
<p><span className="ico ico-health"></span>{this.props.items.health!=undefined?this.props.items.health:0}/{this.props.orig.health}</p>
<p><span className="ico ico-water"></span>Water</p>
<p><span className="ico ico-lava"></span>Lava</p>
</div>
);
}
}
class PlayerStats extends React.Component{
constructor(props){
super(props);
this._goDown=this._goDown.bind(this);
}
render(){
var percent=Math.ceil(this.props.life/this.props.totalLife*100);
var barColor='green';
if(percent>=75){
barColor='green';
}else if(percent>=33){
barColor='orange';
}else{
barColor='red';
}
var pBar='';
if(this.props.life>0){
pBar=(<div className="progress">
<span>{this.props.life}/{this.props.totalLife}</span>
<div className="bar-wrap"> <div className={'bar '+barColor} style={{width:percent+'%'}}></div></div>
</div>)
}
var button='';
var btnClass=" btn-green"
var btnText="Go deeper in the cave..."
var bntOnClick=this._goDown;
var hostiles=(this.props.items.enemy || 0)+(this.props.items.boss||0);
if(hostiles > 0){
btnClass=" disabled";
btnText=hostiles+ ' hostile(s) remaining...';
bntOnClick=null;
}
var button=(<button className={'btn btn-block'+btnClass} onClick={bntOnClick}>> {btnText} <</button>);
return(
<div className="avatar" id="target-player">
<div className="avatar-header">
<div className="avatar-img avatar-player">
</div>
<div className="avatar-content">
<p className="avatar-name">
{this.props.name}
</p>
<p className="avatar-description">
{this.props.description}
</p>
</div>
</div>
<table>
<tbody>
<tr>
<th><span className="ico ico-stat-level"></span>Level:</th>
<td>
{this.props.level}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-life"></span>Life:</th>
<td>
{pBar}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-xp"></span>Experience:</th>
<td>
{this.props.experience}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-strength"></span>Strength:</th>
<td>
{this.props.strength}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-damage"></span>Damage:</th>
<td>
{this.props.damage}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-armor"></span>Armor:</th>
<td>
{this.props.armor}
</td>
</tr>
<tr>
<th><span className="ico ico-stat-celerity"></span>Celerity:</th>
<td>
{this.props.celerity}
</td>
</tr>
</tbody>
</table>
{button}
</div>
);
}
_goDown(){
this.props.goDown();
}
}
class QuestLog extends React.Component{
constructor(props){
super(props);
}
render(){
var content=[]
for(let i=0; i<this.props.messages.length; i++){
content.push(<div className={'message-'+this.props.messages[i].type} key={this.props.messages[i].id}>{this.props.messages[i].message}</div>);
}
return(
<div className="panel">
<div id="questLog">
{content}
</div>
</div>
);
}
}
class Reset extends React.Component{
render(){
if(this.props.gameOver==false){
return null;
}
return(
<div>
<div className="overlay"/>
<div id="reset" className="panel">
<p>{this.props.message}</p>
<button className="btn btn-block btn-green" onClick={this._reset.bind(this)}>Restart</button>
</div>
</div>
);
}
_reset(){
this.props.reset();
}
}
class Row extends React.Component{
render(){
var cells=[];
for(let x in this.props.rowData){
var position=x+':'+this.props.y;
var itemId=null;
var item=null;
cells.push(
<Cell
cellData={this.props.cells[position]}
position={position}
key={position}
items={this.props.items}
/>
);
}
return (
<div className="map-row">
{cells}
</div>
);
}
}
class App extends React.Component{
render(){
if(this.state.loading){
return (
<div>
<div class="overlay"></div>
<div id="loading"></div>
</div>);
}else{
return(
<div>
<div className="panel message-info" id="infoMessage">
<p>
This "game" is a React Challenge for <a href="https://freecodecamp.com" target="_blank">FreeCodeCamp</a>. You can find the sources on the <a href="https://github.com/mtancoigne/FCC-15-Rogelike-dungeon-crawler"target="_blank">GitHub</a> repository.
<a href="https://github.com/mtancoigne/FCC-15-Rogelike-dungeon-crawler/issues" target="_blank">Issues and feedback are welcome</a>.
</p>
<p>
As everything is randomly generated, you may have sometimes trouble to win, so watch for the availables chests and lives...<br/> And walk on lava if you feel you have no choices, it's not that bad.
</p>
<p>All code and sprites created by Manuel Tancoigne, code is under the <a href="https://opensource.org/licenses/MIT" target="_blank">MIT license</a> and sprites assigned under a <a href="https://creativecommons.org/licenses/by-nc-sa/3.0/" target="_blank"><img src="http://mirrors.creativecommons.org/presskit/buttons/80x15/png/by-nc-sa.png" height="15px" title="Creative Commons BY-NC-SA license" alt="Creative Commons BY-NC-SA license" /></a></p>
<button className="btn btn-block btn-blue" onClick={this._hideInfo}>Got it, close this now.</button>
</div>
<h1>The Big Crawl <small className="info">~ Kill 'em all for next level. ~</small></h1>
<MapInfos
items={this._countItems()}
orig={this.state.startItems}
mapName={this.state.mapName}
mapLevel={this.state.mapLevel}
/>
<div id="main">
<div id="menu">
<PlayerStats
name={this.state.items['player'].name}
description={this.state.items['player'].description}
link={this.state.items['player'].link}
level={this.state.items['player'].stats.level}
life={this.state.items['player'].stats.life}
totalLife={this.state.items['player'].stats.totalLife}
experience={this.state.items['player'].stats.experience}
damage={this.state.items['player'].stats.damage}
strength={this.state.items['player'].stats.strength}
armor={this.state.items['player'].stats.armor}
celerity={this.state.items['player'].stats.celerity}
items={this._countItems()}
goDown={this._goDown}
/>
<EnemyInfos
level={this.state.currentEnemy!=null?this.state.items[this.state.currentEnemy].stats.level:null}
name={this.state.currentEnemy!=null?this.state.items[this.state.currentEnemy].name:null}
description={this.state.currentEnemy!=null?this.state.items[this.state.currentEnemy].description:null}
link={this.state.currentEnemy!=null?this.state.items[this.state.currentEnemy].more:null}
stats={this.state.currentEnemy!=null?this.state.items[this.state.currentEnemy].stats:null}
className={this.state.currentEnemy!=null?this.state.items[this.state.currentEnemy].className:null}
/>
</div>
<div id="game">
<Grid
gridWidth={this.state.grid[0].length}
gridHeight={this.state.grid[1].length}
grid={this.state.vp}
cells={this.state.cells}
items={this.state.items}
/>
</div>
<div>
<QuestLog messages={this.state.messages}/>
</div>
<div>
<Reset gameOver={this.state.gameOver} reset={this._reset} message={this.state.endGameMessage}/>
</div>
</div>
</div>
);
}
}
constructor(props){
super(props);
this._reset=this._reset.bind(this);
this._goDown=this._goDown.bind(this);
}
componentWillMount(){
this._reset();
}
/**
* Generates the map, place the objects,...
*
* @param {promise-Callback} cb the callback that handles the promise
* @return {Promise} Returns a promise
*/
_generate(options){
if(options==undefined){options={player:null, mapLevel:0};}
return new Promise((resolve, reject)=>{
let error=false;
let map=new MapGen(this.props.options);
map.createMap();
map.createRooms();
map.removeSmallRooms(300);
let player={};
if(options.player){
player=options.player;
}else{
player=Object.assign({}, DEFAULT_ITEM, {
name:'John',
description: 'You, with your funny hat of destruction.',
canMove:true,
className:'player',
stats: Object.assign({}, DEFAULT_STATS, this._levelStats(1)),
})
}
var mapLevel=-1;
if(options.mapLevel){
mapLevel+=options.mapLevel;
}
map.addItems({player: player}, true);
let walkableCells=map._getWalkableCells(false);
let walkableSafeCells=map._getWalkableCells(true);
let playerLevel=player.stats.level;
// Creating enemies
let enemies=new Object(this._createEnemies('enemy', playerLevel, walkableSafeCells.length, 2));
let boss=new Object(this._createBoss(playerLevel, 1));
map.addItems(enemies, true);
map.addItems(boss, true);
// Creating items
for(let i in ITEMS){
if(ITEMS[i].type=='item'){
let items=this._createItems(i, walkableSafeCells.length, 2);
map.addItems(items, true);
}
}
// Appliying random cell style
for(let i in map.cells){
if(map.cells[i].type.name=='wall'){
if(Math.random()>0.9){
map.cells[i].type.classNames.push('wall-'+Math.floor(Math.random()*4));
}
}else if(map.cells[i].type.name=='floor'){
if(Math.random()>0.9){
map.cells[i].type.classNames.push('floor-'+Math.floor(Math.random()*4));
}
}
}
// Testing superpowers
// map.items.player.stats.damage=1000;
// map.items.player.stats.celerity=1000;
// map.items.player.stats.life=1;
resolve ({
mapLevel: mapLevel,
map:map,
messages:[],
vp:[],
startItems:[],
currentEnemy:null,
gameOver:false,
loading:false,
mapName:this._mapName(),
grid:map.grid,
items:map.items,
cells:this._discoverAroundPlayer(map.items, map.grid, map.cells),
startItems:this._countItems(map.items),
vp:this._viewportGrid(map.grid, map.items),
});
});
}
/**
* Generates a new map and resets this.state
*/
_reset(options){
this.setState({loading:true});
this._generate(options)
.then(map => this.setState(map))
.catch(error => console.error(error));
}
/**
* Generates an array representing the viewport (which portion of the map to be displayed)
* Use grid and items vars when you generate before the state is set.
*
* @param array grid - The grid from MapGen.grid. If null, will use the state grid.
* @param obj items - The object containing the items. If null, will use state items.
*
* @return array - An array similar to MapGen's grid, but smaller.
*/
_viewportGrid(grid=null, items=null){
if(!grid){
grid=this.state.grid;
}
if(!items){
items=this.state.items;
}
var pos=items.player.position.split(':');
// Determine theorical VP origin
var posX=pos[0]-Math.floor((this.props.vpSize-1)/2);
var posY=pos[1]-Math.floor((this.props.vpSize-1)/2);
var vp=[];
// Check if vp is coherent
if(posX < 0){posX=0};
if(posX > grid[0].length - this.props.vpSize){
// Check if vp>grid
if(this.props.vpSize>grid[0].length){
posX=0;
}else{
posX = grid[0].length - this.props.vpSize
}
};
if(posY < 0){posY=0};
if(posY > grid.length - this.props.vpSize){
if(this.props.vpSize>grid.length){
posY=0;
}else{
posY = grid.length - this.props.vpSize;
}
}
for(let y=0; y<this.props.vpSize; y++){
var row=[];
for(let x=0; x<this.props.vpSize; x++){
//let nPos=(x+posX)+':'+(y+posY);
row[String(x+posX)]=(grid[y+posY][x+posX]);
}
vp[String(y+posY)]=row;
}
return vp;
}
/**
* Creates a given number of bosses.
* The bosses level is higher than the playerLevel.
*
* @param int playerLevel - Player level to base to boss generation on
* @param int number - Number of bosses to return;
*
* @return an object of items like {boss_1:{boss object}, boss_2:{boss object},...}
*/
_createBoss(playerLevel, number){
var bosses={};
var bossStats={};
for (let i=0; i<number; i++){
let boss=BOSSES[Math.floor(Math.random()*BOSSES.length)];
bossStats=this._levelStats(playerLevel+5);
bosses['boss_'+i]=Object.assign({}, DEFAULT_ITEM, {
name:boss.name,
description:boss.description,
more:boss.more,
canMove:true,
stats: Object.assign({}, DEFAULT_STATS, bossStats),
className: 'boss',
type:boss.type,
})
}
//this._conslog('system','Created '+ number+ ' bosses');
return bosses;
}
/**
* Generates stats for a character of a given level.
*
* @param int level - Level you want the char to be.
*
* @return {stats} - A stats object
*/
_levelStats(level){
var stats={}
// experience
var experience=0
if(level==2){
experience=20;
}else if(level>2){
experience=20
for(let i=0; i<level; i++){
experience*=2;
}
}
var life= 50+(level*50);
var damage= 9+level;
var strength= 1+level;
var armor= 1+level;
var giveXp= 10*level;
var celerity= 1+level;
return {
level:level,
life:life,
totalLife:life,
damage:damage,
strength:strength,
armor:armor,
experience:experience,
giveXp:giveXp,
celerity:celerity,
};
}
/**
* Checks if a character gained a level and apply modifications
*
* @param string target - Character to check.
*
* @return bool - True if the character gained a level.
*/
_hasLeveledUp(target){
let tStats=this.state.items[target].stats;
// Compare levels
let theorical=this._determineLevel(tStats.experience);
if(theorical>tStats.level){
// Calculates new stats
let bStats=this._levelStats(tStats.level);
let nStats=this._levelStats(theorical);
// Update the stats
tStats.level=theorical;
tStats.totalLife=nStats.life;
tStats.life=nStats.life;
tStats.damage+=(nStats.damage-bStats.damage)
tStats.strength+=(nStats.strength-bStats.strength)
tStats.armor+=(nStats.armor-bStats.armor)
tStats.celerity+=(nStats.celerity-bStats.celerity)
let items=this.state.items;
items[target].stats=tStats;
this.setState({items:items});
this._conslog('system', '---');
this._conslog('success', 'You gained one level !');
return true;
}
return false;
}
/**
* Returns a character's level, given its experience. This is a very basic calculus
*
* @param int experience - Actual experience
*
* @return int - The character's level
*/
_determineLevel(experience){
if(experience < 20){return 1};
// return 0;
let level=0;
let done=false;
while(!done){
experience/=2;
level++;
done=(experience-10<0);
}
return level;
}
/**
* Creates ennemies.
* Ennemies have a small chance of being of a superior level
* The number will be a percent of walkable cells on the map
* The enemy name/description is from the global ENEMY_NAMES array
*
* @param string type - prefix for the items array. Can be any string you want.
* @param int playerLevel - Actual player level for stats generation
* @param int walkableCells - Number of walkable cells (get it with MapGen.getWalkableCells())
* @param float enemiesPercent - Percentage of ennemies to create
* @param int number - Override the percentage calculation and fixes the number to create
*
* @return {object} - A list like {type_1:{enemy obj}, type_2;{enemy obj}, ...}
*/
_createEnemies(type, playerLevel, walkableCells, enemiesPercent, number){
// Number of enemies to generate:
var nb=(number!=undefined)?number:Math.floor(enemiesPercent*walkableCells/100);
// Create enemies
var enemies=[];
for(let i=0; i<nb; i++){
// Select random enemy
let enemy=ENEMIES[Math.floor(Math.random()*ENEMIES.length)];
let enemyStats={};
// Generate random states
var randomGen=Math.floor(Math.random()*101);
if(randomGen<80){
enemyStats=this._levelStats(playerLevel);
}else if (randomGen<80) {
enemyStats=this._levelStats(playerLevel+1);
}else if(randomGen<95) {
enemyStats=this._levelStats(playerLevel+2);
}else{
enemyStats=this._levelStats(playerLevel+3);
}
enemies[type+'_'+i]=Object.assign({}, DEFAULT_ITEM, {
name:enemy.name,
description:enemy.description,
more:enemy.more,
canMove:true,
stats: Object.assign({}, DEFAULT_STATS, enemyStats),
className: type,
type:enemy.type,
});
}
//this._conslog('system', "Created "+nb+' "'+type+'"')
return enemies;
}
/**
* Creates items.
* The number will be a percent of walkable cells on the map
*
* @param string name - Item name from the ITEMS global array
* @param int walkableCells - Number of walkable cells (get it with MapGen.getWalkableCells())
* @param float percent - Percentage of items to create
* @param int number - Override the percentage calculation and fixes the number to create
*
* @return {object} - A list like {name_1:{item obj}, name_2;{item obj}, ...}
*/
_createItems(name, walkableCells, percent, number){
var nb=0;
var items={};
if(number!=undefined){
nb=number
}else{
nb=Math.floor((walkableCells*Math.random()*percent)/100)
}
for(let i=0; i<nb; i++){
items[name+'_'+i]=Object.assign({}, DEFAULT_ITEM, ITEMS[name]);
}
//this._conslog('system', 'Created '+nb+' "'+name+'"');
return items;
}
/**
* Handles the player's move, given the key pressed.
* This function handles the following events:
* - Checks if player can go where he want,
* - Initiate combat if any
* - Handle item picking if any
* - Move the enemies
* - Update the state for all this.
*
* @param int keyCode - Key code
*/
_move(keyCode){
if(this.state.gameOver){return false}
var playerPos=this.state.items.player.position.split(':')
var playerX=Number(playerPos[0]);
var playerY=Number(playerPos[1]);
var nextX=playerX;
var nextY=playerY;
switch(keyCode){
case 38: // Up
nextY--;
break;
case 39: // Right
nextX++;
break;
case 40: // Down
nextY++;
break;
case 37:
nextX--;
break;
}
var nextPos=nextX+':'+nextY;
// Check if cell is walkable
if(this.state.cells[nextPos].type.isWalkable===true){
// Check cell content: if enemy stick to it
var preventMove=false;
for(let i in this.state.items){
if(this.state.items[i].position===nextPos && i!='player'){
if(HOSTILES.indexOf(this.state.items[i].type) > -1){
preventMove=true;
this._combat(i);
}else if(this.state.items[i].type==='item'){
var items=this.state.items;
this._conslog('info', 'You picked up a '+items[i].name);
switch (items[i].effect) {
case '_potion_life':
this._potion_life();
break;
case '_token_armor':
this._token_armor();
break;
case '_token_damage':
this._token_damage();
break
case '_token_strength':
this._token_strength();
break;
case '_token_celerity':
this._token_celerity();
break;
default:
this._conslog('fatal', 'Unknown item...');
}
delete items[i];
this.setState({items:items});
}else{
this._conslog('info', 'You walked on '+this.state.items[i].name);
}
}
}
if(!preventMove){
if(this.state.currentEnemy!=null){
this.setState({currentEnemy:null});
}
var items=this.state.items;
items.player.position=nextPos;
this.setState({items:items, cells:this._discoverAroundPlayer()});
var life=this._doDamages(this.state.cells[nextPos].type.damage, 'player');
if(life==0){
this._gameOver('You died, burnt by hot lava.');
}
this.setState({vp:this._viewportGrid()});
}
}
// Move ennemies
var items=this.state.items;
for(let i in this.state.items){
if(this.state.items[i].canMove && i!=this.state.currentEnemy && i!='player'){
var pos=this.state.items[i].position.split(':')
var possibilities=this._getWalkableCellsAround(pos[0],pos[1], null, null, items, true);
if(possibilities.length>0){
var newPos=possibilities[Math.floor(Math.random()*possibilities.length)];
items[i].position=newPos[0]+':'+newPos[1];
}
}
}
this.setState({items:items});
}
/**
* Returns cells on which an ennemy can move. Enemies are clever, they won't
* walk on damaging cells.
*
* @param int x - Inital X position
* @param int y - Initial Y position
* @param {cells list} cells - Cell list
* @param {items list} items - Items list
* @param bool stillIfPlayer - If true, returns an empty array so the enemy
* don't move when the player is near it.
*
* @return array - An array of coordinates of empty cells like [[x1, y1], [x2, y2],...]
*/
_getWalkableCellsAround(x,y, grid, cells, items, stillIfPlayer){
x=Number(x);
y=Number(y);
// Don't move if the player is around.
if(!stillIfPlayer){
stillIfPlayer=false;
}
if(!grid){
grid=this.state.grid;
}
if(!cells){
cells=this.state.cells;
}
if(!items){
items=this.state.items;
}
var inside=[ [x-1, y], [x+1, y], [x, y-1], [x, y+1] ];
var results=[];
// Looking around
for(let i in inside){
let newX=inside[i][0];
let newY=inside[i][1];
if(newX>=0 && newY>=0 && newX<grid[0].length && newY<grid.length){
if(cells[newX+':'+newY].type.isWalkable === true && cells[newX+':'+newY].type.damage<=0){
var nb=0;
for(let j in items){
if(items[j].position==newX+':'+newY){
nb++;
if(j == 'player' && stillIfPlayer){
// don't Move
return [];
}
}
}
if(nb==0){
results.push(inside[i]);
}
}
}
}
return results;
}
/**
* Handles the player's death
*
* @param string msg - Message to display in log an on the gameOver panel.
*/
_gameOver(msg){
this.setState({gameOver:true, endGameMessage:msg});
this._conslog('system', '---');
this._conslog('fatal', msg);
}
_goDown(){
// Save player stats
var player=JSON.parse(JSON.stringify(this.state.items.player));
this._reset({player: player, mapLevel:this.state.mapLevel});
// Increment map level
// Reset
/*this.setState({gameOver:true, endGameMessage:msg});
this._conslog('system', '---');
this._conslog('success', msg);*/
}
/**
* Counts items of each types and returns an array
*
* @param obj items - Items obj from MapGen. If null, will use this.state.items
*
* @return array - An array like ["className":number, "otherClassName":nb];
*/
_countItems(items=null){
if(!items){items=this.state.items;}
var out=[]
for(let i in items){
if(out[items[i].className]==undefined){
out[items[i].className]=0;
}
out[items[i].className]++;
}
return out;
}
/**
* Engage combat with an enemy.
* - Calculates who hit first
* - hit
* - If the other character is still alive, fight back.
*
* @param string target - Name of the target, to be found in state.items
*/
_combat(target){
var pl=this.state.items.player;
var tg=this.state.items[target];
// Define who engage first
var playerFirst=(pl.stats.celerity>tg.stats.celerity);
// Lock the enemy
this.setState({currentEnemy:target});
// Log
this._conslog((playerFirst?'info':'danger'), '---');
this._conslog('danger', 'You engaged with a...');
this._conslog('danger', ' - level '+this._determineLevel(this.state.items[target].stats.experience)+' '+this.state.items[target].name);
this._conslog((!playerFirst?'info':'danger'), (playerFirst?'You':'The enemy')+' attacked first');
// Do the damages
var isDead=this._doAttack(playerFirst?'player':target, playerFirst?target:'player');
// Check death
if(!isDead){
playerFirst=!playerFirst;
this._conslog((playerFirst?'info':'danger'), (playerFirst?'You':'The enemy')+' fought back !');
isDead=this._doAttack(playerFirst?'player':target, playerFirst?target:'player');
}
}
/**
* Actually do damage to a character.
* - Handles death and state change in the items list
*
* @param int damage - Amount of damage
* @param string target - Name of targeted character in item list
*
* @return int - Character's life after taking the damages
*/
_doDamages(damage, target){
if(damage>0){
var items=this.state.items;
// Eyecandy efect on target
if(target=='player' || target==this.state.currentEnemy){
var div='#target-'+(target=='player'?'player':'enemy');
$(div).css('backgroundColor', 'rgba(200, 0, 0,0.5)');
var interval=setInterval(function () {
$(div).css('backgroundColor', 'rgb(234, 234, 234)');
clearInterval(interval);
}, 200);
$(div).fadeTo(100, 0.5).fadeTo(100,1);
}
// Remove life
items[target].stats.life=items[target].stats.life-damage;
// Check state
if(items[target].stats.life <= 0){
// Copy position
let pPos=items[target].position;
// Nice tomb
var tomb='';
switch(items[target].type){
case 'player':
tomb='tomb_stone_player';
break;
case 'enemy':
tomb='tomb_stone';
break;
case 'boss':
tomb='tomb_stone_boss';
break;
}
items[target]=Object.assign({}, DEFAULT_ITEM, ITEMS[tomb]);
// Re-set position
items[target].position=pPos;
}
this.setState({items:items});
}
return this.state.items[target].stats.life || 0;
}
/**
* Attack a target and manage resulting state:
* - Xp gain for player
* - Death of player
*
* @param string attacker - Attacker identifier in this.state.items
* @param string target - Target identifier in this.state.items
*
* @return bool - True on target's death, otherwise false.
*/
_doAttack(attacker, target){
// Copy some values:
var tName=this.state.items[target].name;
var tXp=this.state.items[target].stats.giveXp;
// Flag to know if the target is dead
var dead=false;
// See if the attacker is the player
var playerFirst=(attacker=="player");
// Calculates the damages
var dmg=this._calcDamages(attacker, target);
// Get the new life from state
var currentLife=this.state.items[target].life;
// Log action and result
if( dmg > 0){
this._conslog((playerFirst?'info':'danger'),'...and did '+dmg+' damage !');
// Apply damages
currentLife=this._doDamages(dmg, target);
//Check results
if(currentLife<=0){
this._conslog(playerFirst?'success':'fatal', (!playerFirst? 'You': tName) +' died');
if(playerFirst){
// Get a fresh list of items
var items=this.state.items;
// Add experience to player
items['player'].stats.experience+=tXp;
this._conslog('success', '... You gained '+tXp+' experience points');
// Update the state
this.setState({items:items, currentEnemy:null});
// Check the current level
this._hasLeveledUp('player');
// Check the remaining enemies
/*var objects=this._countItems();
if(objects['enemy']==undefined && objects['boss']==undefined){
this._gameOver('You win !')
}*/
}else{
this._gameOver('You have been defeated.');
}
return true
}
}else{
this._conslog((playerFirst?'danger':'info'),'...but failed. !');
}
return false;
}
/**
* Calculates the damages an attacker do on a target
* @todo Add a failing possibility percentage
*
* @param string attacker - Attacker identifier in this.state.items
* @param string target - Target identifier in this.state.items
*
* @return int - The amount of damage
*/
_calcDamages(attacker, target){
var f=this.state.items[attacker].stats;
var s=this.state.items[target].stats;
// Calculates the damage
return Math.ceil(((10+f.strength)/10)*f.damage-s.armor);
}
/**
* Adds messages to the quest log
*
* @param string type - Message type (info, success, danger fatal)
* @param string message - The message. use '---' to create a separator
*/
_conslog(type, message){
var messages=this.state.messages
if(message=='---'){
messages.push({id:messages.length, type:'hr', message:'::--::--::'});
}else{
messages.push({id:messages.length, type:type, message:message});
}
this.setState({messages:messages});
}
/**
* Changes the discovered property of cells around player.
*
* @param object items - Object containing the items, from MapGen. If null, will use state.items.
* @param array grid - Grid from MapGen.If null, will use state.grid.
* @param object cells - Object containing cells from MapGen. If null, will use state.cells.
*/
_discoverAroundPlayer(items=null, grid=null, cells=null){
if(!items){items=this.state.items;}
if(!grid){grid=this.state.grid;}
if(!cells){cells=this.state.cells;}
var position=items.player.position;
var playerPos=position.split(':')
var playerX=Number(playerPos[0]);
var playerY=Number(playerPos[1]);
var mapWidth=grid[0].length;
var mapHeight=grid.length;
var matrix=[
[-1,-3], [ 0,-3], [ 1,-3],
[-2,-2], [-1,-2], [ 0,-2], [ 1,-2], [ 2,-2],
[-3,-1], [-2,-1], [-1,-1], [ 0,-1], [ 1,-1], [ 2,-1], [ 3,-1],
[-3, 0], [-2, 0], [-1, 0], [ 0, 0], [ 1, 0], [ 2, 0], [ 3, 0],
[-3, 1], [-2, 1], [-1, 1], [ 0, 1], [ 1, 1], [ 2, 1], [ 3, 1],
[-2, 2], [-1, 2], [ 0, 2], [ 1, 2], [ 2, 2],
[-1, 3], [ 0, 3], [ 1, 3]
];
var matrix2=[
[-1,-4], [ 0,-4], [ 1,-4],
[-2,-3], /*[-1,-3], [ 0,-3], [ 1,-3],*/ [ 2,-3],
[-3,-2], /*[-2,-2], [-1,-2], [ 0,-2], [ 1,-2], [ 2,-2],*/ [ 3,-2],
[-4,-1], /*[-3,-1], [-2,-1], [-1,-1], [ 0,-1], [ 1,-1], [ 2,-1], [ 3,-1],*/ [ 4,-1],
[-4, 0], /*[-3, 0], [-2, 0], [-1, 0], [ 0, 0], [ 1, 0], [ 2, 0], [ 3, 0],*/ [ 4, 0],
[-4, 1], /*[-3, 1], [-2, 1], [-1, 1], [ 0, 1], [ 1, 1], [ 2, 1], [ 3, 1],*/ [ 4, 1],
[-3, 2], /*[-2, 2], [-1, 2], [ 0, 2], [ 1, 2], [ 2, 2],*/ [ 3, 2],
[-2, 3], /*[-1, 3], [ 0, 3], [ 1, 3],*/ [ 2, 3],
[-1, 4], [ 0, 4], [ 1, 4],
];
for(let i=0; i<matrix.length; i++){
let newX=playerX+matrix[i][0];
let newY=playerY+matrix[i][1];
if(newX>=0 && newX<=mapWidth-1 && newY>=0 && newY<=mapHeight-1){
cells[newX+':'+newY].type.discovered='visible';
}
}
for(let i=0; i<matrix2.length; i++){
let newX=playerX+matrix2[i][0];
let newY=playerY+matrix2[i][1];
if(newX>=0 && newX<=mapWidth-1 && newY>=0 && newY<=mapHeight-1){
if(cells[newX+':'+newY].type.discovered=='no'){
cells[newX+':'+newY].type.discovered='half';
}
}
}
return cells;
}
/**
* Update player's stat when i picks a life potion
*/
_potion_life(){
var quantity=50;
var items=this.state.items;
items['player'].stats.life=(items['player'].stats.life+quantity>items['player'].stats.totalLife?items['player'].stats.totalLife:items['player'].stats.life+quantity);
this.setState({items:items});
}
/**
* Update player's stat when i picks an armor token
*/
_token_armor(){
var items=this.state.items;
items['player'].stats.armor+=1;
this.setState({items:items});
}
/**
* Update player's stat when i picks a damage token
*/
_token_damage(){
var items=this.state.items;
items['player'].stats.damage+=1;
this.setState({items:items});
}
/**
* Update player's stat when i picks a strength token
*/
_token_strength(){
var items=this.state.items;
items['player'].stats.strength+=1;
this.setState({items:items});
}
/**
* Update player's stat when i picks a celerity token
*/
_token_celerity(){
var items=this.state.items;
items['player'].stats.celerity+=1;
this.setState({items:items});
}
/**
* Generates a map name
*
* @return string - A wonderful map name
*/
_mapName(){
var adj=MAP_NAME_ADJECTIVES[Math.floor(Math.random()*MAP_NAME_ADJECTIVES.length)];
var noun=MAP_NAME_NOUNS[Math.floor(Math.random()*MAP_NAME_NOUNS.length)];
var adverb=MAP_NAME_ADVERBS[Math.floor(Math.random()*MAP_NAME_ADVERBS.length)];
return adj + ' ' + noun + ' of ' + adverb;
}
/**
* Hides the info area using jquery as not important.
*/
_hideInfo(){
$('#infoMessage').toggle()
}
}
/*******************************************************************************
GAME CONSTANTS FOR INITIAL generation
*/
// Some names for level naming
const MAP_NAME_ADJECTIVES=['Dark', 'Scealled', 'Obscure', 'Final', 'Chaotic', 'Rancid', 'Moldy', 'Hellish'];
const MAP_NAME_NOUNS=['floor', 'cave', 'den', 'lair', 'cavern'];
const MAP_NAME_ADVERBS=['destruction', 'sorrow', 'chaos', 'Cthulhu', 'famine', 'death', 'pain', 'Doom', 'Hell'];
const DEFAULT_PLAYER_STATS={life:50, totalLife:50, damage:10, strength:1, armor:1, level:1, celerity:1, experience:0,};
const DEFAULT_ITEM={
name:'NO NAME',
description: 'NO DESCRIPTION',
type:null,
canMove:false,
storable: false,
consumable: false,
position:null,
effect:null,
inventory:[],
stats:{},
className:'item',
};
const DEFAULT_STATS={life:-1, totalLife:0, damage:0, strength:0, armor:0, level:1, experience:0, giveXp:0, celerity:1};
const CELL_TYPES={
wall: {name:'wall', isWalkable:false, classNames:['wall'], discovered:'no', isBaseCell:true},
floor: {name:'floor', isWalkable:true, classNames:['floor'], discovered:'no', isBaseCell:true},
lava: {name:'lava', isWalkable:true, classNames:['lava'], discovered:'no', damage:1},
water: {name:'water', isWalkable:true, classNames:['water'], discovered:'no'},
};
const ITEMS={
tomb_stone_player: {name:'You', description: 'Something took your life away... And broke your hat...', type:'special', storable:false, consumable:false, className:'tomb-player', stats:{life:0}},
tomb_stone: {name:'a corpse', description: 'A dead enemy', type:'special', storable:false, consumable:false, className:'tomb', stats:{life:0}},
tomb_stone_boss: {name:'a large corpse', description: 'A dead powerful enemy', type:'special', storable:false, consumable:false, className:'tomb-boss', stats:{life:0}},
life_potion: {name:'Life potion', description: 'Gives you 50 points of life', type:'item', storable:false, consumable:true, className:'health', effect:'_potion_life'},
token_strength: {name:'Token of strength', description: 'Adds 1 to your strength', type:'item', storable:true, consumable:false, className:'chest', effect:'_token_strength'},
token_damage: {name:'Token of damage', description: 'Adds 1 to your damage', type:'item', storable:true, consumable:false, className:'chest', effect:'_token_damage'},
token_armor: {name:'Token of armor', description: 'Adds 1 to your armor', type:'item', storable:true, consumable:false, className:'chest', effect:'_token_armor'},
token_celerity: {name:'Token of celerity', description: 'Adds 1 to your celerity', type:'item', storable:true, consumable:false, className:'chest', effect:'_token_celerity'},
};
// Some bosses :
const BOSSES=[
{type:'boss', name:'Joker', description:'The Joker is a fictional supervillain who appears in American comic books published by DC Comics. The character was created by Jerry Robinson, Bill Finger, and Bob Kane, and first appeared in Batman #1.', more:'http://www.ranker.com/review/joker/2498261?ref=name_320416'},
{type:'boss', name:'Hannibal Lecter', description:'Dr. Hannibal Lecter is a character in a series of suspense novels by Thomas Harris. Lecter was introduced in the 1981 thriller novel Red Dragon as a forensic psychiatrist and cannibalistic serial killer.', more:'http://www.ranker.com/review/hannibal-lecter/1125580?ref=name_320416'},
{type:'boss', name:'Jack Torrence', description:'John Daniel "Jack" Torrance is a fictional character in the 1977 novel The Shining by Stephen King.', more:'http://www.ranker.com/review/jack-torrance/1254047?ref=name_320416'},
{type:'boss', name:'Freddy Kruegger', description:'Fred "Freddy" Krueger is the main antagonist of the A Nightmare on Elm Street film series. He first appeared in Wes Craven\'s A Nightmare on Elm Street as a burnt serial killer who uses a glove armed with razors to kill his victims in their dreams', more:'http://www.ranker.com/review/freddy-krueger/1022376?ref=name_320416'},
];
// Some enemies :
const ENEMIES=[
{type:'enemy', name:'Gloom Lad', description:'I have all the characteristics of a human being: blood, flesh, skin, hair; but not a single, clear, identifiable emotion, except for greed and disgust.', more:null},
{type:'enemy', name:'Killer Woman', description:'I visited your home this morning after you\'d left. I tried to play husband. I tried to taste the life of a simple man. It didn\'t work out, so I took a souvenir... her pretty head.', more:null},
{type:'enemy', name:'Master Man', description:'The point is ladies and gentlemen that greed, for lack of a better word, is good.', more:null},
{type:'enemy', name:'Necrotic Murderer', description:'Human beings are a disease, a cancer of this planet. You\'re a plague and we are the cure.', more:null},
{type:'enemy', name:'Sickness Master', description:'I\'m going to do something now they used to do in Vietnam. It\'s called making a head on a stick.', more:null},
{type:'enemy', name:'Viral Shade', description:'If Mr. McMurphy doesn\'t want to take his medication orally, I\'m sure we can arrange that he can have it some other way. But I don\'t think that he would like it.', more:null},
];
const HOSTILES=['boss', 'enemy']
const MAP_OPTIONS={x:5, y:5, passes:3, cleanLevel:5, wallPercent:10, sameSubCellPercent:80, cssPrefix:'map-', cellTypes:CELL_TYPES};
var appRendered=ReactDOM.render(<App options={MAP_OPTIONS} startLevel='1' newGame={true} playerStats={null} vpSize={15}/>, document.getElementById('app'));
$(document).keydown(function(e) {
if([37,38,39,40].indexOf(e.keyCode)>-1){
e.preventDefault();
appRendered._move(e.keyCode);
var element = document.getElementById("questLog");
element.scrollTop = element.scrollHeight;
return false;
}
return true;
});
Also see: Tab Triggers