Friday, December 27, 2013

Fascinating - HTML5 CANVAS 3D CUBES

This webpage was built entirely using only HTML5, basically CSS3 and JavaScript. With this project, I wanted to show using a real and interesting example, how we can combine the power of CSS3 3D transforms, with some of the new features and API's that HTML5 provides.


DEMO 

HTML & JavaScript Code:
<!DOCTYPE HTML>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Fascinating - HTML5 CANVAS 3D CUBES demo</title>
<style>
html {
overflow: hidden;
}
body {
position: absolute;
margin: 0px;
padding: 0px;
background: #000;
width: 100%;
height: 100%;
}
#screen {
position: absolute;
width: 100%;
height: 100%;
background: #000;
overflow: hidden;
font-family: Segoe UI, Verdana, Arial, Sans-Serif;
color: #fff;
font-size: 13px;
}
#screen canvas {
position: absolute;
width: 100%;
height: 100%;
background: #000;
}
#info {
position: absolute;
text-align: left;
top: 19%;
left: 2%;
width: 180px;
height: 340px;
color: #666;
font-size: 1em;
}
#info .background {
position: absolute;
width: 100%;
height: 100%;
background: #000;
opacity: 0.3;
}
#info .content {
position: absolute;
padding: 3px;
width: 100%;
height: 100%;
}
#info .w {
color: #fff;
}
#info hr {
width: 90%;
border: none;
background-color: #666;
height: 1px;
}
#info h1 {
color: #fff;
text-align: center;
}
.button {
font-family: Arial, Helvetica, sans-serif;
font-size: 10px;
color: #ffffff;
padding: 10px 20px;
background: -moz-linear-gradient(  top,  #333333 0%,  #000000);
background: -webkit-gradient(  linear, left top, left bottom,  from(#333333),  to(#000000));
-moz-border-radius: 1px;
-webkit-border-radius: 1px;
border-radius: 1px;
border: 1px solid #000000;
-moz-box-shadow: 0px 1px 1px rgba(000,000,000,0.5),  inset 0px 0px 1px rgba(255,255,255,1);
-webkit-box-shadow: 0px 1px 1px rgba(000,000,000,0.5),  inset 0px 0px 1px rgba(255,255,255,1);
box-shadow: 0px 1px 1px rgba(000,000,000,0.5),  inset 0px 0px 1px rgba(255,255,255,1);
text-shadow: 0px -1px 0px rgba(000,000,000,0.7),  0px 1px 0px rgba(255,255,255,0.3);
}
</style>
</head>
<body>
 <h1 align="center" style="color:#fff;">Fascinating - HTML5 CANVAS 3D CUBES demo</h1>
<div id="screen">
  <canvas id="canvas">HTML5 CANVAS</canvas>
  <div id="info">
    <div class="background"></div>
    <div class="content">
      <h1>3D Cubes</h1>
      <table>
        <tr>
          <td class="w">drag</td>
          <td>→ rotate X,Y axis</td>
        </tr>
        <tr>
          <td class="w">wheel</td>
          <td>→ rotate Z axis</td>
        </tr>
        <tr>
          <td class="w">click</td>
          <td>→ create cube</td>
        </tr>
      </table>
      <hr>
      <input type="checkbox" id="white">
      <label for="white"> white background</label>
      <br>
      <input type="checkbox" id="alpha">
      <label for="alpha"> transparency</label>
      <br>
      <input type="checkbox" id="autor">
      <label for="autor"> auto rotation</label>
      <br>
      <input type="checkbox" id="destroy">
      <label for="destroy"> destroy cubes</label>
      <br>
      <hr>
      - <span id="fps" class="w">00</span> FPS<br>
      - <span id="npoly" class="w">00</span> Faces<br>
      <p align="center">
        <input type="button" value="RESET" id="reset" class="button">
        </input>
        <input type="button" value="STOP" id="stopgo" class="button">
        </input>
      </p>
    </div>
  </div>
</div>
<script>
// =============================================================
//           ===== CANVAS 3D experiment =====
//     ===== simple 3D cubes HTML5 engine ====
// http://phpdevlovers.blogspot.in
// =============================================================

"use strict";

(function () {
// ======== private vars ========
var scr, canvas, cubes, faces, nx, ny, nw, nh, xm = 0, ym = 0, cx = 50, cy = 50, cz = 0, cxb = 0, cyb = 0;
var white, alpha, fps = 0, ncube, npoly, faceOver, drag, moved, startX = 0, startY = 0;
var cosY, sinY, cosX, sinX, cosZ, sinZ, minZ, angleY = 0, angleX = 0, angleZ = 0;
var bkgColor1 = "rgba(0,0,0,0.1)";
var bkgColor2 = "rgba(32,32,32,1)";
var autorotate = false, destroy = false, running = true;
// ---- fov ----
var fl = 250;
var zoom = 0;
// ======== canvas constructor ========
var Canvas = function (id) {
this.container = document.getElementById(id);
this.ctx = this.container.getContext("2d");
this.resize = function (w, h) {
this.container.width = w;
this.container.height = h;
}
};
// ======== vertex constructor ========
var Point = function (parent, xyz, project) {
this.project = project;
this.xo = xyz[0];
this.yo = xyz[1];
this.zo = xyz[2];
this.cube = parent;
};
Point.prototype.projection = function () {
// ---- 3D rotation ----
var x = cosY * (sinZ * this.yo + cosZ * this.xo) - sinY * this.zo;
var y = sinX * (cosY * this.zo + sinY * (sinZ * this.yo + cosZ * this.xo)) + cosX * (cosZ * this.yo - sinZ * this.xo);
var z = cosX * (cosY * this.zo + sinY * (sinZ * this.yo + cosZ * this.xo)) - sinX * (cosZ * this.yo - sinZ * this.xo);
this.x = x;
this.y = y;
this.z = z;
if (this.project) {
// ---- point visible ----
if (z < minZ) minZ = z;
this.visible = (zoom + z > 0);
// ---- 3D to 2D projection ----
this.X = (nw * 0.5) + x * (fl / (z + zoom));
this.Y = (nh * 0.5) + y * (fl / (z + zoom));
}
};
// ======= polygon constructor ========
var Face = function (cube, index, normalVector) {
// ---- parent cube ----
this.cube = cube;
// ---- coordinates ----
this.p0 = cube.points[index[0]];
this.p1 = cube.points[index[1]];
this.p2 = cube.points[index[2]];
this.p3 = cube.points[index[3]];
// ---- normal vector ----
this.normal = new Point(this, normalVector, false)
// ---- # faces ----
npoly++;
document.getElementById('npoly').innerHTML = npoly;
};
Face.prototype.pointerInside = function () {
// ---- Is Point Inside Triangle? ----
var fAB = function (p1, p2, p3) { return (ym - p1.Y) * (p2.X - p1.X) - (xm - p1.X) * (p2.Y - p1.Y); };
var fCA = function (p1, p2, p3) { return (ym - p3.Y) * (p1.X - p3.X) - (xm - p3.X) * (p1.Y - p3.Y); };
var fBC = function (p1, p2, p3) { return (ym - p2.Y) * (p3.X - p2.X) - (xm - p2.X) * (p3.Y - p2.Y); };
if (
fAB(this.p0, this.p1, this.p3) * fBC(this.p0, this.p1, this.p3) > 0 &&
fBC(this.p0, this.p1, this.p3) * fCA(this.p0, this.p1, this.p3) > 0
) return true;
if (
fAB(this.p1, this.p2, this.p3) * fBC(this.p1, this.p2, this.p3) > 0 &&
fBC(this.p1, this.p2, this.p3) * fCA(this.p1, this.p2, this.p3) > 0
) return true;
// ----
return false;
};
Face.prototype.faceVisible = function () {
// ---- points visible ----
if (this.p0.visible && this.p1.visible && this.p2.visible && this.p3.visible) {
// ---- back face culling ----
if ((this.p1.Y - this.p0.Y) / (this.p1.X - this.p0.X) < (this.p2.Y - this.p0.Y) / (this.p2.X - this.p0.X) ^ this.p0.X < this.p1.X == this.p0.X > this.p2.X) {
// ---- face visible ----
this.visible = true;
return true;
}
}
// ---- face hidden ----
this.visible = false;
this.distance = -99999;
return false;
};
Face.prototype.distanceToCamera = function () {
// ---- distance to camera ----
var dx = (this.p0.x + this.p1.x + this.p2.x + this.p3.x ) * 0.25;
var dy = (this.p0.y + this.p1.y + this.p2.y + this.p3.y ) * 0.25;
var dz = (zoom + fl) + (this.p0.z + this.p1.z + this.p2.z + this.p3.z ) * 0.25;
this.distance = Math.sqrt(dx * dx + dy * dy + dz * dz);
};
Face.prototype.draw = function () {
// ---- shape face ----
canvas.ctx.beginPath();
canvas.ctx.moveTo(this.p0.X, this.p0.Y);
canvas.ctx.lineTo(this.p1.X, this.p1.Y);
canvas.ctx.lineTo(this.p2.X, this.p2.Y);
canvas.ctx.lineTo(this.p3.X, this.p3.Y);
canvas.ctx.closePath();
// ---- light ----
if (this == faceOver) {
var r = 256;
var g = 0;
var b = 0;
} else {
// ---- flat (lambert) shading ----
this.normal.projection();
var light = (
white ?
this.normal.y + this.normal.z * 0.5 :
this.normal.z
) * 256;
var r = g = b = light;
}
// ---- fill ----
canvas.ctx.fillStyle = "rgba(" +
Math.round(r) + "," +
Math.round(g) + "," +
Math.round(b) + "," + this.cube.alpha + ")";
canvas.ctx.fill();
};
// ======== Cube constructor ========
var Cube = function(parent, nx, ny, nz, x, y, z, w) {
if (parent) {
// ---- translate parent points ----
this.w = parent.w;
this.points = [];
var i = 0, p;
while (p = parent.points[i++]) {
this.points.push(
new Point(
parent,
[p.xo + nx, p.yo + ny, p.zo + nz],
true
)
);
}
} else {
// ---- create points ----
this.w = w;
this.points = [];
var p = [
[x-w, y-w, z-w],
[x+w, y-w, z-w],
[x+w, y+w, z-w],
[x-w, y+w, z-w],
[x-w, y-w, z+w],
[x+w, y-w, z+w],
[x+w, y+w, z+w],
[x-w, y+w, z+w]
];
for (var i in p) this.points.push(
new Point(this, p[i], true)
);
}
// ---- faces coordinates ----
var f  = [
[0,1,2,3],
[0,4,5,1],
[3,2,6,7],
[0,3,7,4],
[1,5,6,2],
[5,4,7,6]
];
// ---- faces normals ----
var nv = [
[0,0,1],
[0,1,0],
[0,-1,0],
[1,0,0],
[-1,0,0],
[0,0,-1]
];
// ---- cube transparency ----
this.alpha = alpha ? 0.5 : 1;
// ---- push faces ----
for (var i in f) {
faces.push(
new Face(this, f[i], nv[i])
);
}
ncube++;
};
////////////////////////////////////////////////////////////////////////////
var resize = function () {
// ---- screen resize ----
nw = scr.offsetWidth;
nh = scr.offsetHeight;
var o = scr;
for (nx = 0, ny = 0; o != null; o = o.offsetParent) {
nx += o.offsetLeft;
ny += o.offsetTop;
}
canvas.resize(nw, nh);
};
var reset = function () {
// ---- create first cube ----
cubes = [];
faces = [];
ncube = 0;
npoly = 0;
cubes.push(
new Cube(false,0,0,0,0,0,0,50)
);
};
var detectFaceOver = function () {
// ---- detect pointer over face ----
var j = 0, f;
faceOver = false;
while ( f = faces[j++] ) {
if (f.visible) {
if ( f.pointerInside() ) {
faceOver = f;
}
} else break;
}
};
var click = function () {
// ---- click cube ----
detectFaceOver();
if (faceOver) {
if (destroy) {
if (ncube > 1) {
var c = faceOver.cube;
faceOver.clicked = false;
// ---- destroy faces ----
var i = 0, f;
while ( f = faces[i++] ) {
if (f.cube == c) {
faces.splice(--i, 1);
npoly--;
}
}
document.getElementById('npoly').innerHTML = npoly;
// ---- destroy cube ----
var i = 0, o;
while ( o = cubes[i++] ) {
if (o == c) {
cubes.splice(--i, 1);
ncube--;
break;
}
}
}
} else {
if (!faceOver.clicked) {
// ---- create new cube ----
faceOver.clicked = true;
var w = -2.25 * faceOver.cube.w;
cubes.push(
new Cube(
faceOver.cube,

w * faceOver.normal.xo,
w * faceOver.normal.yo,
w * faceOver.normal.zo
)
);
detectFaceOver();
}
}
}
};
////////////////////////////////////////////////////////////////////////////
var init = function () {
// ---- init script ----
scr = document.getElementById("screen");
canvas  = new Canvas("canvas");
// ======== unified touch/mouse events handler ========
scr.ontouchstart = scr.onmousedown = function (e) {
if (!running) return true;
// ---- touchstart ----
if (e.target !== canvas.container) return;
e.preventDefault(); // prevents scrolling
if (scr.setCapture) scr.setCapture();
moved = false;
drag = true;
startX = (e.clientX !== undefined ? e.clientX : e.touches[0].clientX) - nx;
startY = (e.clientY !== undefined ? e.clientY : e.touches[0].clientY) - ny;
};
scr.ontouchmove = scr.onmousemove = function(e) {
if (!running) return true;
// ---- touchmove ----
e.preventDefault();
xm = (e.clientX !== undefined ? e.clientX : e.touches[0].clientX) - nx;
ym = (e.clientY !== undefined ? e.clientY : e.touches[0].clientY) - ny;
detectFaceOver();
if (drag) {
cx = cxb + (xm - startX);
cy = cyb - (ym - startY);
}
if (Math.abs(xm - startX) > 10 || Math.abs(ym - startY) > 10) {
// ---- if pointer moves then cancel the tap/click ----
moved = true;
}
};
scr.ontouchend = scr.onmouseup = function(e) {
if (!running) return true;
// ---- touchend ----
e.preventDefault();
if (scr.releaseCapture) scr.releaseCapture();
drag = false;
cxb = cx;
cyb = cy;
if (!moved) {
// ---- click/tap ----
xm = startX;
ym = startY;
click();
}
};
scr.ontouchcancel = function(e) {
if (!running) return true;
// ---- reset ----
if (scr.releaseCapture) scr.releaseCapture();
moved = false;
drag = false;
cxb = cx;
cyb = cy;
startX = 0;
startY = 0;
};
// ---- Z axis rotation (mouse wheel) ----
scr.addEventListener('DOMMouseScroll', function(e) {
if (!running) return true;
cz += e.detail * 12;
return false;
}, false);
scr.onmousewheel = function () {
if (!running) return true;
cz += event.wheelDelta / 5;
return false;
}
// ---- multi-touch gestures ----
document.addEventListener('gesturechange', function(e) {
if (!running) return true;
e.preventDefault();
// ---- Z axis rotation ----
cz = event.rotation;
}, false);
// ---- screen size ----
resize();
window.addEventListener('resize', resize, false);
// ---- fps count ----
setInterval(function () {
document.getElementById('fps').innerHTML = fps * 2;
fps = 0;
}, 500); // update every 1/2 seconds
// ---- some UI options ----
document.getElementById("white").onchange = function () {
white = this.checked;
if (white) {
bkgColor1 = "rgba(256,256,256,0.1)";
bkgColor2 = "rgba(192,192,192,1)";
} else {
bkgColor1 = "rgba(0,0,0,0.1)";
bkgColor2 = "rgba(32,32,32,1)";
}
}
document.getElementById("alpha").onchange = function () {
alpha = this.checked;
}
document.getElementById("autor").onchange = function () {
autorotate = this.checked;
}
document.getElementById("destroy").onchange = function () {
destroy = this.checked;
}
document.getElementById("stopgo").onclick = function () {
running = !running;
document.getElementById("stopgo").value = running ? "STOP" : "GO!";
if (running) run();
}
document.getElementById("reset").onclick = function () {
reset();
}
// ---- engine start ----
reset();
run();
}
////////////////////////////////////////////////////////////////////////////
// ======== main loop ========
var run = function () {
// ---- screen background ----
canvas.ctx.fillStyle = bkgColor1;
canvas.ctx.fillRect(0, Math.floor(nh * 0.15), nw, Math.ceil(nh * 0.7));
canvas.ctx.fillStyle = bkgColor2;
canvas.ctx.fillRect(0, 0, nw, Math.ceil(nh * 0.15));
canvas.ctx.fillStyle = bkgColor2;
canvas.ctx.fillRect(0, Math.floor(nh * 0.85), nw, Math.ceil(nh * 0.15));
// ---- easing rotations ----
angleX += ((cy - angleX) * 0.05);
angleY += ((cx - angleY) * 0.05);
angleZ += ((cz - angleZ) * 0.05);
if (autorotate) cz += 1;
// ---- pre-calculating trigo ----
cosY = Math.cos(angleY * 0.01);
sinY = Math.sin(angleY * 0.01);
cosX = Math.cos(angleX * 0.01);
sinX = Math.sin(angleX * 0.01);
cosZ = Math.cos(angleZ * 0.01);
sinZ = Math.sin(angleZ * 0.01);
// ---- points projection ----
minZ = 0;
var i = 0, c;
while ( c = cubes[i++] ) {
var j = 0, p;
while ( p = c.points[j++] ) {
p.projection();
}
}
// ---- adapt zoom ----
var d = -minZ + 100 - zoom;
zoom += (d * ((d > 0) ? 0.05 : 0.01));
// ---- faces light ----
var j = 0, f;
while ( f = faces[j++] ) {
if ( f.faceVisible() ) {
f.distanceToCamera();
}
}
// ---- faces depth sorting ----
faces.sort(function (p0, p1) {
return p1.distance - p0.distance;
});
// ---- painting faces ----
j = 0;
while ( f = faces[j++] ) {
if (f.visible) {
f.draw();
} else break;
}
// ---- animation loop ----
fps++;
if (running) setTimeout(run, 16);
}
return {
////////////////////////////////////////////////////////////////////////////
// ---- onload event ----
load : function () {
window.addEventListener('load', function () {
init();
}, false);
}
}
})().load();
</script>
</body>
</html>



3D Canvas Cube - HTML5 & JavaScript

I wanted to see if it was possible to make a cube that worked pretty much anywhere(mobile/desktop) without a plugin... The answer is, YES.
If you are impatient, then just go checkout the DEMO
I've optimized the canvas to look crisp on high density screens and mobile devices. The canvas uses devicePixelRatio to attempt to use hardware pixels instead of computed pixels(i.e. the image quality of the canvas is stupid good on the iPhone4 and high density Androids).



HTML & JavaScript Code:
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<style>
body {
background-color: #ffffff;
margin: 0;
overflow: hidden;
}
</style>
</head>
<body>
<h1 align="center">Canvas Cube 3D</h1>
<script src="http://cdnjs.cloudflare.com/ajax/libs/three.js/r57/three.min.js"></script>
<script>

var camera, scene, renderer;
var geometry, material, mesh;

var init = function () {

renderer = new THREE.CanvasRenderer();
renderer.setSize( window.innerWidth, window.innerHeight );
document.body.appendChild( renderer.domElement );

camera = new THREE.PerspectiveCamera( 75, window.innerWidth / window.innerHeight, 1, 1000 );
camera.position.z = 500;

scene = new THREE.Scene();

geometry = new THREE.CubeGeometry( 200, 200, 200 );
material = new THREE.MeshBasicMaterial( { color: 0x000000, wireframe: true, wireframeLinewidth: 2 } );

mesh = new THREE.Mesh( geometry, material );
scene.add( mesh );

}

var animate = function () {

requestAnimationFrame( animate );

mesh.rotation.x = Date.now() * 0.0005;
mesh.rotation.y = Date.now() * 0.001;

renderer.render( scene, camera );

}

init();
animate();

</script>
<script src="//ajax.googleapis.com/ajax/libs/jquery/1.8.3/jquery.min.js" type="text/javascript"></script>
<script src="//ajax.googleapis.com/ajax/libs/mootools/1.4.5/mootools-yui-compressed.js" type="text/javascript"></script>
</body>
</html>



Animate a Fill Circle using Canvas & JavaScript


The HTML5 <canvas> element is used to draw graphics, on the fly, via scripting (usually JavaScript).
The <canvas> element is only a container for graphics. You must use a script to actually draw the graphics.
Canvas has several methods for drawing paths, boxes, circles, text, and adding images.
However, the <canvas> element has no drawing abilities of its own (it is only a container for graphics) - you must use a script to actually draw the graphics.
The getContext() method returns an object that provides methods and properties for drawing on the canvas.

HTML CODE:
The markup looks like this:
<div id="container">
  <h1>html5 Canvas - Animate a Fill Circle using Canvas - by <a href="http://cathodesoft.com/">CathodeSoft</a></h1>
  <canvas id="Circle" width="578" height="200"></canvas>
</div>


JavaScript Code :
<script>
// Script by http://phpdevlovers.blogspot.in
// requestAnimationFrame Sping  
var canvas = document.getElementById('Circle');
var context = canvas.getContext('2d');
var centerX = canvas.width / 2;
var centerY = canvas.height / 2;
var radius = 80;

var full = radius*2;
var amount = 0;
var amountToIncrease = 10;

function draw() {
    context.save();
    context.beginPath();
    context.arc(centerX, centerY, radius, 0, 2 * Math.PI, false);
    context.clip(); // Make a clipping region out of this path
    // instead of filling the arc, we fill a variable-sized rectangle
    // that is clipped to the arc
    context.fillStyle = '#13a8a4';
    // We want the rectangle to get progressively taller starting from the bottom
    // There are two ways to do this:
    // 1. Change the Y value and height every time
    // 2. Using a negative height
    // I'm lazy, so we're going with 2
    context.fillRect(centerX - radius, centerY + radius, radius * 2, -amount);
    context.restore(); // reset clipping region

    context.beginPath();
    context.arc(centerX, centerY, radius, 0, 2 * Math.PI, false);
    context.lineWidth = 10;
    context.strokeStyle = '#000000';
    context.stroke();
    
    // Every time, raise amount by some value:
    amount += amountToIncrease;
    if (amount > full) amount = 0; // restart
}

draw();
// Every second we'll fill more;
setInterval(draw, 1000);
</script>
<script src="//ajax.googleapis.com/ajax/libs/jquery/1.8.3/jquery.min.js" type="text/javascript"></script>
<script src="//ajax.googleapis.com/ajax/libs/mootools/1.4.5/mootools-yui-compressed.js" type="text/javascript"></script>

HTML5 Canvas Circle - Drawing circles with Animation & Number



HTML Code:
<div class="circleStatsItem"> <span class="num-value">20</span> <span class="percent">%</span>
  <canvas id="myCanvas1" width="78" height="78"></canvas>
</div>
<div class="circleStatsItem"> <span class="num-value">35</span> <span class="percent">%</span>
  <canvas id="myCanvas2" width="78" height="78"></canvas>
</div>
<div class="circleStatsItem"> <span class="num-value">55</span> <span class="percent">%</span>
  <canvas id="myCanvas3" width="78" height="78"></canvas>
</div>
<div class="circleStatsItem"> <span class="num-value">85</span> <span class="percent">%</span>
  <canvas id="myCanvas4" width="78" height="78"></canvas>
</div>

CSS Styles:
<style>
.circleStatsItem {
position: relative;
background: rgba(255,255,255,0.7);
-webkit-box-shadow: inset 0 0px 0 5px rgba(0,0,0,.2), 0 0px 0 3px rgba(0,0,0,.1);
-moz-box-shadow: inset 0 0px 0 5px rgba(0,0,0,.2), 0 0px 0 3px rgba(0,0,0,.1);
box-shadow: inset 0 0px 0 5px rgba(0,0,0,.2), 0 0px 0 3px rgba(0,0,0,.1);
-webkit-border-radius: 50em;
-moz-border-radius: 50em;
border-radius: 50em;
width: 78px;
height: 78px;
margin: 10px 10px 10px auto;
color: #6b6b6d;
float:left;
}
.circleStatsItem .num-value {
font-size: 18px;
font-weight: bold;
position: absolute;
top: 36%;
margin-left: 23px;
}
.circleStatsItem .percent {
font-size: 16px;
position: absolute;
top: 38%;
margin-left: 44px;
}
</style>

JAVASCRIPT code :
<script>
// requestAnimationFrame Sping
(function() {
  var requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame ||
                              window.webkitRequestAnimationFrame || window.msRequestAnimationFrame;
  window.requestAnimationFrame = requestAnimationFrame;
})();

 function animate(elementId, endPercent, color) {
     var canvas = document.getElementById(elementId);
     var context = canvas.getContext('2d');
     var x = canvas.width / 2;
     var y = canvas.height / 2;
     var radius = 35;
     var curPerc = 0;
     var counterClockwise = false;
     var circ = Math.PI * 2;
     var quart = Math.PI / 2;
    
     context.lineWidth = 6;
     context.strokeStyle = color;
     context.shadowOffsetX = 0;
     context.shadowOffsetY = 0;
    
     function render(current) {
         context.clearRect(0, 0, canvas.width, canvas.height);
         context.beginPath();
         context.arc(x, y, radius, -(quart), ((circ) * current) - quart, false);
         context.stroke();
         curPerc++;
         if (curPerc < endPercent) {
             requestAnimationFrame(function () {
                 render(curPerc / 100);
             });
         }
     }
     render();
 }

animate('myCanvas1', 20, '#fb7e12');
animate('myCanvas2', 35, '#54a7dd');
animate('myCanvas3', 55, '#806ce5');
animate('myCanvas4', 85, '#5fc53b');
</script>
<script src="//ajax.googleapis.com/ajax/libs/jquery/1.8.3/jquery.min.js" type="text/javascript"></script>
<script src="//ajax.googleapis.com/ajax/libs/mootools/1.4.5/mootools-yui-compressed.js" type="text/javascript"></script>

Monday, December 23, 2013

How to Setup a Dedicated Web Server for Free

All great websites have a great server behind them. In this tutorial, I’ll show you how to set up a dedicated web server (with Apache, MySQL, and PHP) using that old computer you have lying around the house and some free software.

1. A Quick Overview

In this tutorial, we are aiming to accomplish several things:
  • We’re going to install the Ubuntu Server operating system. I commonly use Ubuntu because of its ease of use and
    simple administration. It also has a rather large and extremely active community behind it, which makes getting
    support a breeze.
  • We’re going to install an OpenSSH server. This allows you to administer your server from remote computers.
  • A LAMP (Linux, Apache, MySQL, and PHP) stack is going to be installed. This provides the backbone that will run
    your web site. Apache is the industry standard web server on Unix-based operating systems; it’s what most web hosts
    use (NETTUTS is using it right now!) and it’s what we’re going to use.
  • We’re going to install a firewall to protect your server from unauthorized access.
In order to follow this tutorial, you’re going to need a few items:
  • A computer to use as your server. It doesn’t need to be powerful; as long as it’s not ancient, it’ll work fine. Please
    don’t do this on your desktop PC; Ubuntu will completely wipe your computer.
  • A CD burner and a blank CD. These are so that you can burn Ubuntu to a disk in order to install it.
  • Time. Seriously, this process is time-consuming, especially if you run into problems. Try to set aside an afternoon
    to follow this tutorial.
You may be asking why you’d want to have your own web server. There are several reasons, a few of them being: you can have your own testing
ground for your websites; with a little modification, you could host your own site; and, you will learn a lot about Linux/Unix as you go.
With that said, let’s get started!

2. Download Ubuntu Server

First and foremost, we’re going to need a CD with Ubuntu on it. Point your web browser tohttp://www.ubuntu.com/,
and click download from the menu to the left. You will now be presented with a box with two tabs: “Desktop Edition” and
“Server Edition”. Click the “Server Edition” tab, and select “Ubuntu 8.04 LTS”. Next, select a download location from the
drop-down box. Finally, hit the “Begin Download” button.
Now you need to burn the ISO (the file that you downloaded) to a blank CD. If you don’t know how to do this, there is an
excellent guide at https://help.ubuntu.com/community/BurningIsoHowto

3. Install Ubuntu Server

Now that you’ve downloaded and burned the ISO, let’s get Ubuntu installed on your server. Put the disk in the drive, and boot from the CD. In most modern computers, this will happen by default if a disk is in the drive when you turn it on. If it doesn’t, then you need to press a key on your keyboard right when you turn it on. For my laptop, it’s F12, and for my server, it’s F2. It just depends on your computer. You can find it by looking at the text on your screen right when you turn the computer on, during the BIOS. You’ll see something like “Press [KEY] to change boot order”. Press that key, and select your CD drive.
Still with me? Good. Now that you’ve booted up Ubuntu, you should see the following screen:
Select your language, and hit enter. Now you’ll see this screen:
Select “Install Ubuntu Server”, and away we go!
The installer will now ask you if you want it to detect your keyboard layout. Personally, I always choose no, because
it’s faster to select a standard american keyboard from the list than to have the installer detect it. Either option is fine,
just follow the on-screen instructions.
After you’ve done that, you’ll now see a bunch of loading screens saying things like “Detecting CD-ROM drives” and such.
These should pass quickly and without problems. However, during these screens, the installer will try to auto-configure your
network settings. For most cases, this will work without complaint. However, if it doesn’t work for you, just follow the
on-screen instructions to get it working.
After it’s done with all of that, it will ask you for a host name. You can usually set this to anything; I always set
mine to “web-server”.
The system will now want you to set the time zone for your clock. For me, it’s Pacific. Choose the one that applies to
you.
Now, the system will detect more hardware, and you’ll be prompted to “partion the disk(s)”. Select “Guided – use entire
disk”.
You will now need to select the disk you wish to partition. For most setups, only one disk will be available; however,
for more specialized systems, more options will be available here. Choose the one that applies to you.
It will ask you if you want to write the changes to the disk. Select “Yes” and hit enter. The installer will now proceed
to format the drive and set up the partitions.
Now the magic happens. The system will begin to install. While this happens, go get a cup of coffee. This can take anywhere
from 10 minutes to an hour. It just depends on your system. There might be times that it seems like it’s frozen; don’t worry,
it isn’t. Just let it do it’s thing. However, if it’s stuck on one thing for upwards of an hour, then yes, it is frozen.
Now that the system is installed, it needs to set up the account you are going to login with. First, give it your full
name and hit “Continue”.
Now give it your username. It will normally just set it as your first name,
but you can change it. One name you may not use is “root”.
You will now be asked to provide a password. It is ESSENTIAL that you choose a strong password, or your server will not
be secure at all. I recommend at LEAST a mixture of numbers, lowercase letters, and uppercase letters. However, for my servers
I use symbols, as well as a mixture of the above. DO NOT use a password shorter than 7 characters.
Then, re-enter your password to verify that you typed it correctly.
The system will now attempt to configure the “Package Manager” (we’ll get to what that is shortly). Provide it with your
proxy information, or leave it blank if you don’t use a proxy, and select “Continue”.
The system will now scan several servers looking for updates and configuration settings.
After that has completed, you will be presented with several options to install server software. Now, listen VERY carefully.
Select OpenSSH server, and press SPACE, NOT ENTER. If you hit enter, the install will proceed without installing the OpenSSH server.
You could install “LAMP server” as well, but I have no experience with this option, so we’re going to install it all with a different
command later on.
The system will now install your selected software, as well as other system components.
Finally, the install will finish. Remove the CD, and hit enter. The computer will reboot. If all goes well, you will be
presented with a screen that looks similar to the following:
Congratulations! You’ve just finished the hardest part. Ubuntu is now installed, and it is time to turn this computer into
a web server.

4. Update Your New Server

Before we go any further, we need to make sure your server is up-to-date. To do this, you need to login. First, type your username
(the one you chose earlier), press enter, and then type your password. As you’re typing your password, you’ll notice that nothing
seems to be happening. Don’t worry, that’s the way it was designed to work. After you’ve finished typing your password, hit enter,
and your screen should look similar to the one below if all went well:
Now, type:
sudo aptitude update && sudo aptitude dist-upgrade
It will ask you for you password, and again, you won’t see anything as you’re typing it. After you’ve done that, it will ask you if
you want to continue. Type “y” and press enter. Your screen will look similar to the following:
Your system will now download and install all the latest updates. This will take a while depending on your internet connection. After
it has finished, your computer will need to be rebooted. To do this, type:
sudo shutdown -r now
And let it reboot. Your server is now completely updated.

A Quick Note About “Sudo”

By now, you may have noticed that all of the commands you have typed have started with “sudo”. This is because they require
administrator privileges, and that’s what “sudo” does. It runs the command (i.e. “shutdown”) as an administrator, allowing it to work
properly. This is also why it asks you for your password. However, after you have typed “sudo” once and entered your password,
you do not have to enter your password again for five minutes. Not all commands require sudo, only ones that modify parts of the system.
Got all of that? Good.

5. Install Apache, MySQL, and PHP

It is now time to install some programs. In order to access your sites from the internet, we’re going to need to install a web server (Apache). In additon to the web server, we’ll
also want a database server (MySQL) and a server-side language (PHP) so that we can run popular applications such as WordPress. So,
let’s get to it!
Installing programs on Ubuntu is a lot different than installing programs on Windows or
OS X, in that Ubuntu will download and install the programs for you with a simple command. This is because Ubuntu has something called
Package Manager, which manages nearly all the programs on your system. All we have to do is tell the package manager
(called “aptitude”) that we want it to install Apache, MySQL, and PHP. To do this, type the following command:
sudo aptitude install apache2 php5-mysql libapache2-mod-php5 mysql-server
And press enter. Aptitude will download and install of the programs you specified. It will also download and install any
dependencies.
During the install process, MySQL will ask you for a root password. You can set this to anything, just be sure you make it long and secure.
Whatever you do, DO NOT leave this blank.
After that has all finished, you now have a fully working web server. To test it out, first find your server’s IP by typing:
ifconfig | grep inet
It’s usually the first IP returned. In my case, it’s 192.168.177.129. Now that you know the IP, open your web browser and point it
to your server IP. If you see the “It works!” message, then congratulations, it works.
However, we’re not done yet. We don’t want Apache or PHP to disclose any information about themselves, as this information is not needed
by your users and could pose a security risk. First, back up the original Apache configuration file:
sudo cp /etc/apache2/apache2.conf /etc/apache2/apache2.conf.bak
Now open the configuration file:
sudo nano /etc/apache2/apache2.conf
Scroll down (down arrow) to where it says “ServerTokens Full” and change it to read “ServerTokens Prod”
Now, scroll down a little further and change “ServerSignature On” to “ServerSignature Off”
Finally, press Control-O followed by Control-X. That will save the file and exit the text editor.
Now, we need to do the same thing for PHP. First, back up the original PHP configuration file:
sudo cp /etc/php5/apache2/php.ini /etc/php5/apache2/php.ini.bak
Open the configuration file:
sudo nano /etc/php5/apache2/php.ini
Change “expose_php = On” to “expose_php = Off”
Again, press Control-O followed by Control-X. Now that the configuration files are updated, restart Apache:
sudo /etc/init.d/apache2 restart
You are done setting up Apache, MySQL, and PHP.

6. Install a Firewall

We now are going to lock down our server a bit more by installing Shorewall, a command-line firewall. To install it:
sudo aptitude install shorewall
By default, Shorewall is installed with no rules, allowing complete access. However, this is not the behavior we want.
Instead, we’re going to block all connections to anything other than port 80 (HTTP) and port 22 (SSH). First, copy the configuration
files to the Shorewall directory:
sudo cp /usr/share/doc/shorewall-common/examples/one-interface/* /etc/shorewall/
Now, open the “rules” file:
sudo nano /etc/shorewall/rules
Add these lines above where it says “#LAST LINE”
HTTP/ACCEPT	net		$FW
SSH/ACCEPT	net		$FW
Then press Control-O and Control-X. Your firewall is now configured to only accept HTTP and SSH traffic. The last thing we need to
do is tell Shorewall to start on boot. So, open up the main Shorewall configuration file:
sudo nano /etc/shorewall/shorewall.conf
Scroll down to “STARTUP_ENABLED=No” and set it to “STARTUP_ENABLED=Yes”
Press Control-O and Control-X. Now, open the Shorewall default configuration file:
sudo nano /etc/default/shorewall
And change “startup=0″ to “startup=1″. Press Control-O and Control-X. Finally, start your firewall:
sudo /etc/init.d/shorewall start
Congratulations! Your firewall is now set up and protecting your server.

7. Add Your Website to Your Web Server

Now that you’ve got everything all set up, you’d probably like to add a website to it. By default, all of the files Apache serves
up to the internet are located at “/var/www/”. However, you cannot write to this folder. Let’s make it so you can:
sudo usermod -g www-data [YOUR USERNAME]
sudo chown -R www-data:www-data /var/www
sudo chmod -R 775 /var/www
What happened there was you added yourself to the “www-data” group, and made the website folder writable to the members of the “www-data”
group.
Now, you’re going to log into your server using SFTP (not to be confused with FTPS). Some clients that support SFTP are:
WinSCP (Windows, Free), FileZilla (Windows, Linux, OS X, Free),
Cyberduck (OS X, Free), and, my personal favorite, Transmit
(OS X, $30)
Connect to your server using your username and password, and, if your client supports it, a default path of “/var/www” (if it doesn’t,
simply browse to /var/www once you have logged in): (Transmit pictured)
You may now add your files to this folder (/var/www) and they will show up on your server when you browse to it with your
web browser.
Now, you may wonder why we’re using SFTP instead of FTP. Mainly, because SFTP is already built into OpenSSH (which you installed
earlier). However, it is also a lot more secure than FTP, and makes it difficult (if not impossible) for malicious users to gain access
to your login credentials.

8. Make Your Server Accesible to the Internet

Most modern home networks are behind a router these days. Because of this, your web server will not be visible to the internet without
a little work. As I don’t have every router available to test with, I can only give you general directions in this area.
There are two ways to open your server up to the internet: a DMZ or Port Forwarding. The main difference you’ll notice is that with a DMZ, your server uses the
firewall we installed earlier to protect itself. However, with Port Forwarding, your server will be protected by your router’s firewall.
However, before we go on, you’re going to want to give your server a static LAN address. To do that, login to your router, and look for something
along the lines of “Static IPs” or “Static Routing”. After you have given your server a static LAN address, you can do these next parts. Remember,
Google is your friend.
To port foward, there is an excellent website, PortForward.com, that, while ugly, can
help you get the job done for almost any router. The ports that you want to forward are 22 and 80.
To create a DMZ, you need to login to your router and look for something like “DMZ settings”. Once you find it, add your server to
the DMZ, and you’ll be set. Again, Google is helpful in situations like this.
Now, find your public IP, and voila! You can access your server from anywhere as long
as your IP doesn’t change.

9. Managing Your Server Remotely

Beside allowing you to upload files, OpenSSH allows you to login to your server from anywhere as long as you know it’s IP. For Windows, you’ll
need an SSH client. I recommend Putty. For OS X, SSH is
already installed. Simply open up Terminal, and type “ssh you@yourip“. For Putty, choose SSH, and put in your IP, username, and password
when it asks for it. You’ll notice that, once you login, it looks exactly the same as the screen on the server:
You can do anything from here that you would do actually sitting at the server. To logout from the server, simply type “exit” and hit enter.

10. That’s It!

You now have a completely functioning web server. It makes for a great testing ground, and would even be suitable to host websites with fairly
low traffic. There is obviously a lot left to be learned, but hopefully you have gained a little insight into how web servers work.