Skip to content

Instantly share code, notes, and snippets.

@e1blue
Created March 21, 2018 13:52
Show Gist options
  • Save e1blue/5ff46636e2766801215110fc5702e0a6 to your computer and use it in GitHub Desktop.
Save e1blue/5ff46636e2766801215110fc5702e0a6 to your computer and use it in GitHub Desktop.
Jack in a Box Factory

Jack in a Box Factory

These Jack in a boxes are all made, tested and then discarded straight away. Not sure how this factory makes any money.

Made with GSAP and Three.js. Everything except the face texture was created in code.

A Pen by Steve Gardner on CodePen.

License.

console.clear();
declare var THREE:any;
declare var TweenMax:any;
declare var TimelineMax:any;
declare var Bounce:any;
declare var Power2:any;
declare var Power3:any;
declare var Elastic:any;
TweenMax.lagSmoothing(0);
THREE.ImageUtils.crossOrigin = '';
interface JackInABox {
jack: Jack;
box: Box;
group: any;
}
let colors =
{
blue: '#4281A4',
green: '#48A9A6',
background: '#D0CBC7',
ground: '#E4DFDA',
yellow: '#D4B483',
red: '#C1666B',
lid: '#B05D62',
skin: '#F5D6BA'
}
let settings =
{
openSpeed: 0.8
}
let Utils =
{
degToRad: function(degrees)
{
return degrees * Math.PI / 180;
}
}
class Stage
{
floor: any;
private container: any;
private camera: any;
private scene: any;
private renderer: any;
private light: any;
private softLight: any;
private group: any;
constructor()
{
// container
this.container = document.createElement( 'div' );
document.body.appendChild( this.container );
// renderer
this.renderer = new THREE.WebGLRenderer({
antialias: true,
alpha: false
});
this.renderer.setSize(window.innerWidth, window.innerHeight);
this.renderer.setClearColor(colors.background, 1);
this.renderer.shadowMap.enabled = true;
this.renderer.shadowMap.type = THREE.PCFSoftShadowMap;
this.container.appendChild( this.renderer.domElement );
// scene
this.scene = new THREE.Scene();
//this.scene.fog = new THREE.Fog( colors.ground, 0, 10);
// camera
let aspect = window.innerWidth / window.innerHeight;
let d = 20;
this.camera = new THREE.OrthographicCamera( - d * aspect, d * aspect, d, - d, -100, 1000);
this.camera.position.x = 30;
this.camera.position.y = 10;
this.camera.position.z = -10;
this.camera.lookAt(new THREE.Vector3(0, 5, 0));
//light
this.light = new THREE.DirectionalLight(0xffffff, 0.5);
this.light.castShadow = true;
this.light.position.set(8, 15, -5);
this.scene.add(this.light);
this.softLight = new THREE.AmbientLight( 0xffffff, 0.5 );
this.scene.add(this.softLight)
// floor
var width = 500;
var depth = 1000;
var geometry = new THREE.PlaneGeometry(depth, width);
var material = new THREE.MeshBasicMaterial({
color: colors.ground,
shading: THREE.FlatShading
});
var plane = new THREE.Mesh(geometry, material);
plane.rotation.x = -Utils.degToRad(90); //1.57079633;
plane.position.y = 0;
plane.position.z = - width / 2;
this.floor = new THREE.Group();
this.floor.add(plane);
this.scene.add(this.floor);
// group
this.group = new THREE.Group();
this.scene.add(this.group);
}
render = function()
{
this.renderer.render(this.scene, this.camera);
}
add = function(elem)
{
this.scene.add(elem);
}
remove = function(elem)
{
this.scene.remove(elem);
}
}
class Box
{
group: any;
private lid: any;
private crank: any;
private openAnimation = new TimelineMax();
private closeAnimation = new TimelineMax();
private dropAnimation = new TimelineMax();
private fallAnimation = new TimelineMax();
closed: boolean;
private crankSpeed: number;
private crankAnimation: any;
constructor()
{
this.closed = true;
this.group = new THREE.Group();
var material = new THREE.MeshToonMaterial({
color: colors.red
});
var geometry = new THREE.BoxGeometry(4, 1, 4);
var boxBottom = new THREE.Mesh(geometry, material);
boxBottom.position.y = 0.5;
//boxBottom.castShadow = true;
//boxBottom.receiveShadow = true;
var geometry = new THREE.BoxGeometry(6, 1, 6);
var boxBack = new THREE.Mesh(geometry, material);
boxBack.position.y = 2.5;
boxBack.position.x = -2.5;
boxBack.rotation.z = Utils.degToRad(90);
//boxBack.castShadow = true;
//boxBack.receiveShadow = true;
var geometry = new THREE.BoxGeometry(6, 1, 6);
var boxFront = new THREE.Mesh(geometry, material);
boxFront.position.y = 2.5;
boxFront.position.x = 2.5;
boxFront.rotation.z = Utils.degToRad(90);
//boxFront.castShadow = true;
//boxFront.receiveShadow = true;
var geometry = new THREE.BoxGeometry(4, 1, 6);
var boxLeft = new THREE.Mesh(geometry, material);
boxLeft.position.y = 2.5;
boxLeft.position.z = 2.5;
boxLeft.rotation.x = Utils.degToRad(90);
//boxLeft.castShadow = true;
//boxLeft.receiveShadow = true;
var geometry = new THREE.BoxGeometry(4, 1, 6);
var boxRight = new THREE.Mesh(geometry, material);
boxRight.position.y = 2.5;
boxRight.position.z = -2.5;
boxRight.rotation.x = Utils.degToRad(90);
//boxRight.castShadow = true;
//boxRight.receiveShadow = true;
var geometry = new THREE.BoxGeometry(6, 1.75, 6);
geometry.applyMatrix(new THREE.Matrix4().makeTranslation(3, 1, 0));
var material = new THREE.MeshToonMaterial({
color: colors.lid
});
this.lid = new THREE.Mesh(geometry, material);
this.lid.position.y = 5.4;
this.lid.position.x = -3;
//this.lid.castShadow = true;
//this.lid.receiveShadow = true;
this.group.add(this.lid);
this.group.add(boxBottom);
this.group.add(boxBack);
this.group.add(boxLeft);
this.group.add(boxRight);
this.group.add(boxFront);
this.crank = new THREE.Group();
var geometry = new THREE.CylinderGeometry(0.3, 0.3, 0.6, 8);
var material = new THREE.MeshToonMaterial({
color: colors.yellow
});
var axel = new THREE.Mesh(geometry, material);
axel.rotation.x = Utils.degToRad(90);
//axel.castShadow = true;
//axel.receiveShadow = true;
var geometry = new THREE.BoxGeometry(1, 0.3, 3, 8);
var material = new THREE.MeshToonMaterial({
color: colors.yellow
});
var shaft = new THREE.Mesh(geometry, material);
shaft.position.y = 1;
shaft.rotation.x = Utils.degToRad(90);
//shaft.castShadow = true;
//shaft.receiveShadow = true;
var geometry = new THREE.CylinderGeometry(0.5, 0.5, 1, 8);
var material = new THREE.MeshToonMaterial({
color: colors.yellow
});
var handle = new THREE.Mesh(geometry, material);
handle.position.z = -0.6;
handle.position.y = 2;
handle.rotation.x = Utils.degToRad(90);
//handle.castShadow = true;
//handle.receiveShadow = true;
this.crank.add(axel);
this.crank.add(shaft);
this.crank.add(handle);
this.crank.position.z = -3.5;
this.crank.position.y = 3.5;
this.crank.rotation.z = Utils.degToRad(Math.random() * 360);
this.group.add(this.crank);
this.group.position.y = 20;
this.setAnimation();
}
setAnimation = function()
{
var speed = 1;
var boxBounceSpeed = speed / 3;
this.openAnimation.clear();
this.openAnimation.stop();
this.openAnimation.to(this.group.position, boxBounceSpeed, {
y: 4,
delay: 0,
ease: Power2.easeInOut
});
this.openAnimation.to(this.group.position, boxBounceSpeed * 2, {
y: 0,
x: 0,
z: 0,
ease: Bounce.easeOut
});
this.openAnimation.to(this.lid.rotation, speed, {
z: 2,
ease: Bounce.easeOut
}, 0);
this.openAnimation.to(this.crank.rotation, speed * 3, {
z: Utils.degToRad(180),
ease: Elastic.easeOut
}, 0);
// -----
this.crankSpeed = 0;
this.crankAnimation = setInterval(() => {this.moveCrank()}, 20);
// -----
this.dropAnimation.clear();
this.dropAnimation.stop();
this.dropAnimation.to(this.group.position, 1, {y: 0, delay: 0.5, ease: Bounce.easeOut});
// -----
this.fallAnimation.clear();
this.fallAnimation.stop();
this.fallAnimation.to(this.group.position, 1, {y: -20, delay: 0.2, ease: Power3.easeIn});
this.fallAnimation.to(this.group.rotation, 1, {x: Utils.degToRad(120), delay: 0.2, ease: Power3.easeIn}, 0);
}
moveCrank = function()
{
if (this.closed)
{
this.crank.rotation.z = this.crank.rotation.z > Utils.degToRad(360) ? 0 : this.crank.rotation.z + this.crankSpeed;
if (this.crankSpeed < 0.1) this.crankSpeed += 0.005;
}
}
add = function(elem)
{
this.group.add(elem);
}
stopAnimations = function()
{
this.openAnimation.stop();
this.closeAnimation.stop();
}
open = function(skipAnimation:boolean = false)
{
this.closed = false;
//this.stopAnimations();
this.openAnimation.duration(skipAnimation ? 0.1 : 3);
this.openAnimation.restart();
}
close = function()
{
this.closed = true;
//this.stopAnimations();
this.closeAnimation.duration(3);
this.closeAnimation.restart();
}
drop = function(skipAnimation:boolean = false)
{
this.dropAnimation.duration(skipAnimation ? 0.1 : 1);
this.dropAnimation.restart();
}
fall = function(skipAnimation:boolean = false)
{
this.fallAnimation.duration(skipAnimation ? 0.1 : 1);
this.fallAnimation.restart();
}
}
class Jack
{
group: any;
private body: any;
private head: any;
private leftHand: any;
private rightHand: any;
private openAnimation = new TimelineMax();
private closeAnimation = new TimelineMax();
private moveAnimation = new TimelineMax();
constructor(color: string, face: string)
{
this.group = new THREE.Group();
var geometry = new THREE.CylinderGeometry(1.1, 1.3, 2.5, 15);
var material = new THREE.MeshToonMaterial({
color: color
});
this.body = new THREE.Mesh(geometry, material);
this.body.position.y = 2.5;
//this.body.castShadow = true;
//this.body.receiveShadow = true;
var material = new THREE.MeshToonMaterial({
color: colors.skin,
map: face
});
var geometry = new THREE.SphereGeometry(1.7, 32, 32);
this.head = new THREE.Mesh(geometry, material);
this.head.position.y = 4;
//this.head.castShadow = true;
//this.head.receiveShadow = true;
var material = new THREE.MeshToonMaterial({
color: colors.skin
});
var geometry = new THREE.SphereGeometry(0.5, 32, 32);
this.leftHand = new THREE.Mesh(geometry, material);
this.leftHand.position.z = -1.5;
this.leftHand.position.y = 2;
//this.leftHand.castShadow = true;
//this.leftHand.receiveShadow = true;
var geometry = new THREE.SphereGeometry(0.5, 32, 32);
this.rightHand = new THREE.Mesh(geometry, material);
this.rightHand.position.z = 1.5;
this.rightHand.position.y = 2;
//this.rightHand.castShadow = true;
//this.rightHand.receiveShadow = true;
this.group.add(this.body);
this.group.add(this.head);
this.group.add(this.leftHand);
this.group.add(this.rightHand);
this.setAnimation();
}
setAnimation = function()
{
var speed = 1;
// open animation
this.openAnimation.clear();
this.openAnimation.stop();
var jackDelay = 0.2;
this.openAnimation.to(this.group.position, speed * 2, {
y: 5,
ease: Elastic.easeOut
}, jackDelay);
this.openAnimation.to(this.leftHand.position, speed * 2, {
z: -2.1,
y: 3.8,
ease: Elastic.easeOut
}, jackDelay)
this.openAnimation.to(this.rightHand.position, speed * 2, {
z: 2.1,
y: 3.8,
ease: Elastic.easeOut
}, jackDelay)
this.openAnimation.to(this.group.rotation, speed / 3, {
x: Math.random() - 0.5,
z: Math.random() - 0.5,
ease: Power3.easeInOut
}, jackDelay);
this.openAnimation.to(this.group.rotation, speed * 4, {
x: 0,
z: (Math.random() / 10) - 0.1,
ease: Elastic.easeOut
}, speed / 3);
this.openAnimation.to(this.head.position, speed * 3, {
y: 6,
ease: Elastic.easeOut
}, jackDelay)
// move animation
this.moveAnimation.clear();
this.moveAnimation.stop();
this.moveAnimation.to(this.group.rotation, 0.6, {
x: -0.5,
ease: Power3.easeIn
}, 0);
this.moveAnimation.to(this.group.rotation, speed * 2, {
x: 0,
ease: Elastic.easeOut
});
}
stopAnimations = function()
{
this.openAnimation.pause();
this.closeAnimation.pause();
this.moveAnimation.pause();
}
open = function(skipAnimation:boolean = false)
{
this.openAnimation.duration(skipAnimation ? 0.1 : 3);
this.openAnimation.restart();
}
close = function()
{
//this.stopAnimations();
//this.closeAnimation.duration(6);
this.closeAnimation.restart();
}
move = function(speed: number)
{
this.moveAnimation.restart();
}
}
class Dispenser
{
group: any;
private tube: any;
private lidLeft: any;
private lidRight: any;
private openAnimation = new TimelineMax();
constructor()
{
this.group = new THREE.Group();
var geometry = new THREE.BoxGeometry(10, 30, 10);
geometry.applyMatrix(new THREE.Matrix4().makeTranslation(0, 15, 0));
var material = new THREE.MeshToonMaterial({
color: '#eeeeee'
});
this.tube = new THREE.Mesh(geometry, material);
this.group.add(this.tube);
var geometry = new THREE.BoxGeometry(10, 1, 5);
geometry.applyMatrix(new THREE.Matrix4().makeTranslation(0, -0.5, 2.5));
var material = new THREE.MeshToonMaterial({
color: '#dddddd'
});
this.lidLeft = new THREE.Mesh(geometry, material);
this.group.add(this.lidLeft);
var geometry = new THREE.BoxGeometry(10, 1, 5);
geometry.applyMatrix(new THREE.Matrix4().makeTranslation(0, -0.5, -2.5));
var material = new THREE.MeshToonMaterial({
color: '#dddddd'
});
this.lidRight = new THREE.Mesh(geometry, material);
this.group.add(this.lidRight);
this.lidLeft.position.z = -5
this.lidLeft.rotation.x = Utils.degToRad(90)
this.lidRight.position.z = 5
this.lidRight.rotation.x = Utils.degToRad(-90)
this.setAnimation();
}
setAnimation = function()
{
this.openAnimation.clear();
this.openAnimation.stop();
this.openAnimation.fromTo(this.lidRight.rotation, 1, {x: Utils.degToRad(0)}, {x: Utils.degToRad(-90), ease: Elastic.easeOut}, 0);
this.openAnimation.fromTo(this.lidLeft.rotation, 1, {x: Utils.degToRad(0)}, {x: Utils.degToRad(90), ease: Elastic.easeOut}, 0);
this.openAnimation.fromTo(this.lidRight.rotation, 0.5,{x: Utils.degToRad(-90)}, {x: Utils.degToRad(0), ease: Power3.easeInOut}, 1.5);
this.openAnimation.fromTo(this.lidLeft.rotation, 0.5, {x: Utils.degToRad(90)}, {x: Utils.degToRad(0), ease: Power3.easeInOut}, 1.5);
}
open = function()
{
this.openAnimation.restart();
}
}
class App
{
private colorChoices: string[];
private face: any;
private stage: Stage;
private dispenser: Dispenser;
private jacks: JackInABox[];
private time: number = 4;
private count: number = 0;
private belt: any;
private beltTexture: any;
constructor(count:number = 5)
{
this.count = count;
this.jacks = [];
this.face = THREE.ImageUtils.loadTexture('');
//this.beltTexture = THREE.ImageUtils.loadTexture('');
this.colorChoices = [colors.blue];//, colors.green, colors.yellow];
this.stage = new Stage();
this.stage.floor.position.z = (Math.floor(this.count / 2) * 15) - 8;
this.stage.floor.position.y = -1.5;
this.dispenser = new Dispenser();
this.dispenser.group.position.y = 16;
this.dispenser.group.position.z = Math.floor(this.count / 2) * -15;
this.stage.add(this.dispenser.group);
var width = this.count * 20;
var depth = 30;
var geometry = new THREE.PlaneGeometry(depth, width);
var material = new THREE.MeshBasicMaterial({
color: '#777777',
shading: THREE.FlatShading,
//map: this.beltTexture
});
//this.beltTexture.repeat.y = 100;
//this.beltTexture.wrapS = this.beltTexture.wrapT = THREE.RepeatWrapping;
//this.beltTexture.repeat.set( depth / 8, width / 8 );
var plane = new THREE.Mesh(geometry, material);
plane.rotation.x = -Utils.degToRad(90); //1.57079633;
plane.position.y = -1;
plane.position.z = - width / 2;
this.belt = new THREE.Group();
this.belt.add(plane);
this.stage.add(this.belt)
this.belt.position.z = - Math.floor(this.count / 2) * -11;
this.tick();
while(this.jacks.length < this.count)
{
this.addJack(true);
}
this.move(true);
}
tick = function()
{
this.stage.render();
requestAnimationFrame(() => {this.tick()});
}
addJack = function(skipAnimation:boolean = false)
{
let jack = new Jack(this.colorChoices[Math.floor(Math.random() * this.colorChoices.length)], this.face);
let box = new Box();
let group = new THREE.Group();
group.position.z = Math.floor(this.count / 2) * -15;
box.add(jack.group);
group.add(box.group);
this.jacks.push({box: box, jack: jack, group: group});
this.stage.add(group);
box.drop(skipAnimation);
this.dispenser.open();
}
move = function(skipAnimation:boolean = false)
{
while(this.jacks.length >= this.count)
{
let removed = this.jacks.shift();
if(removed) this.stage.remove(removed.group)
}
for(let i = this.jacks.length - 1; i >= 0; i--)
{
if(this.jacks[i])
{
var box: Box = this.jacks[i].box;
var jack: Jack = this.jacks[i].jack;
var group: any = this.jacks[i].group;
if(!box.closed) this.jacks[i].jack.move(2.2);
if(i <= Math.floor(this.count / 2) && box.closed)
{
if(i != Math.floor(this.count / 2))
{
this.jacks[i].box.open(true);
this.jacks[i].jack.open(true);
}
else setTimeout(() =>
{
this.jacks[i].box.open();
this.jacks[i].jack.open();
}, 1000);
}
if(i == 0)
{
this.jacks[i].box.fall(skipAnimation);
}
TweenMax.to(group.position, skipAnimation ? 0 : 1, {z: (Math.floor(this.count/2) - i) * 15, ease:Power3.easeInOut});
//TweenMax.to(this.beltTexture.offset, skipAnimation ? 0 : 1, {y: '+=0.5', ease:Power3.easeInOut});
}
}
this.addJack();
setTimeout(() => {this.move()}, this.time * 1000);
}
}
let app = new App(5);
<script src="//codepen.io/steveg3003/pen/zBVakw"></script>
<script src="//cdnjs.cloudflare.com/ajax/libs/three.js/r83/three.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/gsap/1.19.1/TweenMax.min.js"></script>
html, body
{
margin: 0;
overflow: hidden;
height: 100%;
width: 100%;
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment