Pen Settings

HTML

CSS

CSS Base

Vendor Prefixing

Add External Stylesheets/Pens

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.

+ add another resource

JavaScript

Babel includes JSX processing.

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

Packages

Add Packages

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.

Behavior

Auto Save

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.

Format on Save

If enabled, your code will be formatted when you actively save your Pen. Note: your code becomes un-folded during formatting.

Editor Settings

Code Indentation

Want to change your Syntax Highlighting theme, Fonts and more?

Visit your global Editor Settings.

HTML

              
                <div class="page">
		<div class="wall-bg">
			<ul>
				<li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li>
			</ul>
			<div class="wall-poster">
				<h1>game <span>over</span></h1>
				<i class="mr-akabei">
					<div class="mr-akabei-content">
						<span class="mr-akabei-eye-first"></span>
						<span class="mr-akabei-eye-second"></span>
						<span class="mr-akabei-bottom-1"></span>
						<span class="mr-akabei-bottom-2"></span>
						<span class="mr-akabei-bottom-3"></span>
						<span class="mr-akabei-bottom-4"></span>
						<span class="mr-akabei-bottom-5"></span>
						<span class="mr-akabei-bottom-6"></span>
						<span class="mr-akabei-bottom-7"></span>
					</div>
					<i class="point-first"></i>
					<i class="point-second"></i>
					<i class="point-third"></i>
					<i class="point-four"></i>
					<i class="point-last"></i>
				</i>
				<i class="mr-pacman">
					<i class="point-first"></i>
					<i class="point-second"></i>
					<i class="point-third"></i>
					<i class="point-four"></i>
				</i>
				<span>1980</span>
			</div>
			<div class="wall-desk">
				<div class="timer">
					<i class="timer-shadow"></i>
					<div class="timer-content">
						<div class="timer-hr">
							<div class="timer-digits"></div>
						</div>
						<i class="timer-hr-right"></i>
					</div>
					<i class="timer-right"></i>
					<i class="timer-hr-first"></i>
					<i class="timer-hr-second"></i>
					<i class="timer-hr-third"></i>
					<i class="timer-hr-last"></i>
				</div>
				<i class="wall-desk-shadow"></i>
				<i class="wall-desk-bottom"></i>
				<i class="wall-desk-front"></i>
				<i class="wall-desk-right"></i>
			</div>
		</div>
		<div class="floor-bg"><div class="floor-hr"><i></i></div></div>

		<div class="tv-content">
			<div class="tv">
				<div class="tv-top"><i class="item-left"></i><i class="item-right"></i></div>
				<div class="tv-shadow"></div>
				<div class="tv-right"></div>
				<div class="tv-bottom">
					<ul><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li></ul>
					<i></i>
				</div>
				<div class="tv-screen"><a href="" class="pw-btn"></a><div class="tv-hr"><div class="tv-hr-2"><div class="tv-hr-3">
					<div class="tv-glass">
						<canvas></canvas>
						<div class="tv-glass-vintage">
							<ul><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li><li></li></ul>
							
							<i class="tv-noise"></i><i class="tv-noise-second"></i>
							<i class="tv-glow"></i>
						</div>
					</div>
					<div class="tv-flashing">
						<i class="tv-flashing-left"></i>
						<i class="tv-flashing-bottom"></i>
						<i class="tv-flashing-bottom-placeholder"></i>
					</div>
				</div></div></div></div>
			</div>
			<div class="tv-desk">
				<i class="tv-desk-shadow"></i>
				<i class="tv-desk-item-left-shadow"></i>
				<i class="tv-desk-item-left"></i>
				<i class="tv-desk-item-right-shadow"></i>
				<i class="tv-desk-item-right"></i>
				<i class="tv-desk-item-rear-shadow"></i>
				<i class="tv-desk-item-rear"></i>
				<i class="tv-desk-top"></i>
				<i class="tv-desk-front"></i>
				<i class="tv-desk-right"></i>
			</div>
		</div>

		<div class="console">
			<i class="console-shadow"></i>
			<div class="console-top">
				<i class="console-game-top"></i>
				<div class="console-top-panel">
					<i class="console-top-panel-item-1"></i>
					<i class="console-top-panel-item-2"></i>
					<i class="console-top-panel-item-3"></i>
					<i class="console-top-panel-item-4"></i>
					<i class="console-top-panel-item-5"></i>
					<i class="console-top-panel-item-6"></i>
					<i class="console-top-panel-item-7"></i>
					<i class="console-top-panel-item-8"></i>
				</div>
			</div>
			<i class="console-right-top"></i>
			<i class="console-right-bottom"></i>
			<div class="console-front-panel">
				<div class="front-panel-top">
					<i class="console-game"></i>
					<i class="console-power-dark"></i>
				</div>
				<div class="front-panel-bottom">
					<i class="console-power-indicator"></i>
					<i class="console-btn-first"></i>
					<i class="console-btn-second"></i>
					<div class="console-power">
						<div class="console-power-plug"><i></i></div>
						<i class="console-power-cable"></i>
					</div>
				</div>
			</div>
		</div>

		<div class="player-1">
			<div class="player-hand-left">
				<div class="hand-content">
					<i class="hand-left"></i><i class="hand-inner"></i>
					<i class="hair-item-1"></i>
					<i class="hair-item-2"></i>
					<i class="hair-item-3"></i>
					<i class="hair-item-4"></i>
					<i class="hair-item-5"></i>
					<i class="hair-item-6"></i>
					<i class="hair-item-7"></i>
					<i class="hair-item-8"></i>
					<i class="hair-item-9"></i>
					<i class="hair-item-10"></i>
					<i class="hair-item-11"></i>
					<i class="hair-item-12"></i>
					<i class="hair-item-13"></i>
					<i class="hair-item-14"></i>
					<i class="hair-item-15"></i>
					<i class="hair-item-16"></i>
				</div>
				<div class="finger-content"><i class="finger-placeholder"></i><i class="finger-touch"></i></div>
			</div>
			<div class="player-hand-right">
				<div class="hand-content">
					<i class="hand-left"></i><i class="hand-inner"></i>
					<i class="hair-item-1"></i>
					<i class="hair-item-2"></i>
					<i class="hair-item-3"></i>
					<i class="hair-item-4"></i>
					<i class="hair-item-5"></i>
					<i class="hair-item-6"></i>
					<i class="hair-item-7"></i>
					<i class="hair-item-8"></i>
					<i class="hair-item-9"></i>
					<i class="hair-item-10"></i>
					<i class="hair-item-11"></i>
					<i class="hair-item-12"></i>
					<i class="hair-item-13"></i>
					<i class="hair-item-14"></i>
					<i class="hair-item-15"></i>
					<i class="hair-item-16"></i>
				</div>
				<div class="finger-content"><i class="finger-placeholder"></i><i class="finger-touch"></i></div>
			</div>

			<div class="controller-nes"><i class="controller-nes-cable"></i><div class="in-controller-nes">
				<i class="controller-nes-pad"></i>
				<div class="controller-nes-option">
					<i class="hr-first"></i>
					<i class="hr-second"></i>
					<i class="hr-third"></i>
					<i class="hr-last"></i>
					<div class="controller-nes-option-btn"></div>
				</div>
				<div class="controller-nes-btn"><i></i><i></i></div>
			</div></div>
		</div>
	</div>
              
            
!

CSS

              
                // LAYOUT
// ---------
.page{
  position:relative;
  position:absolute;
  left:50%;
  top:50%;
  margin:-270px 0 0 -480px;
  .size(960px, 540px);
  overflow:hidden;
}




// WALL
// ---------
.wall-bg{
  position:absolute;
  left:0;
  top:0;
  .sq(100%);

  ul{
    .sq(100%);
    .c;
  }
  li{
    float:left;
    .size(32px, 100%);
    background:#120a20;
  }
  li:nth-child(odd){
    background:#160d24;
  }
}


// WALL - POSTER
.wall-poster{
  position:absolute;
  left:620px;
  top:-70px;
  .size(200px, 252px);
  background:#190e28;
  background:#211138;
  overflow:hidden;
  
  h1{
    position:absolute;
    left:14px;
    top:84px;
    .font-size(24, 1);
    color:#291c3b;
    color:#4d2d79;
    text-transform:uppercase;
    text-align:left;

    span{
      position:relative;
      bottom:10px;
      display:block;
      .font-size(40);
      text-align:left;
    }
  }
  span{
    position:absolute;
    right:0;
    bottom:10px;
    .size(75px, 40px);
    .font-size(30, 40px);
    color:#291c3b;
    color:#4d2d79;
    text-align:center;
    letter-spacing:-1.4px;
  }
}
.mr-akabei{
  position:absolute;
  right:12px;
  bottom:80px;
  .size(77px, 74px);
  .border-radius(38px 38px 0 0);
  background:#241637;
  background:#4d2d79;

  i{
    position:absolute;
    top:80%;
    right:100%;
    margin-top:-5px;
    .sq(10px);
    .border-radius(6px);
    background:#241637;
    background:#43226f;
  }
  .point-first{
    margin-right:10px;
  }
  .point-second{
    margin-right:30px;
  }
  .point-third{
    margin-right:50px;
  }
  .point-four{
    margin-right:70px;
  }
  .point-last{
    margin-right:90px;
  }
}
.mr-akabei-content{
  .mr-akabei-eye-first,
  .mr-akabei-eye-second{
    position:absolute;
    top:26px;
    .sq(20px);
    .border-radius(10px);
    background:#2f2143;
    background:#8454c7;

    &:before{
      content:'';
      position:absolute;
      left:3px;
      top:3px;
      .sq(9px);
      .border-radius(6px);
      background:#1c112b;
    }
  }
  .mr-akabei-eye-first{
    left:12px;
  }
  .mr-akabei-eye-second{
    right:12px;
  }
  .mr-akabei-bottom-1{
    position:absolute;
    left:0;
    top:68px;
    .size(11px, 20px);
    .border-radius(0 0 6px 6px);
    background:#241637;
    background:#4d2d79;
  }
  .mr-akabei-bottom-2{
    position:absolute;
    left:11px;
    top:62px;
    .size(11px, 20px);
    .border-radius(6px);
    background:#190e28;
  }
  .mr-akabei-bottom-3{
    position:absolute;
    left:22px;
    top:68px;
    .size(11px, 24px);
    .border-radius(0 0 6px 6px);
    background:#241637;
    background:#4d2d79;
  }
  .mr-akabei-bottom-4{
    position:absolute;
    left:33px;
    top:62px;
    .size(11px, 20px);
    .border-radius(6px);
    background:#190e28;
  }
  .mr-akabei-bottom-5{
    position:absolute;
    left:44px;
    top:68px;
    .size(11px, 20px);
    .border-radius(0 0 6px 6px);
    background:#241637;
    background:#4d2d79;
  }
  .mr-akabei-bottom-6{
    position:absolute;
    left:55px;
    top:66px;
    .size(11px, 24px);
    .border-radius(6px);
    background:#190e28;
  }
  .mr-akabei-bottom-7{
    position:absolute;
    left:66px;
    top:68px;
    .size(11px, 24px);
    .border-radius(0 0 6px 6px);
    background:#241637;
    background:#4d2d79;
  }
}
.mr-pacman{
  position:absolute;
  right:70px;
  bottom:10px;
  .sq(40px);
  .border-radius(20px);
  background:#241637;
  background:#4d2d79;

  &:after{
    content:'';
    position:absolute;
    left:0;
    top:50%;
    margin-top:-8px;
    .sq(0);
    border-style: solid;
    border-width: 8px 0 8px 20px;
    border-color: transparent transparent transparent #1c112b;
    border-color: transparent transparent transparent #211138;
  }
  i{
    position:absolute;
    top:50%;
    right:100%;
    margin-top:-5px;
    .sq(10px);
    .border-radius(6px);
    background:#241637;
    background:#43226f;
  }
  .point-first{
    margin-right:10px;
  }
  .point-second{
    margin-right:30px;
  }
  .point-third{
    margin-right:50px;
  }
  .point-four{
    margin-right:70px;
  }
}



