Skip to content

Instantly share code, notes, and snippets.

@xbeat
Created April 25, 2018 15:13
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save xbeat/34dcb5a701fa97ea547f502283490f93 to your computer and use it in GitHub Desktop.
Save xbeat/34dcb5a701fa97ea547f502283490f93 to your computer and use it in GitHub Desktop.
Pong 3D
<canvas id="floor-texture" width="512" height="256"></canvas>
<div id="score-left"></div>
<div id="score-right"></div>
<div id="how">Position mouse left and right over game area to control paddles / bats</div>
<div id="btn-container">
<button id="btn-go-fs" class="btn-fs" onclick="requestFullscreen()">go fullscreen</button>
<button id="btn-exit-fs" class="btn-fs" onclick="exitFullscreen()">exit fullscreen</button>
</div>
<div id="debug-container">
<div id="out"></div>
</div>
// Author: darcey@aftc.io
// Libs: aftc.js & three
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
var Application = function () {
var args = {};
var params = {
dom: {
container: false,
canvas1:false,
},
ctx1:false,
textures: {
floor: false,
room: false,
},
materials: {
floor: false,
room: false,
},
meshes: {
floor: false,
room: false,
},
canvas1HalfW:0, canvas1HalfH:0,
cX: 0, cY: 0, // Cneter
sX: 0, sY: 0, // Start
pX: 0, pY: 0, // Previous
tX: 0, tY: 0, // Target
w: 0, h: 0,
x: 0, y: 0,
t: 0,
leftScore: 0,
rightScore: 0,
running: false,
batPosYPercent: false,
batPosZ: false,
BALLDIAMETER: 1,
HEIGHT: 1,
newBallX: 0,
newBallY: 0,
batLeftLowerZ: false,
batLeftUpperZ: false,
batRightLowerZ: false,
batRightUpperZ: false,
leftCollisionPoint: false,
rightCollisionPoint: false,
};
// constructor
function init() {
if (isMobile()) {
// setupGame();
} else {
var opener = new Audio("https://dev.aftc.io/assets/sounds/shall_we_play_a_game.mp3");
// opener.addEventListener("ended",setupGame);
opener.play();
}
html("v1", "html(\"v1\",str)");
html("v2", "html(\"v2\",str)");
html("v3", "html(\"v3\",str)");
html("v4", "html(\"v4\",str)");
params.dom.container = getElementById("container");
//params.dom.container.getBoundingClientRect().width;
// params.w = Math.round(parseFloat(getComputedStyle(params.dom.container).width));
// params.h = Math.round(parseFloat(getComputedStyle(params.dom.container).height));
params.w = window.innerWidth;
params.h = window.innerHeight;
params.cX = params.w / 2;
params.cY = params.h / 2;
params.dom.canvas1 = getElementById("floor-texture");
params.ctx1 = params.dom.canvas1.getContext("2d");
params.canvas1HalfW = params.dom.canvas1.width / 2;
params.canvas1HalfH = params.dom.canvas1.height / 2;
setupScene();
}
function updateFloorTexture(){
html("score-right", params.rightScore);
html("score-left", params.leftScore);
params.ctx1.fillStyle = "#232323";
params.ctx1.fillRect(0, 0, params.dom.canvas1.width, params.dom.canvas1.height);
params.ctx1.beginPath();
params.ctx1.setLineDash([15, 15]);
params.ctx1.moveTo(params.canvas1HalfW, 0);
params.ctx1.lineTo(params.canvas1HalfW, params.dom.canvas1.height);
params.ctx1.strokeStyle = "RGBA(255,255,255,0.5)";
params.ctx1.lineWidth = 10;
params.ctx1.stroke();
params.ctx1.font = '100pt VT323';
params.ctx1.fillStyle = "RGBA(255,255,255,0.9)";
// Classic pong score positions
// params.ctx1.fillText(params.leftScore, (params.canvas1HalfW-76), 50);
// params.ctx1.fillText(params.rightScore, (params.canvas1HalfW+30), 50);
params.ctx1.fillText(params.leftScore, 60, 120);
params.ctx1.fillText(params.rightScore, 320, 120);
if (params.meshes.floor){
params.textures.floor = new THREE.Texture(params.dom.canvas1);
params.textures.floor.needsUpdate = true;
params.materials.floor.map = params.textures.floor;
}
if (params.shader){
if (params.shader.renderToScreen){
params.shader.renderToScreen = false;
} else {
params.shader.renderToScreen = true;
}
}
}
function setupScene(){
if (!Detector.webgl) Detector.addGetWebGLMessage();
params.scene = new THREE.Scene();
params.renderer = new THREE.WebGLRenderer({ antialias: true });
params.renderer.setPixelRatio(window.devicePixelRatio);
params.renderer.setSize(params.w, params.h);
params.renderer.setClearColor("#0C7CA5", 0);
document.body.appendChild(params.renderer.domElement);
// camera
params.camera = new THREE.PerspectiveCamera(60, params.w / params.h, 1, 5000);
params.camera.position.set(2, 8, 10);
params.scene.add(params.camera);
// controls
params.controls = new THREE.OrbitControls(params.camera, params.renderer.domElement);
// params.controls.minDistance = 5;
// params.controls.maxDistance = 10;
params.controls.maxPolarAngle = Math.PI / 2;
// stats
// params.stats = new Stats()
// document.body.appendChild(params.stats.dom);
// lights
params.scene.add(new THREE.AmbientLight(0x222222));
// params.light = new THREE.PointLight( 0xffffff, 1 );
// params.camera.add( params.light );
var light = new THREE.DirectionalLight(0xffffff, 1);
light.position.set(75, 75, 50);
light.castShadow = true;
var dLight = 200;
var sLight = dLight * 0.25;
light.shadow.camera.left = -sLight;
light.shadow.camera.right = sLight;
light.shadow.camera.top = sLight;
light.shadow.camera.bottom = -sLight;
light.shadow.camera.near = dLight / 30;
light.shadow.camera.far = dLight;
light.shadow.mapSize.x = 1024 * 2;
light.shadow.mapSize.y = 1024 * 2;
params.scene.add(light);
// helper
// params.scene.add( new THREE.AxesHelper( 5 ) );
// scene
// params.group = new THREE.Group();
// params.scene.add(params.group);
updateFloorTexture();
params.textures.floor = new THREE.Texture(params.dom.canvas1);
params.textures.floor.needsUpdate = true;
params.materials.floor = new THREE.MeshLambertMaterial({
// color: 0x333333,
map: params.textures.floor
});
var matWall = new THREE.MeshLambertMaterial({
color: 0xffffff,
});
var matBat = new THREE.MeshLambertMaterial({
color: 0x00FF00,
});
var ballMat = new THREE.MeshLambertMaterial({
color: 0xFFFF00,
});
params.gameW = 70;
params.gameH = 30;
// Floor
var floorGeom = new THREE.PlaneGeometry(params.gameW, params.gameH);
params.meshes.floor = new THREE.Mesh(floorGeom, params.materials.floor);
params.meshes.floor.rotation.x = degToRad(-90);
params.scene.add(params.meshes.floor);
// Walls
var topWallGeom = new THREE.BoxGeometry(params.gameW-0.5, 1, 1);
var topWall = new THREE.Mesh(topWallGeom, matWall);
topWall.position.x = 0;
topWall.position.y = 0.5;
topWall.position.z = -(params.gameH / 2);
params.scene.add(topWall);
var btmWall = new THREE.Mesh(topWallGeom, matWall);
btmWall.position.x = 0;
btmWall.position.y = 0.5;
btmWall.position.z = (params.gameH / 2);
params.scene.add(btmWall);
// bats
var batGeom = new THREE.BoxGeometry(0.5, 1, 5);
params.batLeft = new THREE.Mesh(batGeom, matBat);
params.batLeft.position.x = -(params.gameW / 2);
params.batLeft.position.y = 0.5;
params.batLeft.position.z = 0;
params.scene.add(params.batLeft);
params.batRight = new THREE.Mesh(batGeom, matBat);
params.batRight.position.x = (params.gameW / 2);
params.batRight.position.y = 0.5;
params.batRight.position.z = 0;
params.scene.add(params.batRight);
var ballGeom = new THREE.BoxGeometry(0.5, 0.5, 0.5);
params.ball = new THREE.Mesh(ballGeom, ballMat);
params.ball.position.y = 0.5;
params.scene.add(params.ball);
params.ballX = params.sX;
params.ballY = params.sY;
params.ballVx = -0.2;
params.ballVy = 0.1;
params.upperLim = (params.gameH / 2) - 0.5;
params.lowerLim = -params.upperLim;
params.mX = 0;
params.mY = 0;
document.body.addEventListener("mousemove", setMouseCoords);
window.addEventListener('resize', resizeHandler, false);
// postprocessing
params.composer = new THREE.EffectComposer( params.renderer );
params.composer.addPass( new THREE.RenderPass( params.scene, params.camera ) );
params.shader = new THREE.FilmPass( 0.35, 0.5, 2048, true );
params.shader.renderToScreen = true;
params.composer.addPass( params.shader );
var texture = new THREE.Texture();
var manager = new THREE.LoadingManager();
manager.onProgress = function (item, loaded, total) {
// console.log(item, loaded, total);
};
var onProgress = function (xhr) {
if (xhr.lengthComputable) {
var percentComplete = xhr.loaded / xhr.total * 100;
console.log(Math.round(percentComplete, 2) + '% downloaded');
}
};
var onError = function (xhr) { };
function imgLoadedHandler(image) {
texture.image = image;
texture.needsUpdate = true;
}
var loader = new THREE.ImageLoader(manager);
loader.load("", imgLoadedHandler);
// flip textures
texture.wrapS = THREE.RepeatWrapping;
texture.repeat.x = - 1;
// room
params.materials.room = new THREE.MeshLambertMaterial({
color: 0x003399,
// receiveShadow:true,
// castShadow:true
// opacity: 1,
// transparent: false
side: THREE.BackSide,
map: texture
});
var roomGeom = new THREE.BoxGeometry(200, 40, 200);
params.meshes.room = new THREE.Mesh(roomGeom, params.materials.room);
params.meshes.room.position.x = 0;
params.meshes.room.position.y = 10;
params.meshes.room.position.z = 0;
params.scene.add(params.meshes.room);
// Animate and Render
animate();
}
// - - - - - - - - - - - - - - - - -
function setMouseCoords(e) {
// params.mX = e.clientX-params.dom.container.offsetLeft;
// params.mY = e.clientY-params.dom.container.offsetTop;
params.mX = e.clientX;
params.mY = e.clientY;
}
function getDirection() {
}
function getBatLeftY() {
return params.batLeft.z;
}
function getBatRightY() {
return params.batRight.z;
}
function animate() {
// Position the bats
params.batPosYPercent = Math.floor((100 / params.w) * (Math.abs(params.mX)));
params.batPosZ = -15 + (30 / 100) * params.batPosYPercent;
params.batLeft.position.z = params.batPosZ;
params.batRight.position.z = -params.batPosZ;
params.newBallX = params.ballX + params.ballVx * 1;
params.newBallY = params.ballY + params.ballVy * 1;
// Top and bottom edges, simply bounce
if (params.newBallY < params.lowerLim) {
params.ballVy = -params.ballVy;
params.newBallY = params.ballY + params.ballVy * 1;
playSound("wall");
} else if (params.newBallY > params.upperLim) {
params.ballVy = -params.ballVy;
params.newBallY = params.ballY + params.ballVy * 1;
playSound("wall");
}
// Left paddle (paddle1)
params.batLeftLowerZ = params.batLeft.position.z - 3;
params.batLeftUpperZ = params.batLeft.position.z + 3
params.batRightLowerZ = params.batRight.position.z - 3;
params.batRightUpperZ = params.batRight.position.z + 3
params.leftCollisionPoint = -(params.gameW / 2) + 0.5;
params.rightCollisionPoint = -params.leftCollisionPoint;
var reset = false;
if (params.newBallX < params.leftCollisionPoint) {
if (params.newBallY >= params.batLeftLowerZ && params.newBallY <= params.batLeftUpperZ) {
params.ballVx = -params.ballVx;
params.newBallX = params.ballX + params.ballVx * 1;
playSound("bat");
} else {
playSound("fail");
params.rightScore++;
updateFloorTexture();
params.newBallX = params.sX;
params.newBallY = params.sY;
restart();
reset = true;
}
} else if (params.newBallX > params.rightCollisionPoint) {
if (params.newBallY >= params.batRightLowerZ && params.newBallY <= params.batRightUpperZ) {
params.ballVx = -params.ballVx;
params.newBallX = params.ballX + params.ballVx * 1;
playSound("bat");
} else {
playSound("fail");
params.leftScore++;
updateFloorTexture();
restart();
reset = true;
}
}
if (!reset) {
params.ballX = params.newBallX;
params.ballY = params.newBallY;
params.ball.position.x = params.newBallX;
params.ball.position.z = params.newBallY;
// speed up!
params.ballVx *= 1.001;
if (params.ballVy > 0) {
params.ballVy *= 1.0001;
}
// Make the camera more interesting
// left target -36 9 12
// right target 36 9 12
params.camXLim = 42;
params.camXasPercent = (100 / (params.gameW / 2)) * params.newBallX; // -100 to 100
// html("out",params.camXasPercent);
params.camRange = 10;
params.camX = (params.camRange / 10) * params.camXasPercent;
// html("out",camX);
if (params.camX > params.camXLim) {
params.camX = params.camXLim;
} else if (params.camX < -params.camXLim) {
params.camX = -params.camXLim;
}
params.camera.position.x = params.camX;
params.camera.position.y = 8;
params.camera.position.z = 16;
params.camera.lookAt(params.ball.position);
}
params.ball.rotation.z -= (params.ballVx / 3);
params.ball.rotation.x -= (params.ballVx / 3);
// if (params.ballVx > 0){
// params.ball.rotation.z += params.ballVx;
// } else {
// params.ball.rotation.z += params.ballVx;
// }
requestAnimationFrame(animate);
render();
}
// - - - - - - - - - - - - - - - - -
function playSound(name) {
var url;
var error = false;
switch (name) {
case "wall":
url = "https://dev.aftc.io/assets/sounds/pong.mp3"
break;
case "bat":
url = "https://dev.aftc.io/assets/sounds/pong2.mp3"
break;
case "fail":
url = "https://dev.aftc.io/assets/sounds/fail_01.mp3"
break;
default:
error = true;
break;
}
if (!error) {
var s = new Audio(url);
s.play();
}
}
function render() {
// var msg = "x:" + params.camera.position.x.toFixed(1);
// msg += " y:" + params.camera.position.y.toFixed(1);
// msg += " z:" + params.camera.position.z.toFixed(1);
// html("v1",msg);
// params.stats.update();
if (params.shader.renderToScreen){
params.composer.render();
} else {
params.renderer.render(params.scene, params.camera);
}
}
// - - - - - - - - - - - - - - - - -
function resizeHandler() {
params.w = window.innerWidth;
params.h = window.innerHeight;
params.camera.aspect = window.innerWidth / window.innerHeight;
params.camera.updateProjectionMatrix();
params.renderer.setSize(window.innerWidth, window.innerHeight);
}
// - - - - - - - - - - - - - - - - -
function start() {
log("Application.start()");
}
// - - - - - - - - - - - - - - - - -
function stop() {
log("Application.stop()");
}
// - - - - - - - - - - - - - - - - -
function restart() {
params.ballX = params.sX;
params.ballY = params.sY;
var velXmin = -getRandomFloat(0.1, 0.2);
var velXmax = getRandomFloat(0.1, 0.2);
var velYmin = -getRandomFloat(0.1, 0.15);
var velYmax = getRandomFloat(0.1, 0.15);
var velX = [velXmin, velXmax];
var velY = [velYmin, velYmax];
params.ballVx = velX[getRandomInt(0, 1)];
params.ballVy = velY[getRandomInt(0, 1)];
html("v4", "vx:" + params.ballVx.toFixed(3) + " vy:" + params.ballVy.toFixed(3));
}
// - - - - - - - - - - - - - - - - -
// Utils
// - - - - - - - - - - - - - - - - -
// Public
this.start = function () {
};
this.stop = function () {
};
this.reset = function () {
restart();
};
// - - - - - - - - - - - - - - - - -
// Constructor simulation
init();
// - - - - - - - - - - - - - - - - -
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// Globals
var app1;
function start() {
app1.start();
}
function stop() {
app1.stop();
}
function reset() {
app1.reset();
}
function requestFullscreen() {
var ele = document.body;
getElementById("btn-go-fs").style.display = "none";
getElementById("btn-exit-fs").style.display = "inline-block";
if (ele.requestFullscreen) {
ele.requestFullscreen();
} else if (ele.webkitRequestFullscreen) {
ele.webkitRequestFullscreen();
} else if (ele.mozRequestFullScreen) {
ele.mozRequestFullScreen();
} else if (ele.msRequestFullscreen) {
ele.msRequestFullscreen();
} else {
console.log('Fullscreen API is not supported.');
}
};
function exitFullscreen() {
getElementById("btn-go-fs").style.display = "inline-block";
getElementById("btn-exit-fs").style.display = "none";
if (document.exitFullscreen) {
document.exitFullscreen();
} else if (document.webkitExitFullscreen) {
document.webkitExitFullscreen();
} else if (document.mozCancelFullScreen) {
document.mozCancelFullScreen();
} else if (document.msExitFullscreen) {
document.msExitFullscreen();
} else {
console.log('Fullscreen API is not supported.');
}
};
function setupGame() {
app1 = new Application();
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// Document Ready
onReady(function () {
logTo("out");
log("Darcey@AFTC.io");
log("<span style='color:#FFCC00'>Position mouse left and right over game area to control paddles / bats</a>");
getElementById("btn-go-fs").style.display = "inline-block";
getElementById("btn-exit-fs").style.display = "none";
app1 = new Application();
});
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
<script src="https://cdn.jsdelivr.net/npm/aftc.js@1.5.12/dist/aftc.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/gsap/latest/TweenMax.min.js"></script>
<script src="https://threejs.org/examples/js/Detector.js"></script>
<script src="https://threejs.org/examples/js/libs/stats.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/91/three.min.js"></script>
<script src="https://threejs.org/examples/js/controls/OrbitControls.js"></script>
<script src="https://threejs.org/examples/js/loaders/OBJLoader.js"></script>
<script src="https://threejs.org/examples/js/shaders/CopyShader.js"></script>
<script src="https://threejs.org/examples/js/shaders/FilmShader.js"></script>
<script src="https://threejs.org/examples/js/postprocessing/EffectComposer.js"></script>
<script src="https://threejs.org/examples/js/postprocessing/RenderPass.js"></script>
<script src="https://threejs.org/examples/js/postprocessing/MaskPass.js"></script>
<script src="https://threejs.org/examples/js/postprocessing/ShaderPass.js"></script>
<script src="https://threejs.org/examples/js/postprocessing/FilmPass.js"></script>
@import url(https://fonts.googleapis.com/css?family=VT323);
/* Generics */
body {
font-family: Arial;
font-size: 12px;
background: #000000;
color: #ffffff;
cursor: corshair;
background: #000000 url("https://dev.aftc.io/assets/img/pong_bg.jpg");
background-size: cover;
background-repeat: no-repeat;
}
.btn-fs {
background: #993300;
color: #FFFFFF;
border: none;
}
.btn-fs:hover {
background: #FFCC00;
color: #000000;
}
a {
color: #FFCC00;
text-decoration: none;
}
a:hover {
text-decoration: underline;
}
button {
cursor: pointer;
}
.outer {
width: 100%;
}
.inner {
width: 760px;
margin: auto;
}
.flex-row {
display: flex;
flex-wrap: nowrap;
justify-content: center;
}
/* Debug */
#debug-container {
position: absolute;
width: 100%;
left: 0;
bottom: 20px;
background: #161616;
opacity: 0.3;
}
#debug-container:hover {
opacity: 0.7;
}
#debug-container .btn {
display: block;
width: 23.5%;
margin: 2px;
padding: 3px;
box-sizing: border-box;
}
#debug-container .btn button {
width: 100%;
font-size: 12px;
font-weight: bold;
background: #cccccc;
}
#debug-container .btn button:hover {
width: 100%;
background: #ffffff;
}
#debug-container .in {
display: block;
width: 23.5%;
margin: 2px;
padding: 3px;
box-sizing: border-box;
}
#debug-container .in .input-label {
font-size: 12px;
}
#debug-container .in input {
width: 100%;
box-sizing: border-box;
}
#debug-container .out {
display: block;
width: 23.5%;
margin: 2px;
padding: 3px;
font-family: "courier";
font-size: 11px;
border: 1px solid #676767;
box-sizing: border-box;
}
#debug-container #out {
max-height: 200px;
overflow: auto;
margin: 10px 10px 20px 10px;
padding: 3px;
font-family: "courier";
font-size: 11px;
border: 1px solid #676767;
}
#debug-container .clear-debug {
display: block;
margin: 10px auto -7px auto;
font-size: 10px;
background: #999999;
color: #000000;
border: none;
outline: none;
}
#debug-container .clear-debug:hover {
background: #FFFFFF;
color: #000000;
}
#container {
position: relative;
width: 760px;
height: 400px;
border: 1px solid #1b2735;
cursor: crosshair;
}
#score-left {
z-index: 10;
position: absolute;
left: 60px;
top: 10px;
font-family: 'VT323', Courier, monospace !important;
font-size: 80px;
opacity: 0.4;
}
#score-right {
z-index: 10;
position: absolute;
right: 60px;
top: 10px;
font-family: 'VT323', Courier, monospace !important;
font-size: 80px;
opacity: 0.4;
}
#how {
z-index: 30;
position: absolute;
width: 100%;
left: 0;
top: 5px;
padding: 5px;
text-align: center;
color: #FFCC00;
opacity: 0.7;
text-transform: uppercase;
}
#floor-texture {
z-index: 1;
display: none;
z-index: 100;
position: absolute;
left: 5px;
top: 5px;
opacity: 0.7;
/* background: #FFCC00; */
}
#btn-container {
z-index: 100;
position: absolute;
left: 0;
top: 25px;
width: 100%;
text-align: center;
}
#btn-exit-fullscreen {
z-index: 100;
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment