css Audio - Active file-generic CSS - Active Generic - Active HTML - Active JS - Active SVG - Active Text - Active file-generic Video - Active header Love html icon-new-collection icon-person icon-team numbered-list123 pop-out spinner split-screen star tv

Pen Settings

CSS Base

Vendor Prefixing

Add External Stylesheets/Pens

Any URL's added here will be added as <link>s in order, and before the CSS in the editor. If you link to another Pen, it will include the CSS from that Pen. If the preprocessor matches, it will attempt to combine them before processing.

+ add another resource

You're using npm packages, so we've auto-selected Babel for you here, which we require to process imports and make it all work. If you need to use a different JavaScript preprocessor, remove the packages in the npm tab.

Add External Scripts/Pens

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.

+ add another resource

Use npm Packages

We can make npm packages available for you to use in your JavaScript. We use webpack to prepare them and make them available to import. We'll also process your JavaScript with Babel.

⚠️ This feature can only be used by logged in users.

Code Indentation

     

Save Automatically?

If active, Pens will autosave every 30 seconds after being saved once.

Auto-Updating Preview

If enabled, the preview panel updates automatically as you code. If disabled, use the "Run" button to update.

HTML Settings

Here you can Sed posuere consectetur est at lobortis. Donec ullamcorper nulla non metus auctor fringilla. Maecenas sed diam eget risus varius blandit sit amet non magna. Donec id elit non mi porta gravida at eget metus. Praesent commodo cursus magna, vel scelerisque nisl consectetur et.

            
              <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">&nbsp;<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">&nbsp;<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}>&gt; {btnText} &lt;</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;
});
            
          
!
999px
🕑 One or more of the npm packages you are using needs to be built. You're the first person to ever need it! We're building it right now and your preview will start updating again when it's ready.
Loading ..................

Console