// WALL - DESK
.wall-desk{
  position:absolute;
  left:30px;
  top:90px;
  .size(200px, 32px);
}
.wall-desk-bottom{
  position:absolute;
  left:0;
  top:5px;
  .size(146px, 10px);
  background-color: #160c25;
  background-image: -webkit-linear-gradient(#3e2b56 0%, #180f28 100%);
  background-image: -o-linear-gradient(#3e2b56 0%, #180f28 100%);
  background-image: linear-gradient(#3e2b56 0%, #180f28 100%);

  .skew(80deg);
  .transform-origin(left top);
}
.wall-desk-shadow{
  position:absolute;
  left:16px;
  top:-1px;
  .size(146px, 10px);
  background-color:#0c0915;

  .skew(74deg);
  .transform-origin(left top);
}
.wall-desk-front{
  position:absolute;
  left:0;
  top:0;
  .size(140px, 5px);
  background:#26173a;
}
.wall-desk-right{
  position:absolute;
  top:0;
  left:140px;
  .size(56px, 5px);
  background:#201233;

  .skew(0deg, 10deg);
  .transform-origin(left top);
}

// WALL - DESK - TIMER
.timer{
  position:absolute;
  left:30px;
  bottom:100%;
  margin-bottom:3px;
  .size(105px, 34px);
  
  &:before{
    content:'';
    position:absolute;
    left:50%;
    top:-2px;
    .size(36px, 2px);
    margin-left:-18px;
    .border-radius(2px 2px 0 0);
    background:#57328d;
  }
  .timer-content{
    position:relative;
    padding:5px;
    .sq(100%);
    .border-radius(18px);
    background-color: #38274e;
    z-index:2;
  }
  .timer-right{
    position:absolute;
    left:100%;
    top:0;
    margin-left:-20px;
    .size(40px, 34px);
    .border-radius(0 18px 20px 0);
    border-top-right-radius:60px 30px;

    background-color: #2b1843;
  }
  .timer-hr{
    position:relative;
    .sq(100%);
    .border-radius(18px);
    background:#160d24;
    .box-shadow(inset 2px 0 0 0 #0c0915);

    &:before{
      content:'';
      position:absolute;
      left:10px;
      top:50%;
      .sq(10px);
      margin-top:-5px;
      .border-radius(6px);
      background:#57328d;
      .box-shadow(2px 0 0 #341760);
    }
  }
  .timer-hr-first,
  .timer-hr-second,
  .timer-hr-third,
  .timer-hr-last{
    position:absolute;
    top:100%;
    left:18px;
    .size(8px, 4px);
    .border-radius(0 0 4px 4px);
    background:#241736;
  }
  .timer-hr-second{
    left:80px;
  }
  .timer-hr-third,
  .timer-hr-last{
    left:36px;
    background:#180e26;
  }
  .timer-hr-last{
    left:102px;
  }
}
.timer-digits{
  padding:4px 8px 0 24px;
  .sq(100%);
  #font > #family > .orbitron();
  .font-size(16, 20px);
  font-weight:500;
  color:#ee8b3c;
  text-align:center;
  .opacity(60);
  animation:timeblink 1s infinite;
}
@-webkit-keyframes timeblink {
  0%  {.opacity(30);}
  49% {.opacity(30);}
  50% {.opacity(60);}
}
@-moz-keyframes timeblink {
  0%  {.opacity(30);}
  49% {.opacity(30);}
  50% {.opacity(60);}
}
@-o-keyframes timeblink {
  0%  {.opacity(30);}
  49% {.opacity(30);}
  50% {.opacity(60);}
}
@keyframes timeblink {
  0%  {.opacity(30);}
  49% {.opacity(30);}
  50% {.opacity(60);}
}
.timer-shadow{
  position:absolute;
  left:40px;
  top:10px;
  .size(105px, 26px);
  .border-radius(20px);
  background:#0c0915;
}
.timer-hr-right{
  position:absolute;
  left:100%;
  top:50%;
  .size(10px, 1px);
  .rotate(14deg);
  background:#160d24;

  &:before,
  &:after{
    content:'';
    position:absolute;
    left:0;
    .size(6px, 1px);
    background:#160d24;
  }
  &:before{
    top:-3px;
  }
  &:after{
    left:2px;
    top:3px;
  }
}




// FLOOR
// ---------
.floor-bg{
  position:absolute;
  left:0;
  bottom:0;
  .size(100%, 145px);
  background-color: #110c1e;
  background-image: -webkit-linear-gradient(#06040b 0%, #110c1e 100%);
  background-image: -o-linear-gradient(#06040b 0%, #110c1e 100%);
  background-image: linear-gradient(#06040b 0%, #110c1e 100%);
}

// FLOOR - HR
.floor-hr{
  position:absolute;
  left:0;
  top:-38px;
  .size(100%, 38px);
  background:#221532;

  &:after,
  &:before{
    content:'';
    position:absolute;
    left:0;
    top:0;
    .size(100%, 2px);
    background:#160d24;
  }
  &:after{
    top:5px;
  }
  i{
    position:absolute;
    left:0;
    top:-7px;
    .size(100%, 7px);
    background:#191028;
  }
}



// TV!
// ---------
.tv-content{
  position:absolute;
  left:190px;
  bottom:110px;
  .size(534px, 318px);
}
.tv{
  position:absolute;
  left:60px;
  top:0;
  .size(340px, 274px);
}
.tv-shadow{
  position:absolute;
  left:80px;
  top:50px;
  .size(320px, 260px);
  .border-radius(20px);
  background:#0c0915;
  transform: perspective( 700px ) rotateX( 150deg );

  -webkit-filter: blur(2px);
  -moz-filter: blur(2px);
  -o-filter: blur(2px);
  -ms-filter: blur(2px);
  filter: blur(2px);
}
.tv-right{
  position:absolute;
  left:100%;
  top:22px;
  margin-left:-30px;
  .size(97px, 232px);
  .border-radius(10px);
  background-color: #36274a;
  background-image: -webkit-linear-gradient(#2b1d3d 0%, #1a0e29 100%);
  background-image: -o-linear-gradient(#2b1d3d 0%, #1a0e29 100%);
  background-image: linear-gradient(#2b1d3d 0%, #1a0e29 100%);
  -webkit-transform: perspective( 200px ) rotateY( 45deg );
  -moz-transform: perspective( 200px ) rotateY( 45deg );
  -ms-transform: perspective( 200px ) rotateY( 45deg );
  -o-transform: perspective( 200px ) rotateY( 45deg );
  transform: perspective( 200px ) rotateY( 45deg );
  z-index:3;  
}
.tv-bottom{
  position:absolute;
  left:28px;
  bottom:-20px;
  .size(282px, 24px);
  .border-radius(0 0 9px 9px);
  background-color: #620e32;
  background-image: -webkit-linear-gradient(bottom, #211333 0%, #110222 100%);
  background-image: -o-linear-gradient(bottom, #211333 0%, #110222 100%);
  background-image: linear-gradient(to top, #211333 0%, #110222 100%);
  z-index:2;
  
  ul{
    position:absolute;
    left:0;
    top:8px;
    padding-left:30px;
    .size(100%, 11px);
    .c;

    li{
      float:left;
      margin:0 4px 0 0;
      .size(3px, 100%);
      .border-radius(2px);
      background:#12081f;
    }
  }
  i{
    position:absolute;
    left:100%;
    bottom:0;
    .size(40px, 22px);
    background:#130622;

    .skew(0deg, -28deg);
    .transform-origin(left bottom);
  }
}


// TV - SCREEN
.tv-screen{
  position:absolute;
  left:0;
  top:0;
  .sq(100%);
  .border-radius(24px);
  background-color: #7f1b46;
  background-image: -webkit-linear-gradient(top right, #4a3863 0%, #34214c 100%);
  background-image: -o-linear-gradient(top right, #4a3863 0%, #34214c 100%);
  background-image: linear-gradient(to bottom left, #4a3863 0%, #34214c 100%);
  z-index:4;
  
  // SAFARI FIX
  transform: translateZ(1000px);
  transform-style: preserve-3d;
}
.pw-btn{
  position:absolute;
  left:36px;
  bottom:4px;
  .size(10px, 3px);
  background:#ee8b3c;
}
.tv-hr{
  position:absolute;
  left:12px;
  top:9px;
  .size(316px, 254px);
  .border-radius(18px);
  background:#57328d;
}
.tv-hr-2{
  position:absolute;
  left:3px;
  top:3px;
  .size(310px, 248px);
  .border-radius(18px);
  background:#1c1330;
}
.tv-hr-3{
  position:absolute;
  left:4px;
  top:4px;
  .size(302px, 240px);
  .border-radius(18px);
  background:#110a20;
}
.tv-glass{
  position:absolute;
  left:6px;
  top:5px;
  .size(290px, 230px);

  .border-radius(12px);
  background:black;
  overflow:hidden;
  -webkit-transform:translate3d(0,0,0);

  canvas{
    position:absolute;
    left:0;
    top:0;
    .scale(0.42);
    .opacity(50);
    .transform-origin(left top);
    .transition(all .2s ease-in-out);

    .show-player &{
      .opacity(100);
    }
  }
}
iframe{
  position:absolute;
  left:0;
  top:0;
  .sq(100%);
  border:0;
  .border-radius(9px);
  overflow:hidden;
  overflow-x:hidden;
  overflow-y:hidden;
}
.ytframe{
  left:-20%;
  .size(140%, 100%);
}

.tv-loading{
  display:block;
  padding:10px 0 0 18px;
  .font-size(24);
  color:@white;
  .opacity(80);
}

// TV - VINTAGE
.tv-glass-vintage{
  position:absolute;
  left:0;
  top:0;
  .sq(100%);
  pointer-events:none;

  ul{
    position:absolute;
    left:0;
    top:0;
    .sq(100%);

    li{
      position:relative;
      .size(100%, 4%);

      &:after{
        content:'';
        position:absolute;
        left:0;
        top:50%;
        .size(100%, 2px);
        background:@white;
        .opacity(07);
      }
    }
  }
}
.tv-noise,
.tv-noise-second{
  position:absolute;
  left:0;
  top:0;
  .sq(100%);
  background-image:url();
  animation: blink .3s infinite linear;
}
.tv-noise-second{
  background-image:url();
  .opacity(0);
  -moz-animation-direction: reverse;
  -webkit-animation-direction: reverse;
  animation-direction: reverse;
}
@-webkit-keyframes blink {
  0%   {.opacity(0);}
  50%  {.opacity(15);}
  100% {.opacity(30);}
}
@-moz-keyframes blink {
  0%   {.opacity(0);}
  50%  {.opacity(15);}
  100% {.opacity(30);}
}
@-o-keyframes blink {
  0%   {.opacity(0);}
  50%  {.opacity(15);}
  100% {.opacity(30);}
}
@keyframes blink {
  0%   {.opacity(0);}
  50%  {.opacity(15);}
  100% {.opacity(30);}
}

.tv-glow{
  position:absolute;
  top:10px;
  right:10px;
  .border-radius(18px 0 0 18px);
  .sq(120px);
  background-image:radial-gradient(circle at 100% 100%, rgba(204,0,0,0) 59px, #fff 60px);
  background-size: 50% 50%;
  background-repeat: no-repeat;
  .opacity(20);
  filter: blur(3px);
  .rotate(90deg);
}

.tv-flashing{
  position:absolute;
  left:0;
  top:0;
  .sq(100%);
  pointer-events:none;

  -webkit-transform:translate3d(0,0,0);

  i{
    -webkit-filter: blur(10px);
    -moz-filter: blur(10px);
    -o-filter: blur(10px);
    -ms-filter: blur(10px);
    filter: blur(10px);

    .opacity(0);

    animation: blink2 3s infinite linear;
  }
  .tv-flashing-left{
    position:absolute;
    right:100%;
    top:-4px;
    margin-right:-73px;
    .size(200px, 254px);

    transform: perspective( 400px ) rotateY( 64.4deg );

    background: -moz-linear-gradient(left,  rgba(255,255,255,0) 0%, rgba(255,255,255,1) 100%);
    background: -webkit-linear-gradient(left,  rgba(255,255,255,0) 0%, rgba(255,255,255,1) 100%);
    background: linear-gradient(to right,  rgba(255,255,255,0) 0%, rgba(255,255,255,1) 100%);
  }
  .tv-flashing-bottom{
    position:absolute;
    left:-27px;
    top:100%;
    margin-top:-40px;
    .size(300px, 100px);

    transform: perspective( 600px ) rotateX( 60deg ) skew(-28deg);

    background: -moz-linear-gradient(bottom,  rgba(255,255,255,0) 0%, rgba(255,255,255,1) 100%);
    background: -webkit-linear-gradient(bottom,  rgba(255,255,255,0) 0%, rgba(255,255,255,1) 100%);
    background: linear-gradient(to top,  rgba(255,255,255,0) 0%, rgba(255,255,255,1) 100%);
  }
  .tv-flashing-bottom-placeholder{
    position:absolute;
    left:0;
    top:100%;
    margin:-28px 0 0 8px;
    .sq(30px);
    background-image:radial-gradient(circle at 100% 0, rgba(204,0,0,0) 14px, #fff 15px);
    background-size: 50% 50%;
    background-repeat: no-repeat;
  }
}
@-webkit-keyframes blink2 {
  0% {
    .opacity(0)
  }
  5% {
    .opacity(10);
  } 
  7% {
    .opacity(0);
  }
  14% {
    .opacity(0);
  }
  22% {
    .opacity(10);
  }
  26% {
    .opacity(0);
  }
  66% {
    .opacity(0);
  }
  78% {
    .opacity(08);
  }
  82% {
    .opacity(0);
  }
  88% {
    .opacity(08);
  }
  92%{
    .opacity(0);
  }
  100% {
    .opacity(0);
  }
}
@-moz-keyframes blink2 {
  0% {
    .opacity(0)
  }
  5% {
    .opacity(10);
  } 
  7% {
    .opacity(0);
  }
  14% {
    .opacity(0);
  }
  22% {
    .opacity(10);
  }
  26% {
    .opacity(0);
  }
  66% {
    .opacity(0);
  }
  78% {
    .opacity(08);
  }
  82% {
    .opacity(0);
  }
  88% {
    .opacity(08);
  }
  92%{
    .opacity(0);
  }
  100% {
    .opacity(0);
  }
}
@-o-keyframes blink2 {
  0% {
    .opacity(0)
  }
  5% {
    .opacity(10);
  } 
  7% {
    .opacity(0);
  }
  14% {
    .opacity(0);
  }
  22% {
    .opacity(10);
  }
  26% {
    .opacity(0);
  }
  66% {
    .opacity(0);
  }
  78% {
    .opacity(08);
  }
  82% {
    .opacity(0);
  }
  88% {
    .opacity(08);
  }
  92%{
    .opacity(0);
  }
  100% {
    .opacity(0);
  }
}
@keyframes blink2 {
  0% {
    .opacity(0)
  }
  5% {
    .opacity(10);
  } 
  7% {
    .opacity(0);
  }
  14% {
    .opacity(0);
  }
  22% {
    .opacity(10);
  }
  26% {
    .opacity(0);
  }
  66% {
    .opacity(0);
  }
  78% {
    .opacity(08);
  }
  82% {
    .opacity(0);
  }
  88% {
    .opacity(08);
  }
  92%{
    .opacity(0);
  }
  100% {
    .opacity(0);
  }
}



// TV - TOP
.tv-top{
  position:absolute;
  left:0;
  bottom:100%;
  margin-bottom:-4px;
  .size(100%, 100px);
  
  i{
    position:absolute;
    left:50%;
    bottom:0;
    .size(4px, 80px);
    background:#36274a;
    .transform-origin(left bottom);

    &:after{
      content:'';
      position:absolute;
      left:-2px;
      top:-2px;
      .sq(8px);
      .border-radius(4px);
      background:#36274a;
    }
  }
  .item-left{
    .rotate(-35deg);
  }
  .item-right{
    margin-left:-6px;
    height:140px;
    .rotate(45deg);
  }
}


// TV - DESK
.tv-desk{
  position:absolute;
  left:0;
  bottom:0;
  .size(100%, 60px);
}
.tv-desk-top,
.tv-desk-shadow{
  position:absolute;
  left:0;
  bottom:8px;
  .size(420px, 51px);
  background-color: #372757;
  background-image: -webkit-linear-gradient(bottom, #130a21 0%, #07000e 100%);
  background-image: -o-linear-gradient(bottom, #130a21 0%, #07000e 100%);
  background-image: linear-gradient(to top, #130a21 0%, #07000e 100%);

  .skew(-60deg);
  .transform-origin(left bottom);
}
.tv-desk-shadow{
  bottom:-30px;
  background:#050409;
}
.tv-desk-front{
  position:absolute;
  left:0;
  bottom:0;
  .size(412px, 10px);
  background:#1f1131;
}
.tv-desk-right{
  position:absolute;
  bottom:0;
  left:412px;
  .size(92px, 10px);
  background:#28183d;

  .skew(0deg, -28deg);
  .transform-origin(left bottom);
}
.tv-desk-item-left,
.tv-desk-item-right,
.tv-desk-item-rear{
  position:absolute;
  left:40px;
  top:100%;
  margin-top:-16px;
  .size(14px, 58px);
  background:#28183d;
  transform: perspective( 100px ) rotateX( 120deg );
}
.tv-desk-item-left-shadow,
.tv-desk-item-right-shadow,
.tv-desk-item-rear-shadow{
  position:absolute;
  left:52px;
  top:100%;
  margin-top:25px;
  .size(8px, 46px);
  background:#170c26;

  transform: skew(0deg, -28deg) perspective( 100px ) rotateX( 120deg );

  .transform-origin(left top);
}
.tv-desk-item-right{
  left:370px;
}
.tv-desk-item-right-shadow{
  left:382px;
}
.tv-desk-item-rear{
  margin-top:-52px;
  left:450px;
  background:#170c26;
}
.tv-desk-item-rear-shadow{
  margin-top:-12px;
  left:462px;
  background:#12091f;
}



// CONSOLE
.console{
  position:absolute;
  left:60px;
  bottom:6px;
  .size(132px, 40px);
}
.console-shadow{
  position:absolute;
  right:0;
  bottom:0;
  margin-right:10px;
  .size(100px, 100px);

  background:black;

  .skew(-69deg);
  .transform-origin(left bottom);
}
.console-top{
  position:absolute;
  left:0;
  bottom:40px;
  .size(132px, 46px);
  background-color: #342448;
  background-image: -webkit-linear-gradient(#180f28 0%, #342448 60%);
  background-image: -o-linear-gradient(#180f28 0%, #342448 60%);
  background-image: linear-gradient(#180f28 0%, #342448 60%);

  .skew(-60deg);
  .transform-origin(left bottom);
}
.console-top-panel{
  position:absolute;
  left:85px;
  bottom:0;
  .size(30px, 46px);
  background:#1b1127;

  i{
    display:block;
    .size(100%, 6px);
    border-bottom:1px solid #160e20;
  }
  .console-top-panel-item-1{
    height:4px;
  }
  .console-top-panel-item-4,
  .console-top-panel-item-5,
  .console-top-panel-item-6{
    border-color:#291d3a;
    background:#2d1e3f;
  }
}
.console-game-top{
  position:absolute;
  left:10px;
  bottom:0;
  .size(60px, 16px);
  border:1px solid #2b1d3e;
  border-bottom:0;
}
.console-right-top{
  position:absolute;
  bottom:16px;
  left:131px;
  .size(80px, 24px);
  background:#28183d;

  .skew(0deg, -30deg);
  .transform-origin(left bottom);
  
  &:after{
    content:'';
    position:absolute;
    top:0;
    left:100%;
    .sq(0);
    border-style: solid;
    border-width: 24px 0 0 5px;
    border-color: transparent transparent transparent #28183d;
  }
}
.console-right-bottom{
  position:absolute;
  bottom:0;
  left:131px;
  .size(80px, 16px);
  background:#1b1127;

  .skew(0deg, -30deg);
  .transform-origin(left bottom);
  
  &:after{
    content:'';
    position:absolute;
    top:0;
    left:100%;
    .sq(0);
    border-style: solid;
    border-width: 16px 6px 0 0;
    border-color: #1b1127 transparent transparent transparent;
  }
}
.console-front-panel{
  position:relative;
  .sq(100%);
}

// CONSOLE - FRONT PANEL - TOP
.front-panel-top{
  position:absolute;
  left:0;
  top:0;
  .size(100%, 50%);
  background:#392652;

  &:before,
  &:after{
    content:'';
    position:absolute;
    top:0;
    .sq(0);
    border-style: solid;
  }
  &:before{
    right:100%;
    border-width: 0 0 20px 4px;
    border-color: transparent transparent #392652 transparent;
  }
  &:after{
    left:100%;
    border-width: 20px 0 0 4px;
    border-color: transparent transparent transparent #392652;
  }
}
.console-game{
  position:absolute;
  left:8px;
  bottom:6px;
  .size(60px, 14px);
  border:1px solid #2b1d3e;
  border-top:0;

  .skew(-10deg, 0deg);
  .transform-origin(left bottom);
}
.console-power-dark{
  position:absolute;
  left:82px;
  bottom:0;
  .size(30px, 20px);
  background:#231831;

  .skew(-10deg, 0deg);
  .transform-origin(left bottom);
}

// CONSOLE - FRONT PANEL - BOTTOM
.front-panel-bottom{
  position:absolute;
  left:0;
  bottom:0;
  .size(100%, 50%);
  background:#28193b;

  &:before,
  &:after{
    content:'';
    position:absolute;
    top:0;
    .sq(0);
    border-style: solid;
  }
  &:before{
    right:100%;
    border-width: 0 4px 20px 0;
    border-color: transparent #28193b transparent transparent;
  }
  &:after{
    left:100%;
    border-width: 20px 4px 0 0;
    border-color: #28193b transparent transparent transparent;
  }
}
.console-power-indicator{
  position:absolute;
  left:9px;
  top:9px;
  .sq(4px);
  .border-radius(2px);
  background:#ee8b3c;
}
.console-btn-first,
.console-btn-second{
  position:absolute;
  left:24px;
  top:7px;
  .size(16px, 8px);
  .border-radius(3px);
  border:1px solid #0e0a19;
  background:#392652;
}
.console-btn-second{
  left:44px;
}

// CONSOLE - FRONT - POWER
.console-power{
  position:absolute;
  right:20px;
  top:0;
  .size(30px, 100%);
  background:#1b1127;
}
.console-power-plug{
  position:absolute;
  left:5px;
  top:4px;
  .size(20px, 12px);
  .border-radius(2px);
  background:#020204;

  i{
    position:absolute;
    left:1px;
    top:2px;
    .size(16px, 10px);
    .border-radius(2px);
    background:#140c1d;
  }
}
.console-power-cable{
  position:absolute;
  right:14px;
  top:8px;
  .size(40px, 6px);
  .border-radius(10px);
  background:black;

  .skew(0deg, -30deg);
  .transform-origin(right top);
}







// PLAYER
.player-1{
  position:absolute;
  right:40px;
  bottom:-10px;
  .size(310px, 140px);
  .transition(all .5s cubic-bezier(.32,0,.56,1.4));
  
  bottom:-180px;
  transform:perspective( 600px ) rotateY(20deg) rotateX( 80deg ) scale(1.15);

  .show-player &{
    bottom:-10px;
    bottom:0px;
    transform:perspective( 600px ) rotateY(0deg) rotateX(0deg) scale(1.15);
  }
  .press-spacebar &{
    transform:rotate(2deg) scale(1.1);
  }
  .press-w &,
  .press-a &,
  .press-s &,
  .press-d &{
    transform:rotate(-2deg) scale(1.1);
  }
}
.player-hand-left{
  position:absolute;
  left:0;
  bottom:0;
  .size(110px, 100px);
  
  .hand-content{
    position:absolute;
    left:0;
    bottom:-20px;
    .size(100px, 160px);
    
    .hand-inner{
      position:absolute;
      right:4px;
      top:34px;
      .sq(36px);
      .border-radius(30px);
      background:#231733;
      z-index:5;
    }
    .hand-left{
      position:absolute;
      left:0;
      bottom:0;
      .size(60px, 140px);
      border-top-left-radius:100px 200px;
      border-bottom-left-radius:50px 50px;
      background:#2f2140;
      .rotate(10deg);

      .transform-origin(left bottom);
    }
  }
  .finger-content{
    position:absolute;
    left:40px;
    bottom:30px;
    .sq(52px);

    .finger-placeholder{
      position:absolute;
      left:0;
      bottom:0;
      .sq(100%);
      .border-radius(30px);
      background:#2f2140;
      .transition(all .2s ease-in-out);

      .press-w &{
        left:0px;
        bottom:4px;
      }
      .press-a &{
        left:-2px;
      }
      .press-s &{
        left:4px;
      }
      .press-d &{
        left:2px;
      }
    }
    .finger-touch{
      position:absolute;
      left:50%;
      top:50%;
      margin-top:-5px;
      .size(32px, 24px);
      background:#2f2140;
      .rotate(-44deg);
      z-index:10;
      .transform-origin(left top);
      .transition(all .2s ease-in-out);
      
      .press-w &{
        width:50px;
        .rotate(-50deg);
      }
      .press-a &{
        width:30px;
        .rotate(-54deg);
      }
      .press-s &{
        width:38px;
        .rotate(-30deg);
      }
      .press-d &{
        width:50px;
        .rotate(-36deg);
      }
      &:before{
        content:'';
        position:absolute;
        top:0;
        right:-12px;
        .sq(24px);
        .border-radius(12px);
        background:#2f2140;
      }
      &:after{
        content:'';
        position:absolute;
        right:-10px;
        top:4px;
        .size(14px, 16px);
        .border-radius(0 8px 8px 0);
        background:#423256;
      }
    }
  }
  .hair-item-1,
  .hair-item-2,
  .hair-item-3,
  .hair-item-4,
  .hair-item-5,
  .hair-item-6,
  .hair-item-7,
  .hair-item-8,
  .hair-item-9,
  .hair-item-10,
  .hair-item-11,
  .hair-item-12,
  .hair-item-13,
  .hair-item-14,
  .hair-item-15,
  .hair-item-16{
    position:absolute;
    left:48px;
    top:54px;
    .size(1px, 10px);
    background:#1d112c;
    .rotate(50deg);
    z-index:15;
  }
  .hair-item-2{
    margin-left:6px;
    margin-top:8px;
  }
  .hair-item-3{
    margin-left:-10px;
    margin-top:6px;
  }
  .hair-item-4{
    margin-left:-10px;
    margin-top:20px;
  }
  .hair-item-5{
    margin-left:6px;
    margin-top:20px;
    .rotate(38deg);
  }
  .hair-item-6{
    margin-left:8px;
    margin-top:36px;
    .rotate(38deg);
  }
  .hair-item-7{
    margin-left:-10px;
    margin-top:36px;
    .rotate(38deg);
  }
  .hair-item-8{
    margin-left:0px;
    margin-top:32px;
    .rotate(38deg);
  }
  .hair-item-9{
    margin-left:-20px;
    margin-top:22px;
    .rotate(38deg);
  }
  .hair-item-10{
    margin-left:-24px;
    margin-top:34px;
    .rotate(22deg);
  }
  .hair-item-11{
    margin-left:-10px;
    margin-top:56px;
    .rotate(22deg);
  }
  .hair-item-12{
    margin-left:-26px;
    margin-top:52px;
    .rotate(22deg);
  }
  .hair-item-13{
    margin-left:0px;
    margin-top:60px;
    .rotate(22deg);
  }
  .hair-item-14{
    margin-left:-12px;
    margin-top:80px;
    .rotate(22deg);
  }
  .hair-item-15{
    margin-left:-22px;
    margin-top:70px;
    .rotate(22deg);
  }
  .hair-item-16{
    margin-left:-34px;
    margin-top:76px;
    .rotate(22deg);
  }
}
.player-hand-right{
  position:absolute;
  right:0;
  bottom:0;
  .size(110px, 120px);
  
  .hand-content{
    position:absolute;
    right:0;
    bottom:-20px;
    .size(100px, 180px);
    
    .hand-inner{
      position:absolute;
      left:4px;
      top:34px;
      .sq(38px);
      .border-radius(30px);
      background:#231733;
      z-index:5;
    }
    .hand-left{
      position:absolute;
      right:0;
      bottom:0;
      .size(60px, 160px);
      border-top-right-radius:100px 200px;
      
      background:#2f2140;
      .rotate(-10deg);

      .transform-origin(right bottom);
    }
  }
  .finger-content{
    position:absolute;
    right:40px;
    bottom:50px;
    .sq(52px);

    .finger-placeholder{
      position:absolute;
      right:0;
      bottom:0;
      .sq(100%);
      .border-radius(30px);
      background:#2f2140;
      .transition(all .2s ease-in-out);

      .press-spacebar &{
        right:5px;
      }
    }
    .finger-touch{
      position:absolute;
      right:50%;
      top:50%;
      margin-top:-16px;
      .size(36px, 24px);
      background:#2f2140;
      .rotate(36deg);
      z-index:10;
      .transition(all .2s ease-in-out);

      .press-spacebar &{
        width:52px;
        .rotate(30deg);
      }
      &:before{
        content:'';
        position:absolute;
        top:0;
        left:-12px;
        .sq(24px);
        .border-radius(12px);
        background:#2f2140;
      }
      &:after{
        content:'';
        position:absolute;
        left:-10px;
        top:4px;
        .size(14px, 16px);
        .border-radius(8px 0 0 8px);
        background:#423256;
      }
    }
  }
  .hair-item-1,
  .hair-item-2,
  .hair-item-3,
  .hair-item-4,
  .hair-item-5,
  .hair-item-6,
  .hair-item-7,
  .hair-item-8,
  .hair-item-9,
  .hair-item-10,
  .hair-item-11,
  .hair-item-12,
  .hair-item-13,
  .hair-item-14,
  .hair-item-15,
  .hair-item-16{
    position:absolute;
    left:62px;
    top:68px;
    .size(1px, 10px);
    background:#1d112c;
    .rotate(-50deg);
    z-index:15;
  }
  .hair-item-2{
    margin-left:6px;
    margin-top:8px;
  }
  .hair-item-3{
    margin-left:20px;
    margin-top:70px;
    .rotate(-22deg);
  }
  .hair-item-4{
    margin-left:-10px;
    margin-top:20px;
  }
  .hair-item-5{
    margin-left:6px;
    margin-top:20px;
    .rotate(-38deg);
  }
  .hair-item-6{
    margin-left:8px;
    margin-top:36px;
    .rotate(-38deg);
  }
  .hair-item-7{
    margin-left:-10px;
    margin-top:36px;
    .rotate(-38deg);
  }
  .hair-item-8{
    margin-left:20px;
    margin-top:50px;
    .rotate(-22deg);
  }
  .hair-item-9{
    margin-left:-20px;
    margin-top:22px;
    .rotate(-38deg);
  }
  .hair-item-10{
    margin-left:-24px;
    margin-top:34px;
    .rotate(-22deg);
  }
  .hair-item-11{
    margin-left:-10px;
    margin-top:56px;
    .rotate(-22deg);
  }
  .hair-item-12{
    margin-left:-26px;
    margin-top:52px;
    .rotate(-22deg);
  }
  .hair-item-13{
    margin-left:0px;
    margin-top:70px;
    .rotate(-22deg);
  }
  .hair-item-14{
    margin-left:-12px;
    margin-top:80px;
    .rotate(-22deg);
  }
  .hair-item-15{
    margin-left:-22px;
    margin-top:70px;
    .rotate(-22deg);
  }
  .hair-item-16{
    margin-left:-34px;
    margin-top:76px;
    .rotate(-22deg);
  }
}


// CONTROLLER NES
.controller-nes{
  position:absolute;
  top:20px;
  left:68px;
  padding:16px 8px 8px 8px;
  .size(160px, 70px);
  .border-radius(4px);
  background:#493762;
  background-image: -webkit-linear-gradient(bottom left, #34214c 0%, #4a3863 100%);
  background-image: -o-linear-gradient(bottom left, #34214c 0%, #4a3863 100%);
  background-image: linear-gradient(to top right, #34214c 0%, #4a3863 100%);
  .rotate(-10deg);
  .transform-origin(left top);
  z-index:8;
}
.in-controller-nes{
  position:relative;
  .sq(100%);
  background:#201330;
}
.controller-nes-pad{
  position:absolute;
  left:8px;
  bottom:8px;
  .sq(30px);

  &:before,
  &:after{
    content:'';
    position:absolute;
    .border-radius(4px);
    background:#06040b;
  }
  &:before{
    top:0;
    left:50%;
    margin-left:-6px;
    .size(12px, 100%);
  }
  &:after{
    top:50%;
    left:0;
    margin-top:-6px;
    .size(100%, 12px);
  }
}
.controller-nes-option{
  position:absolute;
  left:50px;
  top:0;
  .size(34px, 100%);
  overflow:hidden;

  i{
    position:absolute;
    left:0;
    display:block;
    .size(100%, 8px);
    .border-radius(4px);
    background:#43305b;
  }
  .hr-first{
    top:-2px;
  }
  .hr-second{
    top:7px;
  }
  .hr-third{
    top:16px;
  }
  .hr-last{
    bottom:-2px;
  }
  .controller-nes-option-btn{
    position:absolute;
    left:0;
    top:25px;
    .size(100%, 14px);
    .border-radius(4px);
    background:#57328d;

    &:before,
    &:after{
      content:'';
      position:absolute;
      top:5px;
      .size(10px, 4px);
      .border-radius(2px);
      background:#6c41aa;
    }
    &:before{
      left:4px;
    }
    &:after{
      right:4px;
    }
  }
}
.controller-nes-btn{
  position:absolute;
  right:14px; 
  bottom:8px;
  .size(36px, 14px);

  i{
    position:relative;
    float:left;
    margin:0 0 0 4px;
    .sq(14px);
    .border-radius(2px);
    background:#34214c;

    &:after{
      content:'';
      position:absolute;
      left:50%;
      top:50%;
      margin:-4px 0 0 -4px;
      .sq(8px);
      .border-radius(4px);
      background:#730505;
    }
  }
}











// MIXINS
// ---------

// Clearfix
// --------
.c {
  &:after {
    content:"";
    display:table;
    clear:both;
  }
}

// Sizing shortcuts
// -------------------------
.size(@width: 5px, @height: 5px) {
  width: @width;
  height: @height;
}
.square(@size: 5px) {
  .size(@size, @size);
}
.sq(@size: 5px) {
  .size(@size, @size);
}



// FONTS
// --------------------------------------------------

#font {
  #family {
    .sans-serif() {
      font-family: 'VT323', Georgia, "Times New Roman", Times, serif;
    }
    .orbitron() {
      font-family: 'Orbitron', 'Arial CE', Arial, sans-serif;
    }
  }
  .size(@font-size: 16, @line: 1) {
    @rem: (@font-size / 10);
    font-size: @font-size * 1px;
    font-size: ~"@{rem}rem";
    line-height: @line;
  }

}

.font-size(@font-size: 16, @line: 1) {
  @rem: (@font-size / 10);
  font-size: @font-size * 1px;
  font-size: ~"@{rem}rem";
  line-height: @line;
}





// CSS3 PROPERTIES
// --------------------------------------------------


// Border Radius
.border-radius(@radius: 5px) {
  -webkit-border-radius: @radius;
     -moz-border-radius: @radius;
          border-radius: @radius;
}

// Drop shadows
.box-shadow(@shadow: 0 1px 3px rgba(0,0,0,.25)) {
  -webkit-box-shadow: @shadow;
     -moz-box-shadow: @shadow;
          box-shadow: @shadow;
}

.transition(@transition) {
  -webkit-transition: @transition;
     -moz-transition: @transition;
      -ms-transition: @transition;
       -o-transition: @transition;
          transition: @transition;
}

.rotate(@degrees) {
  -webkit-transform: rotate(@degrees);
     -moz-transform: rotate(@degrees);
      -ms-transform: rotate(@degrees);
       -o-transform: rotate(@degrees);
          transform: rotate(@degrees);
}

.scale(@ratio) {
  -webkit-transform: scale(@ratio);
     -moz-transform: scale(@ratio);
      -ms-transform: scale(@ratio);
       -o-transform: scale(@ratio);
          transform: scale(@ratio);
}

.skew(@x: 0, @y: 0) {
  -webkit-transform: skew(@x, @y);
     -moz-transform: skew(@x, @y);
      -ms-transform: skew(@x, @y);
       -o-transform: skew(@x, @y);
          transform: skew(@x, @y);
}

.transform-origin(@origin) {
  -webkit-transform-origin: @origin;
     -moz-transform-origin: @origin;
       -o-transform-origin: @origin;
          transform-origin: @origin;
}

// Opacity
.opacity(@opacity: 100) {
  opacity: @opacity / 100;
   filter: e(%("alpha(opacity=%d)", @opacity));
}




// HTML
// ---------
html{
  -webkit-box-sizing: border-box;
  -khtml-box-sizing: border-box;
  -moz-box-sizing: border-box;
  -ms-box-sizing: border-box;
  box-sizing: border-box;
}

*,
*::before,
*::after{
  box-sizing: inherit;
}

html, body{
  .sq(100%);
}


// BODY
// ---------
body{
  position:relative;
  min-width:960px;
  min-height:540px;
  margin: 0;
  line-height: 1;
  .font-size(12, 18px);
  #font > #family > .sans-serif();
  color:@basecolor;
  background:#110c1e;
  -webkit-font-smoothing: antialiased;
}


// VARIABLES
// ---------

@basecolor:   #000;

// Colors
@white:       #fff;
@black:       #000;




// RESET
// ---------

html, body { margin: 0; padding: 0; }

h1, h2, h3, h4, h5, h6, p, blockquote, pre, a, abbr, acronym, address, cite, code, del, dfn, em, img, q, s, samp, small, strike, strong, sub, sup, tt, var, dd, dl, dt, li, ol, ul, fieldset, form, label, legend, button, table, caption, tbody, tfoot, thead, tr, th, td { margin: 0; padding: 0; border: 0; font-size: 100%; line-height: 1; font-family: inherit;
}

html {
  font-size: 62.5%;
  -webkit-text-size-adjust: 100%;
      -ms-text-size-adjust: 100%;
}


              
            
!

JS

              
                /*
  ------------------------
  Check out my video of how i created this piece of...
  https://www.twitch.tv/videos/115237628 [3hour]
  https://www.twitch.tv/videos/115351889 [6hour]
  ------------------------
  No IMG needed, just pure CSS!
  ------------------------
  Inspiration: http://bit.ly/8bitgaming
  ------------------------
  Design created by @dominikrezek
  ------------------------
  Game created by @remvst
  http://js13kgames.com/entries/taxi-drift
  ------------------------
*/



// JS
$(document).ready(function() {

    // TIME
    setInterval(function(){ 
        var dt = new Date(),
            hours = dt.getHours(),
            minutes = dt.getMinutes();

        if (hours < 10) {
            hours = "0" + hours;
        }
        if (minutes < 10) {
            minutes = "0" + minutes;
        }

        var time = hours + ":" + minutes;

        $('.timer-digits').html(time);
    }, 1000);
   


    /*
        W =         87
        A =         65
        S =         83
        D =         68

        spacebar =  32
        R =         82
    */
    $(document).on('keydown', function(e){
        var code = e.keyCode,
            delay = 300;

        //console.log(code);

        if (code == 13) {
            $('html').addClass('show-player');
        }

        if (code == 87 || code == 38) {
            $('html').addClass('press-w');

            setTimeout(function() {
                $('html').removeClass('press-w');
            }, delay);
        }
        if (code == 65 || code == 37) {
            $('html').addClass('press-a');

            setTimeout(function() {
                $('html').removeClass('press-a');
            }, delay);
        }
        if (code == 83 || code == 40) {
            $('html').addClass('press-s');

            setTimeout(function() {
                $('html').removeClass('press-s');
            }, delay);
        }
        if (code == 68 || code == 39) {
            $('html').addClass('press-d');

            setTimeout(function() {
                $('html').removeClass('press-d');
            }, delay);
        }
        if (code == 32 || code == 82) {
            $('html').addClass('press-spacebar');

            setTimeout(function() {
                $('html').removeClass('press-spacebar');
            }, delay);
        }

    });
});


var _ = window,
    raf = (function(){
        return  _.requestAnimationFrame       ||
                _.webkitRequestAnimationFrame ||
                _.mozRequestAnimationFrame    ||
                function(c){
                    setTimeout(c, 1000 / 60);
                };
    })(),
    M = Math,
    abs = M.abs,
    to = setTimeout;


function rd(a, b){
    if(b === undefined){
        b = a;
        a = 0;
    }
    return M.random() * (b - a) + a;
};

function rp(a){
    return a[~~(rd(a.length))];
};

function normalizeAngle(a){
    while (a < -Math.PI) a += Math.PI * 2;
    while (a > Math.PI) a -= Math.PI * 2;
    return a;
};

function xt(o, x){
    var r = {};

    // Copying
    for(var i in o){
        r[i] = o[i];
    }

    // Overriding
    for(var i in x){
        r[i] = x[i];
    }

    return r;
};

// Shortcuts
var p = CanvasRenderingContext2D.prototype;
p.fr = p.fillRect;
p.sv = p.save;
p.rs = p.restore;
p.tr = p.translate;
p.lt = p.lineTo;
p.mt = p.moveTo;
p.sc = p.scale;
p.bp = p.beginPath;
p.clg = p.createLinearGradient;
p.rt = p.rotate;
p.ft = p.fillText;

p.alpha = function(x){
    this.globalAlpha = x;
};

p.fs = function(p){
    this.fillStyle = p;
};

p.di = function(i, x, y){
    this.drawImage.apply(this, arguments);
};

// Adding all these functions to the global scope
for(var i in p){
    _[i] = (function(f){
        return function(){
            c[f].apply(c, arguments);
        }
    })(i);
}

function shape(points, color){
    var tx = points[0].x,
        ty = points[0].y;

    c.sv();
    c.tr(tx, ty);

    c.fs(color);
    c.bp();
    c.moveTo(0, 0);
    for(var i = 1 ; i < points.length ; i++){
        c.lineTo(points[i].x - tx, points[i].y - ty);
    }
    c.closePath();
    c.fill();

    c.rs();
};

function cache(w, h, rr, t){
    var c = document.createElement('canvas');
    c.width = w;
    c.height = h;

    var r = c.getContext('2d');
    rr(c, r, w, h);
    //setTimeout((rr(c, r, w, h)), 3000);

    if(t === 'pattern'){
        var p = r.createPattern(c, 'repeat');
        p.width = w;
        p.height = h;
        return p;
    }

    return c;
};

function noop(){};

function limit(x, a, b){
    return M.max(a, M.min(b, x));
};

function shuffle(o){
    for(var j, x, i = o.length; i; j = ~~(M.random() * i), x = o[--i], o[i] = o[j], o[j] = x);
    return o;
};

function dist(x1, y1, x2, y2){
    return Math.sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2));
};



function createCycle(pts){
    // Smooth angles
    var l = 40,
        cur,
        next,
        prev,
        res = [],
        anglePrev,
        angleNext;
    for(var i = 0 ; i < pts.length ; i++){
        cur = pts[i];
        next = pts[(i + 1) % pts.length];
        prev = pts[(i - 1 + pts.length) % pts.length];

        anglePrev = Math.atan2(prev.y - cur.y, prev.x - cur.x);
        angleNext = Math.atan2(next.y - cur.y, next.x - cur.x);

        // Smooth before
        res.push({
            x: cur.x + l * Math.cos(anglePrev),
            y: cur.y + l * Math.sin(anglePrev)
        });
        res.push({
            x: cur.x + l * Math.cos(angleNext),
            y: cur.y + l * Math.sin(angleNext)
        });
    }


    // Linking points together
    for(var i = 0 ; i < res.length ; i++){
        res[i].next = res[(i + 1) % res.length];
    }

    return res[0];
};

function newCar(color, noLights){
    return cache(52, 24, function(c, r){
        with(r){
          var b = r;
            fs('rgba(0,0,0,1)');
            fr(0, 0, c.width, c.height);

            tr((c.width - 50) / 2, (c.height - 22) / 2);

            fs(color);
            fr(0, 0, 50, 22);

            fs('#000');
            fr(10, 3, 5, 16);
            fr(30, 3, 10, 16);
            fr(15, 1, 7, 1);
            fr(23, 1, 7, 1);
            fr(15, 20, 7, 1);
            fr(23, 20, 7, 1);

            if(!noLights){
                fs('#ff0');
                fr(48, 0, 2, 3);
                fr(48, 19, 2, 3);

                fs('#f00');
                fr(0, 0, 2, 3);
                fr(0, 19, 2, 3);
            }
        }
    });
};

var P = {
    w: 700,
    h: 550,
    v: 130
};

function Game(){
    window.rotation = true;

    this.can = document.querySelector('canvas');
    with(this.can){
        var b = this.can;
        width = P.w;
        height = P.h;
    }

    this.ctx = window.c = this.can.getContext('2d');

    this.start();

    // Resizing
    //this.resize();
    //addEventListener('resize', this.resize, false);

    addEventListener('keydown', this.keyDown.bind(this), false);
    addEventListener('keyup', this.keyUp.bind(this), false);

    // Loop
    this.lastFrame = Date.now();
    raf(function(){
        G.cycle();
        raf(arguments.callee);
    });

    this.elapsedList = [];
    this.frameCount = 0;
    this.frameCountStart = Date.now();
};

Game.prototype = {
    start: function(){
        this.world = new World();

        this.menu = new Home();
    },
    restart: function(){
        this.world = new World();

        this.menu = null;
    },
    gameOver: function(){
        this.menu = new End();
    },
    cycle: function(){
        sv();
        sc(this.resolution, this.resolution);

        var n = Date.now(),
            e = (n - this.lastFrame) / 1000;

        //e = M.min(e, 1 / 30);
        this.lastFrame = n;

        this.world.cycle(e);
        if(this.menu){
            this.menu.cycle(e);
        }

        Easing.cycle(e);

        this.frameCount++;
        if(this.frameCount === 200){
            var totalTime = Date.now() - this.frameCountStart;
            var fps = this.frameCount / (totalTime / 1000);
            if(fps < 30){
                this.setResolution(.6);
            }
        }

        /*fillStyle = '#fff';
        font = '20pt Arial';
        textBaseline = 'top';
        textAlign = 'left';
        ft(~~(1 / e) + 'fps', 10, 10);

        this.elapsedList.push(e);
        if(this.elapsedList.length > 100){
            this.elapsedList.shift();
        }

        c.fillStyle = '#000';
        c.strokeStyle = '#fff';
        c.fr(0, 0, this.elapsedList.length * 2, 100);
        c.strokeRect(0, 0, this.elapsedList.length * 2, 100);

        var fps;
        c.strokeStyle = '#fff'
        c.beginPath();
        for(var i = 0, x = 0 ; i < this.elapsedList.length ; i++, x+=2){
            fps = ~~(1 / this.elapsedList[i]);
            //c.fr(x, 100 - (fps / 60) * 100, 2, 2);
            c.lineTo(x, 100 - (fps / 60) * 100);
        }
        c.stroke();*/

        rs();
    },
    newWorld: function(){
        this.world = new World();
    },
    /*resize : function(){
        to(function(){
            var maxWidth = innerWidth,
                maxHeight = innerHeight,

                availableRatio = maxWidth / maxHeight,
                baseRatio = P.w / P.h,
                ratioDifference = abs(availableRatio - baseRatio),
                width,
                height,
                s = document.getElementById('canvascontainer').style;

            if(availableRatio <= baseRatio){
                width = maxWidth;
                height = width / baseRatio;
            }else{
                height = maxHeight;
                width = height * baseRatio;
            }

            s.width = width + 'px';
            s.height = height + 'px';
        },100);
    },*/
    keyDown: function(e){
        if(e.keyCode == 32 || e.keyCode == 40 || e.keyCode == 38) e.preventDefault();
        if(e.keyCode == 82) window.rotation = !window.rotation;
        if(this.menu) return this.menu.keyDown(e.keyCode);
        this.world.keyDown(e.keyCode);
    },
    keyUp: function(e){
        if(this.menu) return;
        this.world.keyUp(e.keyCode);
    },
    setResolution: function(r){
        this.can.width = P.w * r;
        this.can.height = P.h * r;

        this.resolution = r;
    }
};

function World(){
    wld = this;

    this.score = 0;

    this.particles = [];
    this.cars = [];
    this.buildings = [];
    this.clients = [];
    this.clientSpots = [];
    this.textures = [];
    this.trails = [];

    this.down = {};

    this.t = 0;

    var w = 8000,
        h = 8000,
        bt = 100;

    //this.addBuilding(new Building(-bt, 0, bt, h));
    //this.addBuilding(new Building(w, 0, bt, h));
    //this.addBuilding(new Building(0, -bt, w, bt));
    //this.addBuilding(new Building(0, h, w, bt));

    var bs = 300;
    var sw = 300;

    var cellSize = 900,
        swSize = 50,
        roadSize = 200,
        xwSize = 50;

    var building = function(x, y, w, h, b){};

    var park = function(x, y, w, h, b){
        var tex = new Texture(grass, x, y, w, h);
        wld.textures.push(tex)

        // I'm lazy so I'm adding an invisible building to get cycles and shit
        b.visible = false;
        b.collides = false;

        for(var i = 0 ; i < 10 ; i++){
            var t = new Tree(tex.x + ~~rd(0, tex.w), tex.y + ~~rd(0, tex.h));
            wld.buildings.push(t);
        }
    };

    var lot = function(x, y, w, h, b){
        var tex = new Texture(parking, x, y, w, h);
        wld.textures.push(tex);

        // I'm lazy so I'm adding an invisible building to get cycles and shit
        b.visible = false;
        b.collides = false;

        // Exits
        wld.textures.push(new Texture(road, tex.x - swSize, tex.y + swSize * 2, swSize, swSize * 2));
        wld.textures.push(new Texture(road, tex.x + tex.w, tex.y + swSize * 2, swSize, swSize * 2));
        wld.textures.push(new Texture(road, tex.x - swSize, tex.y + tex.h - swSize * 4, swSize, swSize * 2));
        wld.textures.push(new Texture(road, tex.x + tex.w, tex.y + tex.h - swSize * 4, swSize, swSize * 2));

        // Random cars
        var positions = [];
        for(var x = tex.x + swSize / 2 ; x < tex.x + tex.w - swSize / 2 ; x += swSize){
            for(var y = tex.y + swSize ; y < tex.y + tex.h ; y += parking.height){
                positions.push({ x: x, y: y });
                positions.push({ x: x, y: y + swSize * 4 });
            }
        }

        for(var i = 0 ; i < 10 ; i++){
            var ind = ~~rd(positions.length);
            var pos = positions[ind];
            positions.splice(ind, 1);

            var s = parking.width / 2;

            var c = new Enemy();
            c.x = pos.x;
            c.y = pos.y;
            c.rotation = rp([M.PI / 2, -M.PI / 2])
            wld.addCar(c);
        }
    };

    var cell = function(x, y, w, h){
        // First, adding a sidewalk
        var tex = new Texture(
            sidewalk,
            x - swSize,
            y - swSize,
            w + 2 * swSize,
            h + 2 * swSize
        );
        wld.textures.push(tex);

        var b = new Building(x, y, w, h);
        wld.buildings.push(b);

        // Then, random type of area
        var type = rp([park, park, building, building, building, lot]);
        type(x, y, w, h, b);
    };

    var cols = 10,
        rows = 10;

    for(var row = 0 ; row <= rows ; row++){
        for(var col = 0 ; col <= cols ; col++){
            // Crosswalks
            wld.textures.push(new Texture(
                xwalkv,
                col * cellSize - roadSize / 2 - xwSize,
                row * cellSize - roadSize / 2,
                xwSize,
                roadSize
            ));
            wld.textures.push(new Texture(
                xwalkv,
                col * cellSize + roadSize / 2,
                row * cellSize - roadSize / 2,
                xwSize,
                roadSize
            ));
            wld.textures.push(new Texture(
                xwalkh,
                col * cellSize - roadSize / 2,
                row * cellSize - roadSize / 2 - xwSize,
                roadSize,
                xwSize
            ));
            wld.textures.push(new Texture(
                xwalkh,
                col * cellSize - roadSize / 2,
                row * cellSize + roadSize / 2,
                roadSize,
                xwSize
            ));

            // Road lines
            wld.textures.push(new Texture(
                hline,
                col * cellSize + roadSize / 2 + xwSize,
                row * cellSize - hline.height / 2,
                cellSize - roadSize - xwSize * 2,
                hline.height
            ));
            wld.textures.push(new Texture(
                vline,
                col * cellSize - vline.width / 2,
                row * cellSize + roadSize / 2 + xwSize,
                vline.width,
                cellSize - roadSize - xwSize * 2
            ));
        }
    }

    var double = false;
    for(var row = 0 ; row < rows ; row++){
        for(var col = 0 ; col < cols ; col++){
            double = !double && col < cols -1 && Math.random() < .5;

            var x = col * cellSize + swSize + roadSize / 2;
            var y = row * cellSize + swSize + roadSize / 2;
            var w = cellSize - 2 * swSize - roadSize;
            var h = cellSize - 2 * swSize - roadSize;

            if(double){
                w = w * 2 + roadSize + 2 * swSize;

                col++;
            }

            cell(x, y, w, h);
        }
    }

    // water
    var sizes = [
        [-roadSize / 2 - swSize, -roadSize / 2 - swSize, cols * cellSize + 4000, -2000],
        [-roadSize / 2 - swSize, rows * cellSize + roadSize / 2 + swSize, cols * cellSize + 4000, 2000],
        [-roadSize / 2 - swSize, -roadSize / 2 - 2000, -2000, rows * cellSize + 6000],
        [cols * cellSize + roadSize / 2 + swSize, -roadSize / 2 - 2000, 2000, rows * cellSize + 6000]
    ];
    sizes.forEach(function(s){
        var b = new Building(s[0], s[1], s[2], s[3]);
        b.visible = false;
        wld.buildings.push(b);
        wld.textures.push(new Texture(water, s[0], s[1], s[2], s[3]));
    });

    // Surround with sidewalks
    this.textures.push(new Texture(
        sidewalk,
        -roadSize / 2 - swSize,
        -roadSize / 2 - swSize,
        cols * cellSize + 2 * swSize + roadSize,
        swSize
    ));
    this.textures.push(new Texture(
        sidewalk,
        -roadSize / 2 - swSize,
        rows * cellSize + roadSize / 2,
        cols * cellSize + 2 * swSize + roadSize,
        swSize
    ));
    this.textures.push(new Texture(
        sidewalk,
        -roadSize / 2 - swSize,
        -roadSize / 2 - swSize,
        swSize,
        rows * cellSize + 2 * swSize + roadSize
    ));
    this.textures.push(new Texture(
        sidewalk,
        cols * cellSize + roadSize / 2,
        -roadSize / 2 - swSize,
        swSize,
        rows * cellSize + 2 * swSize + roadSize
    ));

    this.player = this.addCar(new Player());
    this.player.x = cols / 2 * cellSize,
    this.player.y = rows / 2 * cellSize;

    this.camX = this.player.x - P.w / 2;
    this.camY = this.player.y - P.h / 2;
    this.camRotation = 0;

    for(var i = 0 ; i < this.buildings.length - 4 ; i++){
        var b = this.buildings[i];

        if(b.visible && b.collides || !b.visible && !b.collides){
            var cycle = b.getCycle();
            if(cycle){
                var enemy = this.addCar(new Enemy());
                enemy.x = cycle.x;
                enemy.y = cycle.y;
                enemy.follow(cycle);
            }

            this.clientSpots = this.clientSpots.concat(b.getCorners(25));
        }
    }

    this.nextClientSpawn = 0;
    this.timeleft = 180;
};

World.prototype = {
    cycle: function(e){
        this.t += e;

        this.nextClientSpawn -= e;
        if(this.nextClientSpawn <= 0){
            this.respawnClients();
            this.nextClientSpawn = 5;
        }

        // Background
        fs(road);
        fr(0, 0, P.w, P.h);

        for(var i in this.cars){
            this.cars[i].cycle(e);
        }

        // TODO handle camera
        //var angle = !this.player.dead ? this.player.moveAngle : this.player.moveAngle + M.PI;

        //var idealX = this.player.x - P.w / 2 + M.cos(angle) * this.player.speed * .4;
        //var idealY = this.player.y - P.h / 2 + M.sin(angle) * this.player.speed * .4;

        //idealX = wld.player.x - P.w / 2 + M.cos(angle) * this.player.speed * .4;
        //idealY = wld.player.y - P.h / 2 + M.sin(angle) * this.player.speed * .4;

        var camSpeed = !this.player.dead ? 600 : 100;

        //var distance = dist(idealX, idealY, this.camX, this.camY);
        //var appliedDistance = limit(distance, -camSpeed * e, camSpeed * e);

        //var angle = Math.atan2(idealY - this.camY, idealX - this.camX);
        //this.camX += Math.cos(angle) * appliedDistance;
        //this.camY += Math.sin(angle) * appliedDistance;

        this.camX = wld.player.x - P.w / 2 + M.cos(wld.player.moveAngle) * 100;
        this.camY = wld.player.y - P.h / 2 + M.sin(wld.player.moveAngle) * 100;

        if(this.shakeTime > 0){
            this.camX += rd(-10, 10);
            this.camY += rd(-10, 10);
        }

        //this.camX = idealX;
        //this.camY = idealX;

        var idealRotation = -this.player.rotation - M.PI / 2;
        var diff = idealRotation - this.camRotation;
        diff = normalizeAngle(diff);

        var rotationSpeed = M.max(abs(diff) / M.PI, .01) * M.PI * 2;

        diff = limit(diff, -rotationSpeed * e,rotationSpeed * e);

        this.camRotation += diff;

        this.shakeTime -= e;

        sv();

        if(window.rotation){
            tr(P.w / 2, P.h / 2);
            rotate(this.camRotation);
            tr(-P.w / 2, -P.h / 2);
        }

        tr(-~~this.camX, -~~this.camY);

        fs(road);
        fr(~~this.camX, ~~this.camY, P.w, P.h);

        var sw = 50;

        for(var i in this.textures){
            this.textures[i].render();
        }

        /*for(var i in this.trails){
            this.trails[i].render();
        }*/

        for(var i in this.clients){
            this.clients[i].cycle(e);
        }

        for(var i = this.cars.length - 1 ; i >= 0 ; i--){
            this.cars[i].render();
        }

        for(var i = this.particles.length - 1 ; i >= 0 ; i--){
            this.particles[i].render();
        }

        for(var i in this.buildings){
            this.buildings[i].render();
        }

        this.player.render2();

        rs();

        if(!G.menu){
            this.player.hud.cycle(e);

            if(!this.player.client){
                this.timeleft -= e;
                if(this.timeleft <= 0){
                    G.gameOver();
                }
            }
        }

        if(this.player.x < -this.roadSize / 2){
            this.player.explode();
        }
    },
    keyUp: function(k){
        this.down[k] = 0;
        this.evalKeyboardMovement();
    },
    keyDown: function(k){
        this.down[k] = true;
        this.evalKeyboardMovement();
    },
    evalKeyboardMovement: function(){
        this.player.rotationDir = 0;
        this.player.accelerates = false;
        this.player.brakes = false;
        if(this.down[37]){
            this.player.rotationDir = -1;
        }
        if(this.down[39]){
            this.player.rotationDir = 1;
        }
        if(this.down[38]){
            this.player.accelerates = true;
        }
        if(this.down[40]){
            this.player.brakes = true;
        }
        if(this.down[32]){
            G.start();
        }
    },
    addParticle: function(p){
        this.particles.push(p);
    },
    removeParticle: function(p){
        var ind = this.particles.indexOf(p);
        if(ind >= 0) this.particles.splice(ind, 1);
    },
    /*addTrail: function(p){
        this.trails.push(p);
    },
    removeTrail: function(p){
        var ind = this.trails.indexOf(p);
        if(ind >= 0) this.trails.splice(ind, 1);
    },*/
    addBuilding: function(b){
        this.buildings.push(b);
        return b;
    },
    addCar: function(c){
        this.cars.push(c);
        return c;
    },
    removeCar: function(c){
        var i = this.cars.indexOf(c);
        if(i >= 0){
            this.cars.splice(i, 1);
        }
    },
    addClient: function(c){
        this.clients.push(c);
        return c;
    },
    removeClient: function(c){
        var i = this.clients.indexOf(c);
        if(i >= 0) this.clients.splice(i, 1);
    },
    getRandomDestination: function(){
        return rp(this.clientSpots);
    },
    respawnClients: function(){
        var minD = M.max(P.w, P.h);
        var maxD = M.max(P.w, P.h) * 2;

        for(var i = this.clients.length - 1 ; i >= 0 ; i--){
            var client = this.clients[i];
            var d = dist(client.x, client.y, this.camX + P.w / 2, this.camY + P.h / 2);
            if(d > maxD){
                this.clients.splice(i, 1);
            }
        }

        var potential = [];
        for(var i = 0 ; i < this.clientSpots.length ; i++){
            var spot = this.clientSpots[i];
            var d = dist(spot.x, spot.y, this.camX + P.w / 2, this.camY + P.h / 2);
            if(d < maxD && d > minD){
                potential.push(spot);
            }
        }

        var target = 10;
        while(potential.length > 0 && this.clients.length < target){
            var ind = ~~rd(potential.length);
            var spot = potential[ind];
            potential.splice(ind, 1);

            var client = this.addClient(new Client());
            client.x = spot.x;
            client.y = spot.y;
        }
    },
    findClosestClientSpot: function(x, y){
        var spot,
            minDist,
            d,
            closest;
        for(var i = 0 ; i < this.clientSpots.length ; i++){
            spot = this.clientSpots[i];
            d = dist(spot.x, spot.y, x, y);
            if(!closest || d < minDist){
                closest = spot;
                minDist = d;
            }
        }
        return closest;
    },
    shake: function(){
        this.shakeTime = .5;
    }
};

function Menu(){
    this.alpha = 0;
    Easing.tween(this, 'alpha', 0, 1, .5);
}

Menu.prototype = {
    cycle: function(e){
        alpha(this.alpha);
        fs('rgba(0,0,0,.7)');
        fr(0, 0, P.w, P.h);
    }
};

function Home(g){
    Menu.call(this);
}

Home.prototype = xt(Menu.prototype, {
    cycle: function(e){
        Menu.prototype.cycle.call(this, e);

        var t = 'taxi drift',
            w = textWidth(t);
        drawText(c, t, 'white', (P.w - w) / 2, 80, 1, 1);

        t = 'find customers and drive them';
        w = textWidth(t, .5);
        drawText(c, t, 'white', (P.w - w) / 2, 200, .5, 1);

        t = 'to their destination';
        w = textWidth(t, .5);
        drawText(c, t, 'white', (P.w - w) / 2, 250, .5, 1);

        t = 'press enter to start';
        w = textWidth(t, .5);
        drawText(c, t, 'white', (P.w - w) / 2, 350, .5, 1);

        t = 'press r to toggle rotation';
        w = textWidth(t, .5);
        drawText(c, t, 'white', (P.w - w) / 2, 400, .5, 1);

        t = 'press space to restart game';
        w = textWidth(t, .5);
        drawText(c, t, 'white', (P.w - w) / 2, 450, .5, 1);

        alpha(1);
    },
    keyDown: function(k){
        if(k === 13)
        G.menu = null;
    }
});

function End(s){
    Menu.call(this);
}

End.prototype = xt(Menu.prototype, {
    cycle: function(e){
        Menu.prototype.cycle.call(this, e);

        var t = 'game over',
            w = textWidth(t);
        drawText(c, t, 'white', (P.w - w) / 2, 80, 1, 1);

        var t = 'you served ' + wld.player.dropoffs + ' customers',
            w = textWidth(t, .5);
        drawText(c, t, 'white', (P.w - w) / 2, 200, .5, 1);

        var t = 'and collected $' + wld.player.cash,
            w = textWidth(t, .5);
        drawText(c, t, 'white', (P.w - w) / 2, 250, .5, 1);

        var t = 'press enter to try again',
            w = textWidth(t, .5);
        drawText(c, t, 'white', (P.w - w) / 2, 480, .5, 1);

        alpha(1);
    },
    keyDown: function(k){
        if(k === 13)
        G.restart();
    }
});

function Car(){
    this.l = 50;
    this.w = 30;
    this.x = 0;
    this.y = 0;
    this.rotation = 0;
    this.speed = 0;
    this.rotationSpeed = M.PI;
    this.rotationDir = 0;
    this.vectors = [];
    this.accelerates = false;
    this.brakes = false;
    this.maxSpeed = 500;
    this.drifts = true;

    this.t = 0;

    this.maxAcceleration = 400;
    this.maxDeceleration = 100000;
    this.maxDeceleration = 400;

    this.speedVector = {};

    this.moveAngle = 0;
    this.moveAngleSpeed = M.PI * 1.5;

    this.radius = 10;

    this.carType = rp([
        car.white,
        car.blue,
        car.red,
        car.green,
        car.purple,
        car.gray
    ]);
}

Car.prototype = {
    cycle: function(e){
        this.t += e;

        if(this.dead){
            return;
        }

        var oppositeAngle = this.rotation + Math.PI;

        var speedRatio = limit(1 - this.speed / this.maxSpeed, .5, 1);

        var angleDiff = normalizeAngle(this.rotation - this.moveAngle);
        var appliedDiff = limit(angleDiff, -this.moveAngleSpeed * speedRatio * e, this.moveAngleSpeed * speedRatio * e);
        this.moveAngle += this.drifts ? appliedDiff : angleDiff;

        var r = limit(this.speed * 3 / this.maxSpeed, -1, 1);

        this.rotation += this.rotationSpeed * e * this.rotationDir * r;

        this.x += this.speed * M.cos(this.moveAngle) * e;
        this.y += this.speed * M.sin(this.moveAngle) * e;

        var targetSpeed = 0,
            opposite = false;
        if(this.accelerates){
            targetSpeed = this.maxSpeed;
            if(this.speed < 0) opposite = true;
        }else if(this.brakes){
            targetSpeed = -this.maxSpeed / 2;
            if(this.speed > 0) opposite = true;
        }

        var diff = targetSpeed - this.speed;
        var acc = opposite ? this.maxAcceleration * 2 : this.maxAcceleration;
        diff = limit(diff, -e * acc, e * acc);

        this.speed += diff;
        //return;

        //
        //var acceleration = this.accelerates ? this.maxAcceleration : -this.maxDeceleration;
        //this.speed = limit(this.speed + acceleration * e, 0, this.maxSpeed);

        // Turning "opposition"
        var opposition = abs(normalizeAngle(this.rotation - this.moveAngle)) / M.PI;
        this.speed = limit(this.speed - opposition * e * this.maxDeceleration * 2, -this.maxSpeed, this.maxSpeed);
    },
    render: function(){
        sv();
        tr(this.x, this.y);
        rt(this.rotation);

        var img = this.dead ? brokenCar : this.carType;
        di(img, -img.width / 2, -img.height / 2);

        rs();
    },
    explode: function(){
        this.dead = true;

        for(var i = 0 ; i < 40 ; i++){
            var c = rp(['#ff0', '#f00', '#ff8400', '#000'])
            var p = new Particle(5, c);
            p.x = this.x + rd(-5, 5);
            p.y = this.y + rd(-5, 5);
            wld.addParticle(p);

            var a = rd(M.PI * 2),
                d = rd(20, 100),
                t = rd(.5, 1);

            Easing.tween(p, 'x', p.x, p.x + M.cos(a) * d, t);
            Easing.tween(p, 'y', p.y, p.y + M.sin(a) * d, t);
            Easing.tween(p, 'a', 1, 0, t);
            Easing.tween(p, 's', p.s, p.s * rd(5, 10), t, 0, linear, function(){
                wld.removeParticle(p);
            });
        }
    },
    collidesWith: function(c){
        return dist(c.x, c.y, this.x, this.y) < this.radius + c.radius;
    }
};

function Player(){
    Car.call(this);
    this.carType = car.yellow;
    this.client = null;
    this.hud = new HUD();
    this.lastGoodPosition = null;
    this.nextGoodPosition = null;
    this.nextGoodPositionTimer = 0;
    this.cash = 0;
    this.lives = 3;
    this.dropoffs = 0;
}

Player.prototype = xt(Car.prototype, {
    cycle: function(e){
        var tmpX = this.x,
            tmpY = this.y,
            tmpAngle = this.rotation;

        this.noControlTimer -= e;
        if(this.noControlTimer >= 0){
            this.accelerates = false;
            this.rotationDir = 0;
        }

        Car.prototype.cycle.call(this, e);

        if(this.dead) return;

        this.nextGoodPositionTimer -= e;
        if(this.nextGoodPositionTimer <= 0){
            this.lastGoodPosition = this.nextGoodPosition;
            this.nextGoodPosition = { x: this.x, y: this.y };
            this.nextGoodPositionTimer = .1;
        }

        if(this.accelerates && this.speed < 400 || this.brakes && this.speed > -200
            || abs(this.speed) > 20 && abs(normalizeAngle(this.rotation - this.moveAngle)) > Math.PI / 8){

            var posOnLine = -this.l / 2;

            var p = new Particle(5, '#fff');
            p.x = this.x + M.cos(this.rotation) * posOnLine + rd(-5, 5);
            p.y = this.y + M.sin(this.rotation) * posOnLine + rd(-5, 5);
            wld.addParticle(p);

            var d = 100,
                t = rd(.3, .6),
                a = this.rotation + M.PI + rd(-M.PI / 32, M.PI / 32);

            //Easing.tween(p, 'x', p.x, p.x + M.cos(a) * d, t);
            //Easing.tween(p, 'y', p.y, p.y + M.sin(a) * d, t);
            Easing.tween(p, 'a', 1, 0, t);
            Easing.tween(p, 's', p.s, p.s * rd(5, 10), t, 0, linear, function(){
                wld.removeParticle(this);
            });

            /*var split = 10;
            var t1 = new Trail(
                this.x + M.cos(this.rotation + M.PI / 2) * split,
                this.y + M.sin(this.rotation + M.PI / 2) * split,
                tmpX + M.cos(tmpAngle + M.PI / 2) * split,
                tmpY + M.sin(tmpAngle + M.PI / 2) * split
            );
            wld.addTrail(t1);
            var t2 = new Trail(
                this.x + M.cos(this.rotation - M.PI / 2) * split,
                this.y + M.sin(this.rotation - M.PI / 2) * split,
                tmpX + M.cos(tmpAngle - M.PI / 2) * split,
                tmpY + M.sin(tmpAngle - M.PI / 2) * split
            );
            wld.addTrail(t2);

            Easing.tween(t1, 'a', 1, 0, 1, 2, linear, function(){
                wld.removeTrail(t1);
            });
            Easing.tween(t2, 'a', 1, 0, 1, 2, linear, function(){
                wld.removeTrail(t2);
            });*/
        }

        // Collisions
        var me = this;
        wld.buildings.forEach(function(b){
            if(!me.dead && b.collides && b.contains(me.x, me.y)){
                me.explode();
            }
        });

        wld.cars.forEach(function(c){
            if(!me.dead && c !== me && !c.dead && c.collidesWith(me)){
                me.collided(c);
            }
        });

        // Client
        if(this.client){
            this.clientTimeLeft = M.max(0, this.clientTimeLeft - e);

            var d = dist(this.x, this.y, this.clientSettings.destination.x, this.clientSettings.destination.y);
            if(d < this.clientSettings.radius){
                if(this.speed === 0){
                    this.drop();
                }
            }else{
                if(this.clientTimeLeft == 0 && this.speed < 100){
                    this.drop();
                }
            }
        }
    },
    render2: function(){
        if(this.clientSettings && !this.dead){
            var r = (this.t % 1) * this.clientSettings.radius;

            alpha(.3);
            c.fillStyle = '#0f0';
            c.lineWidth = 4;
            c.strokeStyle = '#0f0';
            c.beginPath();
            c.arc(this.clientSettings.destination.x, this.clientSettings.destination.y, r, 0, 2 * M.PI, true);
            c.fill();
            c.stroke();
            alpha(1);

            var d = dist(this.x, this.y, this.clientSettings.destination.x, this.clientSettings.destination.y);
            var angle = M.atan2(this.clientSettings.destination.y - this.y, this.clientSettings.destination.x - this.x);
            var arrowDist = limit(d / 10000, 0, 1) * 200 + 100;
            if(d > 300){
                sv();
                tr(this.x + M.cos(angle) * arrowDist, this.y + M.sin(angle) * arrowDist);
                c.rotate(angle);
                di(arrow, -arrow.width / 2, -arrow.height / 2);
                rs();
            }
        }
    },
    pickup: function(c){
        if(!this.client){
            this.client = c;
            this.clientSettings = c.getDestinationSettings();
            this.clientTimeLeft = this.clientSettings.time;
            wld.removeClient(c);
        }
    },
    drop: function(){
        // Drop the client on the side
        this.client.done = true;
        this.client.x = this.x + M.cos(this.rotation + M.PI / 2) * 40;
        this.client.y = this.y + M.sin(this.rotation + M.PI / 2) * 40;
        this.client.findSidewalk();
        wld.addClient(this.client);

        var d = dist(this.x, this.y, this.clientSettings.destination.x, this.clientSettings.destination.y);

        var price = d <= this.clientSettings.radius ? this.clientSettings.price : 0;
        if(price > 0){
            this.hud.message('reward: $' + price);
            this.cash += price;
        }else{
            this.hud.message('too slow');
        }

        this.client = null;
        this.clientSettings = null;

        this.dropoffs++;
    },
    collided: function(c){
        this.explode();
        c.explode();
        wld.removeCar(c);

        window.collider = c;
    },
    explode: function(){
        Car.prototype.explode.call(this);

        this.lives--;
        if(this.lives > 0){
            setTimeout(this.respawn.bind(this), 2000);
        }else{
            // game over
            setTimeout(G.gameOver.bind(G), 2000);
        }

        wld.shake();
    },
    respawn: function(){
        this.hud.message('cars left: ' + this.lives);
        this.client = null;
        this.clientTimeLeft = 0;
        this.clientSettings = null;
        this.dead = false;
        this.x = this.lastGoodPosition.x;
        this.y = this.lastGoodPosition.y;
        this.speed = 0;
    }
});

function Enemy(){
    Car.call(this);

    this.path = null;
    this.maxSpeed = 100;
    this.distanceLeft = 0;
    this.drifts = false;
}

Enemy.prototype = xt(Car.prototype, {
    cycle: function(e){
        this.accelerates = !!this.path;

        var a = M.atan2(wld.player.y - this.y, wld.player.x - this.x);
        var d = dist(wld.player.x, wld.player.y, this.x, this.y);
        if(abs(normalizeAngle(a - this.rotation)) < M.PI / 4 && d < 300){
            this.accelerates = false;
        }

        if(this.path){
            var targetAngle = M.atan2(this.path.y - this.y, this.path.x - this.x);
            var diff = normalizeAngle(targetAngle - this.rotation);
            diff = limit(diff, -this.rotationSpeed * e, this.rotationSpeed * e);

            this.rotation += diff;
            //this.moveAngle = targetAngle;

            if(dist(this.x, this.y, this.path.x, this.path.y) < e * this.speed){
                this.x = this.path.x;
                this.y = this.path.y;

                this.follow(this.path.next);
            }
        }

        Car.prototype.cycle.call(this, e);
    },
    follow: function(p){
        this.path = p;
    }
});

function Client(){
    this.x = this.y = 0;
    this.done = false;
    this.type = rp([clientRed, clientBlue, clientBlack, clientYellow]);
};

Client.prototype = {
    cycle: function(e){
        var d = dist(this.x, this.y, wld.player.x, wld.player.y);

        if(d < 200
            && !wld.player.dead
            && wld.player.speed < 100
            && !this.done
            && !wld.player.client){

            this.target = null;

            if(d < 30 && wld.player.speed < 50){
                wld.player.pickup(this);
            }else{
                // Approach
                var diff = Math.min(50 * e, d);

                this.x += M.cos(this.angle) * diff;
                this.y += M.sin(this.angle) * diff;
            }
        }else if(!this.target){
            this.findSidewalk();
        }

        if(this.target){
            var d = dist(this.x, this.y, this.target.x, this.target.y);
            var diff = Math.min(50 * e, d);

            if(diff > 0){
                this.angle = M.atan2(this.target.y - this.y, this.target.x - this.x);

                this.x += M.cos(this.angle) * diff;
                this.y += M.sin(this.angle) * diff;
            }
        }else{
            this.angle = M.atan2(wld.player.y - this.y, wld.player.x - this.x);
        }

        var me = this;
        wld.cars.forEach(function(c){
            var d = dist(me.x, me.y, c.x, c.y);
            if(d < 20 && abs(c.speed) > 20){
                me.die();
            }
        });

        this.render();
    },
    render: function(){
        sv();
        tr(this.x, this.y);
        rt(this.angle);
        di(this.type, -this.type.width / 2, -this.type.height / 2);
        rs();
    },
    getDestinationSettings: function(){
        var dest = wld.getRandomDestination();
        var dist = abs(this.x - dest.x) + abs(this.y - dest.y);

        var exigence = ~~rd(1, 4); // 1-3

        var pricePerPx = 1 / 100;
        var perfectTime = dist / wld.player.maxSpeed; // very best possible time
        var reasonableIdealTime = perfectTime * 4;

        var price = ~~((exigence / 3) * pricePerPx * dist);
        var time = (1 - exigence / 4) * reasonableIdealTime;

        return {
            destination: dest,
            exigence: exigence,
            time: time,
            price: M.max(5, price),
            radius: 200
        };
    },
    die: function(){
        wld.removeClient(this);

        for(var i = 0 ; i < 40 ; i++){
            var p = new Particle(4, '#950000', 1);
            var a = rd(M.PI * 2);
            var d = rd(5, 25);
            var t = rd(.05, .2);

            p.x = this.x;
            p.y = this.y;

            wld.addParticle(p);

            Easing.tween(p, 'a', 1, 0, 1, 3, linear, p.remove.bind(p));
            Easing.tween(p, 'x', p.x, p.x + M.cos(a) * d, t);
            Easing.tween(p, 'y', p.y, p.y + M.sin(a) * d, t);
        }

        wld.player.hud.message('don\'t kill customers!')
    },
    findSidewalk: function(){
        var t = wld.findClosestClientSpot(this.x, this.y);

        this.target = {
            x: t.x + rd(-15, 15),
            y: t.y + rd(-15, 15)
        };
    }
};

function HUD(){
    this.msgT = 0;
};

HUD.prototype = {
    cycle: function(e){
        this.msgT -= e;

        var m;
        if(wld.player.dead){
            m = 'wasted';
        }else if(this.msgT > 0){
            m = this.msg;
        }else if(wld.player.client){
            var tl = M.ceil(M.max(0, wld.player.clientTimeLeft));
            m = 'customer time left: ' + tl;
        }else{
            m = 'find a customer'
        }

        var w = textWidth(m, .5);
        var x = (P.w - w) / 2,
            y = P.h / 2 + 200;
        //drawText(c, m, 'black', x, y + 5, .5);
        drawText(c, m, 'white', x, y, .5, 1);

        m = 'cash: $' + wld.player.cash;
        w = textWidth(m, .5)
        drawText(c, m, 'white', P.w - w - 20, 20, .5, 1);


        drawText(c, 'cars: ' + wld.player.lives, 'white', 20, 20, .5, 1);

        if(!wld.player.client){
            m = 'time: ' + ~~(wld.timeleft);
            w = textWidth(m, .5)
            drawText(c, m, 'white', (P.w - w) / 2, 20, .5, 1);
        }
    },
    message: function(m){
        this.msgT = 2;
        this.msg = m.toLowerCase();
    }
}

function Building(x, y, w, h){
    this.x = x;
    this.y = y;
    this.w = w;
    this.h = h;

    if(this.w < 0){
        this.x += this.w;
        this.w = -this.w;
    }
        if(this.h < 0){
            this.y += this.h;
            this.h = -this.h;
        }

    this.visible = true;
    this.collides = true;
};

Building.prototype = {
    render: function(){
        if(!this.visible) return;
        if(
            this.x > wld.camX + P.w + P.v
            || this.y > wld.camY + P.h + P.v
            || this.x + this.w < wld.camX - P.v
            || this.y + this.h < wld.camY - P.v){
            return;
        }

        var topLeft1 = { x: this.x, y: this.y };
        var topRight1 = { x: this.x + this.w, y: this.y };
        var bottomRight1 = { x: this.x + this.w, y: this.y + this.h };
        var bottomLeft1 = { x: this.x, y: this.y + this.h };

        var topLeft2 = this.pointUpperPosition(this.x, this.y);
        var topRight2 = this.pointUpperPosition(this.x + this.w, this.y);
        var bottomRight2 = this.pointUpperPosition(this.x + this.w, this.y + this.h);
        var bottomLeft2 = this.pointUpperPosition(this.x, this.y + this.h);

        var windowWidth = 25;

        // Bottom
        //shape([topLeft1, topRight1, bottomRight1, bottomLeft1], 'green');

        // Top
        //shape([topLeft2, topRight2, bottomRight2, bottomLeft2], '#6f6f6f');
        //shape([topLeft2, topRight2, bottomRight2, bottomLeft2], roof);

        var x = topLeft2.x,
            y = topLeft2.y,
            w = bottomRight2.x - topLeft2.x,
            h = bottomRight2.y - topLeft2.y,
            r = 20;
        sv();
        tr(x, y);

        // Border
        fs(roofb);
        fr(0, 0, w, h);

        // Main roof
        fs(roof);
        fr(r, r, w - 2 * r, h - 2 * r);

        rs();

        // Sides
        fs('#00f');

        // Left side
        if(topLeft2.x > topLeft1.x){
            //shape([topLeft1, topLeft2, bottomLeft2, bottomLeft1], '#7e7e7e');
            shape([topLeft1, topLeft2, bottomLeft2, bottomLeft1], side1);

            var windows = this.h / (2 * windowWidth),
                stepZ = .3,
                stepY = this.h / windows;

            for(var z = stepZ / 2 ; z < 1 ; z += stepZ){
                for(var y = this.y + stepY / 2 ; y < this.y + this.h ; y += stepY){
                    var lower1 = this.pointUpperPosition(this.x, y - windowWidth / 2, z);
                    var upper1 = this.pointUpperPosition(this.x, y - windowWidth / 2, z + stepZ / 2);
                    var lower2 = this.pointUpperPosition(this.x, y + windowWidth / 2, z);
                    var upper2 = this.pointUpperPosition(this.x, y + windowWidth / 2, z + stepZ / 2);

                    shape([lower1, upper1, upper2, lower2], 'black');
                }
            }
        }

        // Right side
        if(topRight2.x < topRight1.x){
            //shape([topRight2, topRight1, bottomRight1, bottomRight2], '#7e7e7e');
            shape([topRight2, topRight1, bottomRight1, bottomRight2], side1);

            var windows = this.h / (2 * windowWidth),
                stepZ = .3,
                stepY = this.h / windows;

            for(var z = stepZ / 2 ; z < 1 ; z += stepZ){
                for(var y = this.y + stepY / 2 ; y < this.y + this.h ; y += stepY){
                    var lower1 = this.pointUpperPosition(this.x + this.w, y - windowWidth / 2, z);
                    var upper1 = this.pointUpperPosition(this.x + this.w, y - windowWidth / 2, z + stepZ / 2);
                    var lower2 = this.pointUpperPosition(this.x + this.w, y + windowWidth / 2, z);
                    var upper2 = this.pointUpperPosition(this.x + this.w, y + windowWidth / 2, z + stepZ / 2);

                    shape([lower1, upper1, upper2, lower2], 'black');
                }
            }
        }

        // Top side
        if(topLeft2.y > topLeft1.y){
            //shape([topLeft1, topLeft2, topRight2, topRight1], '#999999');
            shape([topLeft1, topLeft2, topRight2, topRight1], side2);

            var windows = this.w / (2 * windowWidth),
                stepZ = .3,
                stepX = this.w / windows;

            for(var z = stepZ / 2 ; z < 1 ; z += stepZ){
                for(var x = this.x + stepX / 2 ; x < this.x + this.w ; x += stepX){
                    var lower1 = this.pointUpperPosition(x - windowWidth / 2, this.y, z);
                    var upper1 = this.pointUpperPosition(x - windowWidth / 2, this.y, z + stepZ / 2);
                    var lower2 = this.pointUpperPosition(x + windowWidth / 2, this.y, z);
                    var upper2 = this.pointUpperPosition(x + windowWidth / 2, this.y, z + stepZ / 2);

                    shape([lower1, upper1, upper2, lower2], 'black');
                }
            }
        }

        // Bottom side
        if(bottomLeft2.y < bottomLeft1.y){
            //shape([bottomLeft1, bottomLeft2, bottomRight2, bottomRight1], '#999999');
            shape([bottomLeft1, bottomLeft2, bottomRight2, bottomRight1], side2);

            var windows = this.w / (2 * windowWidth),
                stepZ = .3,
                stepX = this.w / windows;

            for(var z = stepZ / 2 ; z < 1 ; z += stepZ){
                for(var x = this.x + stepX / 2 ; x < this.x + this.w ; x += stepX){
                    var lower1 = this.pointUpperPosition(x - windowWidth / 2, this.y + this.h, z);
                    var upper1 = this.pointUpperPosition(x - windowWidth / 2, this.y + this.h, z + stepZ / 2);
                    var lower2 = this.pointUpperPosition(x + windowWidth / 2, this.y + this.h, z);
                    var upper2 = this.pointUpperPosition(x + windowWidth / 2, this.y + this.h, z + stepZ / 2);

                    shape([lower1, upper1, upper2, lower2], 'black');
                }
            }
        }
    },
    pointUpperPosition: function(x, y, prct){
        if(isNaN(prct)) prct = 1;

        var fromCenterX = x - (wld.camX + P.w / 2);
        var fromCenterY = y - (wld.camY + P.h / 2);

        return {
            x: x + (fromCenterX / P.h) * 200 * prct,
            y: y + (fromCenterY / P.h) * 200 * prct
        };
    },
    contains: function(x, y){
        return x >= this.x - 10
            && y >= this.y - 10
            && x <= this.x + this.w + 10
            && y <= this.y + this.h + 10;
    },
    getCycle: function(){
        var cycle = createCycle(this.getCorners(100));

        var it = rd(8);
        for(var i = 0 ; i < it ; i++){
            cycle = cycle.next;
        }

        return cycle;
    },
    getCorners: function(r){
        return [
            { x: this.x - r, y: this.y - r },
            { x: this.x + this.w + r, y: this.y - r },
            { x: this.x + this.w + r, y: this.y + this.h + r },
            { x: this.x - r, y: this.y + this.h + r }
        ];
    }
};

function Texture(t, x, y, w, h){
    this.t = t;
    this.x = x;
    this.y = y;
    this.w = w;
    this.h = h;

    if(this.w < 0){
        this.x += this.w;
        this.w = -this.w;
    }
    if(this.h < 0){
        this.y += this.h;
        this.h = -this.h;
    }
}

Texture.prototype = {
    render: function(){
        if(
            this.x > wld.camX + P.w + P.v
            || this.y > wld.camY + P.h + P.v
            || this.x + this.w < wld.camX - P.v
            || this.y + this.h < wld.camY - P.v){
            return;
        }

        sv();
        tr(this.x, this.y);
        fs(this.t);
        fr(0, 0, this.w, this.h);
        rs();
    }
};

var defs = {
    a: [
        [1,1,1],
        [1, ,1],
        [1,1,1],
        [1, ,1],
        [1, ,1]
    ],
    b: [
        [1,1,1],
        [1, ,1],
        [1,1, ],
        [1, ,1],
        [1,1,1]
    ],
    c: [
        [1,1,1],
        [1, , ],
        [1, , ],
        [1, , ],
        [1,1,1]
    ],
    d: [
        [1,1,0],
        [1, ,1],
        [1, ,1],
        [1, ,1],
        [1,1,1]
    ],
    e: [
        [1,1,1],
        [1, , ],
        [1,1, ],
        [1, , ],
        [1,1,1]
    ],
    f: [
        [1,1,1],
        [1, , ],
        [1,1, ],
        [1, , ],
        [1, , ]
    ],
    g: [
        [1,1,1],
        [1, , ],
        [1, , ],
        [1, ,1],
        [1,1,1]
    ],
    h: [
        [1, ,1],
        [1, ,1],
        [1,1,1],
        [1, ,1],
        [1, ,1]
    ],
    i: [
        [1,1,1],
        [ ,1, ],
        [ ,1, ],
        [ ,1, ],
        [1,1,1]
    ],
    j: [
        [ , ,1],
        [ , ,1],
        [ , ,1],
        [1, ,1],
        [1,1,1]
    ],
    k: [
        [1, ,1],
        [1, ,1],
        [1,1, ],
        [1, ,1],
        [1, ,1]
    ],
    l: [
        [1, ,0],
        [1, , ],
        [1, , ],
        [1, , ],
        [1,1,1]
    ],
    m: [
        [1, ,1],
        [1,1,1],
        [1, ,1],
        [1, ,1],
        [1, ,1]
    ],
    n: [
        [1,1,1],
        [1, ,1],
        [1, ,1],
        [1, ,1],
        [1, ,1]
    ],
    o: [
        [1,1,1],
        [1, ,1],
        [1, ,1],
        [1, ,1],
        [1,1,1]
    ],
    p: [
        [1,1,1],
        [1, ,1],
        [1,1,1],
        [1, , ],
        [1, , ]
    ],
    r: [
        [1,1,1],
        [1, ,1],
        [1,1, ],
        [1, ,1],
        [1, ,1]
    ],
    s: [
        [1,1,1],
        [1, , ],
        [1,1,1],
        [ , ,1],
        [1,1,1]
    ],
    '$': [
        [ , ,1, ,0],
        [1,1,1,1,1],
        [1, ,1, , ],
        [1,1,1,1,1],
        [ , ,1, ,1],
        [1,1,1,1,1],
        [ , ,1, , ]
    ],
    t: [
        [1,1,1],
        [ ,1, ],
        [ ,1, ],
        [ ,1, ],
        [ ,1, ]
    ],
    u: [
        [1, ,1],
        [1, ,1],
        [1, ,1],
        [1, ,1],
        [1,1,1]
    ],
    v: [
        [1, ,1],
        [1, ,1],
        [1, ,1],
        [1, ,1],
        [ ,1, ]
    ],
    w: [
        [1, , , ,1],
        [1, , , ,1],
        [1, ,1, ,1],
        [1, ,1, ,1],
        [ ,1, ,1, ]
    ],
    x: [
        [1, ,1],
        [1, ,1],
        [ ,1, ],
        [1, ,1],
        [1, ,1]
    ],
    y: [
        [1, ,1],
        [1, ,1],
        [1,1,1],
        [ ,1, ],
        [ ,1, ]
    ],
    '\'': [
        [1]
    ],
    '.': [
        [0],
        [0],
        [0],
        [0],
        [1]
    ],
    ' ': [
        [ ,0],
        [ , ],
        [ , ],
        [ , ],
        [ , ]
    ],
    '-': [
        [ ,0],
        [ , ],
        [1,1],
        [ , ],
        [ , ]
    ],
    ':': [
        [0],
        [1],
        [ ],
        [1],
        [ ]
    ],
    '?': [
        [1,1,1],
        [ , ,1],
        [ ,1,1],
        [ , , ],
        [ ,1, ]
    ],
    '!': [
        [ ,1, ],
        [ ,1, ],
        [ ,1, ],
        [ , , ],
        [ ,1, ]
    ],
    '1': [
        [1,1,0],
        [ ,1, ],
        [ ,1, ],
        [ ,1, ],
        [1,1,1]
    ],
    '2': [
        [1,1,1],
        [ , ,1],
        [1,1,1],
        [1, , ],
        [1,1,1]
    ],
    '3': [
        [1,1,1],
        [ , ,1],
        [ ,1,1],
        [ , ,1],
        [1,1,1]
    ],
    '4': [
        [1, ,0],
        [1, , ],
        [1, ,1],
        [1,1,1],
        [ , ,1]
    ],
    '5': [
        [1,1,1],
        [1, , ],
        [1,1, ],
        [ , ,1],
        [1,1, ]
    ],
    '6': [
        [1,1,1],
        [1, , ],
        [1,1,1],
        [1, ,1],
        [1,1,1]
    ],
    '7': [
        [1,1,1],
        [ , ,1],
        [ ,1, ],
        [ ,1, ],
        [ ,1, ]
    ],
    '8': [
        [1,1,1],
        [1, ,1],
        [1,1,1],
        [1, ,1],
        [1,1,1]
    ],
    '9': [
        [1,1,1],
        [1, ,1],
        [1,1,1],
        [ , ,1],
        [1,1,1]
    ],
    '0': [
        [1,1,1],
        [1, ,1],
        [1, ,1],
        [1, ,1],
        [1,1,1]
    ]
};

var Font = {};

var createFont = function(color){
    Font[color] = {};

    for(var i in defs){
        var d = defs[i];
        Font[color][i] = cache(d[0].length * 10 + 10, d.length * 10, function(c, r){
            r.fs(color);

            for(var i = 0 ; i < d.length ; i++){
                for(var j = 0 ; j < d[i].length ; j++){
                    if(d[i][j]){
                        r.fr(j * 10, i * 10, 10, 10);
                    }
                }
            }
        });
    }
};

createFont('white');
createFont('black');
createFont('orange');

var drawText = function(r, t, c, x, y, s, b){
    s = s || 1;

    // Shadow
    if(b) drawText(r, t, 'black', x, y + 5, s, false);

    r.sv();
    r.tr(x, y);
    r.sc(s, s);

    x = 0;
    for(var i = 0 ; i < t.length ; i++){
        var ch = t.charAt(i),
            img = Font[c][ch];
        if(img){
            r.di(img, x, 0);
            x += img.width;
        }
    }
    r.rs();
};

var textWidth = function(t, s){
    var w = 0, i = t.length;
    while(i--){
        var img = Font['white'][t.charAt(i)];
        w += img ? img.width : 0;
    }
    return w * (s || 1);
};

var

s = 4,

car = {
    white: newCar('#fff'),
    broken: newCar('#1b1b1b', true),
    yellow: newCar('#ff0'),
    blue: newCar('#00f'),
    red: newCar('#f00'),
    green: newCar('#0f0'),
    purple: newCar('#f0f'),
    gray: newCar('#6c6c6c'),
},

client = function(color){
    return cache(20, 30, function(c, r){
        r.fs(color);
        //r.beginPath();
        //r.arc(c.width / 2, c.height / 2, 9, 0, M.PI * 2, true);
        //r.fill();

        var w = 14,
            h = 18;
        r.fr((c.width - w)/ 2, (c.height - h) / 2, w, h);

        r.bp();
        r.arc(c.width / 2, c.height / 2 - 10, 4, 0, M.PI * 2, true);
        r.arc(c.width / 2, c.height / 2 + 10, 4, 0, M.PI * 2, true);
        r.fill();

        r.fs('#e99a79');
        r.bp();
        r.arc(c.width / 2, c.height / 2, 6, 0, M.PI * 2, true);
        r.fill();

        r.fs('#000');
        r.fr(c.width / 2 + 2, c.height / 2 - 3, 2, 2);
        r.fr(c.width / 2 + 2, c.height / 2 + 3, 2, -2);
    });
},

clientRed = client('#900'),
clientBlack = client('#000'),
clientBlue = client('#00f'),
clientYellow = client('#880'),

arrow = cache(40, 40, function(c, r){
    with(r){
        var b = r;
        tr(c.width / 2, c.height / 2);
        rotate(Math.PI / 2);
        tr(-c.width / 2, -c.height / 2);
        tr(0, c.height);
        sc(1, -1);
        fs('#fff');
        bp();
        mt(20, 40);
        lt(40, 20);
        lt(30, 20);
        lt(30, 0);
        lt(10, 0);
        lt(10, 20);
        lt(0, 20);
        fill();
    }
}),

brokenCar = cache(50, 22, function(c, r){
    with(r){
        var b = r;
        fs('#1b1b1b');
        fr(0, 0, 50, 22);

        fs('#000');
        fr(10, 3, 5, 16);
        fr(30, 3, 10, 16);
        fr(15, 1, 7, 1);
        fr(23, 1, 7, 1);
        fr(15, 20, 7, 1);
        fr(23, 20, 7, 1);
    }
}),

grass = cache(200, 200, function(c, r){
    var s = 4;
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            r.fs('rgb(0,' + (128 + ~~rd(-50, 50)) + ', 0)');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),

sidewalk = cache(100, 100, function(c, r){
    var b = 8;
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var isBorder = x < b
                        || y < b
                        || x >= c.width - b
                        || y >= c.height - b
                        || x >= c.width / 2 - b && x <= c.width / 2 + b
                        || y >= c.height / 2 - b && y <= c.height / 2 + b;

            var v = (isBorder ? 80 : 100) + ~~rd(-10, 10);


            r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),

road = cache(200, 200, function(c, r){
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var v = 40 + ~~rd(-10, 10);
            r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),

water = cache(100, 100, function(c, r){
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            r.fs('rgb(0, ' + ~~(168 + ~~rd(-10, 10)) + ', 255)');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),

xwalkh = cache(25, 50, function(c, r){
    for(var x = ~~(c.width / 4) ; x < c.width * .75 ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var v = 255 - ~~rd(10, 50);
            r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),

xwalkv = cache(50, 25, function(c,r){
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = ~~(c.height / 4) ; y < c.height * .75 ; y += s){
            var v = 255 - ~~rd(10, 50);
            r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern');

roof = cache(100, 100, function(c, r){
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var v = 100 + ~~rd(-10, 10);
            r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),

roofb = cache(100, 100, function(c, r){
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var v = 50 + ~~rd(-10, 10);
            r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),

side1 = cache(100, 100, function(c, r){
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var v = 128 + ~~rd(-5, 5);
            r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),

side2 = cache(100, 100, function(c, r){
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var v = 153 + ~~rd(-5, 5);
            r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),

hline = cache(100, 4, function(c, r){
    for(var x = c.width * .25 ; x < c.width * .75 ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var v = 255 - ~~rd(10, 50);
            r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),

vline = cache(4, 100, function(c, r){
    for(var y = c.height * .25 ; y < c.height * .75 ; y += s){
        for(var x = 0 ; x < c.height ; x += s){
            var v = 255 - ~~rd(10, 50);
            r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
            r.fr(x, y, s, s);
        }
    }
}, 'pattern'),


tree = cache(200, 200, function(c, r){
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var d = dist(c.width / 2, c.height / 2, x, y);
            var f = d < c.width * .4 && Math.random() < .8;

            if(f){
                var v = 50 + ~~rd(-25, 25);
                r.fs('rgb(0, ' + v + ', 0)');
                r.fr(x, y, s, s);
            }
        }
    }
}),

tree2 = cache(150, 150, function(c, r){
    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var d = dist(c.width / 2, c.height / 2, x, y);
            var f = d < c.width * .4 && Math.random() < .8;

            if(f){
                var v = 50 + ~~rd(-25, 25);
                r.fs('rgb(0, ' + v + ', 0)');
                r.fr(x, y, s, s);
            }
        }
    }
}),

parking = cache(100, 300, function(c, r){
    r.fs(road);
    r.fillRect(0, 0, c.width, c.height);

    for(var x = 0 ; x < c.width ; x += s){
        for(var y = 0 ; y < c.height ; y += s){
            var draw = y < s
                    || y >= c.height - s
                    || (x < s || x >= c.width - s || x >= c.width / 2 - s && x <= c.width / 2 + s) && (y < c.height * .3 || y >= c.height * .7);

            if(draw){
                var v = 255 - ~~rd(10, 50);
                r.fs('rgb(' + v + ', ' + v + ', ' + v + ')');
                r.fr(x, y, s, s);
            }
        }
    }
}, 'pattern')

;

addEventListener('load',function(){
    G = new Game();
});

var tweens = [];

function linear(t, b, c, d){
    return (t / d) * c + b;
};

function easeOutBack(t, b, c, d, s) {
    if (s == undefined) s = 1.70158;
    return c*((t=t/d-1)*t*((s+1)*t + s) + 1) + b;
};

function easeOutBounce(t, b, c, d) {
    if ((t/=d) < (1/2.75)) {
        return c*(7.5625*t*t) + b;
    } else if (t < (2/2.75)) {
        return c*(7.5625*(t-=(1.5/2.75))*t + .75) + b;
    } else if (t < (2.5/2.75)) {
        return c*(7.5625*(t-=(2.25/2.75))*t + .9375) + b;
    } else {
        return c*(7.5625*(t-=(2.625/2.75))*t + .984375) + b;
    }
};

var Easing = {
    tween: function(o, p, a, b, d, l, f, e){
        tweens.push({
            o: o, // object
            p: p, // property
            a: a, // from
            b: b, // to
            d: d, // duration
            l: l || 0,
            f: f || linear, // easing function
            e: e || noop, // end callback
            t: 0
        });
    },
    cycle: function(e){
        var tw;
        for(var i = tweens.length - 1 ; i >= 0 ; i--){
            tw = tweens[i];
            if(tw.l > 0){
                tw.l -= e;
                tw.o[tw.p] = tw.a;
            }else{
                tw.t = M.min(tw.d, tw.t + e);
                tw.o[tw.p] = tw.f(tw.t, tw.a, tw.b - tw.a, tw.d);
                if(tw.t == tw.d){
                    tw.e.call(tw.o);
                    tweens.splice(i, 1);
                }
            }
        }
    }
};


function Particle(s, c, a){
    this.s = s;
    this.c = c;
    this.a = a;
};

Particle.prototype = {
    render: function(e){
        alpha(this.a);
        fs(this.c);
        fr(this.x - this.s / 2, this.y - this.s / 2, this.s, this.s);
        alpha(1);
    },
    remove: function(){
        wld.removeParticle(this);
    }
};

function Tree(x, y){
    this.x = x;
    this.y = y;
    this.collides = true;
};

Tree.prototype = xt(Building.prototype, {
    render: function(){
        if(this.x > wld.camX + P.w + 200
            || this.y > wld.camY + P.h + 200
            || this.x < wld.camX - 200
            || this.y < wld.camY - 200){
            return;
        }

        var p1 = this.pointUpperPosition(this.x, this.y, .2);
        var p2 = this.pointUpperPosition(this.x, this.y, .4);

        c.strokeStyle = '#5a3900';
        c.lineWidth = 15;
        bp();
        mt(this.x, this.y);
        lt(p2.x, p2.y);
        stroke();

        di(tree, p1.x - tree.width / 2, p1.y - tree.height / 2);
        di(tree2, p2.x - tree2.width / 2, p2.y - tree2.height / 2);
    },
    contains: function(x, y){
        return abs(x - this.x) < 15
            && abs(y - this.y) < 15;
    },
    getCycle: function(){
        return null;
    },
    getCorners: function(r){
        return [];
    }
});



              
            
!
999px

Console