HTML
xxxxxxxxxx
1
<div id="content">
2
3
<!-- canas -->
4
<div id="paint"></div>
5
6
<!-- text -->
7
<div id="wrapper">
8
<div class="flex">
9
10
<div id="controls">
11
<button class="one">Business class</button>
12
<button class="two">Ahead of wing</button>
13
<button class="three">Over the wing</button>
14
<button class="four">Rear cabin</button>
15
<label class="option">360 view<input type="checkbox" id="myCheck"></label>
16
</div>
17
18
19
20
21
<section id="text">
22
<h1>Survival rates in commercial jet crashes, by seat position.</h1>
23
<p>According to an analysis of all commercial jet crashes in the United States since 1971, In 11 of the 20 crashes that involved both fatalities and survivors, <span class="high">rear passengers had a 69% chance of survival</span> than those compared to a less favorable <span>49%</span> in business class. </p>
24
<small>source: <br>
25
<a href="https://www.statista.com/chart/9805/which-plane-seats-are-safest/" target="_blank">Statista</a><br>
26
<a href="https://www.popularmechanics.com/flight/a1918/4219452/" target="_blank">Popular Mechanics</a></small>
27
</section>
28
29
</div>
30
</div><!-- /text -->
31
32
</div>
33
34
35
<!-- numbers -->
36
<canvas id="number" width="100%" height="100%"></canvas>
37
<canvas id="number2" width="100%" height="100%"></canvas>
38
<canvas id="number3" width="100%" height="100%"></canvas>
39
<canvas id="number4" width="100%" height="100%"></canvas>
40
41
42
<script src="https://threejs.org/build/three.min.js"></script>
43
<script src='https://threejs.org/examples/js/controls/OrbitControls.js'></script>
44
<script src="https://threejs.org/examples/js/loaders/deprecated/LegacyJSONLoader.js"></script>
45
<script src="https://threejs.org/examples/js/controls/DeviceOrientationControls.js"></script>
46
<script src="https://threejs.org/examples/js/effects/StereoEffect.js"></script>
47
<script src="https://threejs.org/examples/js/libs/tween.min.js"></script>
CSS
xxxxxxxxxx
1
@import url("https://fonts.googleapis.com/css?family=Roboto+Condensed|Playfair+Display:700i|Didact+Gothic");
2
html,
3
body {
4
background: black;
5
font-family: monospace;
6
font-family: "Roboto", sans-serif;
7
margin: 0;
8
padding: 0;
9
}
10
11
*,
12
*:before,
13
*:after {
14
box-sizing: inherit;
15
}
16
17
h1 {
18
font-family: "Roboto Condensed", "Helvetica Neue", Helvetica, Arial,
19
sans-serif;
20
font-size: 40px;
21
line-height: 1;
22
}
23
24
h3 {
25
font-weight: lighter;
26
}
27
28
section {
29
color: white;
30
padding: 50px 25px;
31
}
32
33
p {
34
color: rgba(255, 255, 255, 0.5);
35
}
36
37
a {
38
color: white;
39
}
40
41
span {
42
font-weight: bold;
43
font-size: 20px;
44
}
45
46
span.high {
47
color: #ff1493;
48
}
49
50
canvas[id*="number"] {
51
position: absolute;
52
z-index: -1;
53
}
54
55
button {
56
flex: 1;
57
}
58
59
button,
60
label {
61
background: none;
62
border: none;
63
text-align: center;
64
padding: 5px 10px;
65
margin: 5px;
66
border: 1px solid white;
67
color: white;
68
font-size: 13px;
69
}
70
71
#controls {
72
position: fixed;
73
width: 100%;
74
top: 300px;
75
left: 0;
76
background: black;
77
display: flex;
78
-webkit-display: flex;
79
}
80
81
#content {
82
color: white;
83
background: #151515;
84
}
85
86
#paint {
87
position: fixed;
88
top: 0;
89
left: 0;
90
width: 100%;
91
height: 300px;
92
margin: auto;
93
z-index: 100;
94
}
95
96
#wrapper {
97
position: relative;
98
top: 300px;
99
background: black;
100
}
101
102
@media (min-width: 1200px) {
103
section {
104
padding: 25px;
105
pointer-events: initial;
106
max-width: 275px;
107
width: 100%;
108
}
109
110
#paint {
111
width: 100%;
112
height: 100%;
113
margin: 0;
114
top: 0;
115
left: 0;
116
}
117
#wrapper {
118
width: 100%;
119
height: 100%;
120
z-index: 100;
121
top: 0;
122
background: none;
123
position: relative;
124
}
125
.flex {
126
display: flex;
127
-webkit-display: flex;
128
justify-content: space-between;
129
-webkit-justify-content: space-between;
130
align-items: flex-end;
131
-webkit-align-items: flex-end;
132
flex-direction: row-reverse;
133
-webkit-flex-direction: row-reverse;
134
position: fixed;
135
bottom: 0;
136
left: 0;
137
width: 100%;
138
pointer-events: none;
139
/*background: red;*/
140
}
141
#controls {
142
position: relative;
143
top: 0;
144
left: 0;
145
background: none;
146
pointer-events: initial;
147
width: auto;
148
}
149
}
150
JS
xxxxxxxxxx
1
//===================================================== canvas
2
var container = document.getElementById('paint');
3
var renderer = new THREE.WebGLRenderer();
4
container.appendChild(renderer.domElement);
5
renderer.setSize(container.getBoundingClientRect().width, container.getBoundingClientRect().height);
6
renderer.setClearColor(0x151515, 1);
7
8
9
10
//===================================================== camera
11
var scene = new THREE.Scene();
12
var camera = new THREE.PerspectiveCamera(45, container.getBoundingClientRect().width / container.getBoundingClientRect().height, 1, 10000);
13
14
15
16
//===================================================== resize
17
window.addEventListener("resize", function() {
18
let width = container.getBoundingClientRect().width;
19
let height = container.getBoundingClientRect().height;
20
renderer.setSize(width, height);
21
camera.aspect = width / height;
22
camera.updateProjectionMatrix();
23
});
24
25
26
//===================================================== camera
27
var plane = new THREE.GridHelper(150, 50);
28
plane.material.color = new THREE.Color('gray');
29
scene.add(plane);
30
31
32
33
34
//===================================================== controls
35
var controls = new THREE.OrbitControls(camera, renderer.domElement);
36
controls.enableDamping = true;
37
controls.dampingFactor = 0.25;
38
controls.enableZoom = true;
39
controls.maxPolarAngle = Math.PI / 2.1;
40
41
42
43
//===================================================== add Light
44
var light = new THREE.DirectionalLight(0xefefff, 1.5);
45
light.position.set(1, 1, 1).normalize();
46
scene.add(light);
47
48
var light = new THREE.DirectionalLight(0xffefef, 1.5);
49
light.position.set(-1, -1, -1).normalize();
50
scene.add(light);
51
52
53
//===================================================== add blue light on the seats
54
var light = new THREE.DirectionalLight(new THREE.Color("blue"), 1.5);
55
light.position.set(0, 200, 100);
56
light.castShadow = true;
57
light.shadow.camera.top = 280;
58
light.shadow.camera.bottom = -200;
59
light.shadow.camera.left = -200;
60
light.shadow.camera.right = 200;
61
scene.add(light);
62
63
64
65
66
//==================================================== Add obj
67
loader = new THREE.LegacyJSONLoader();
68
loader.load('https://raw.githubusercontent.com/baronwatts/models/master/airplane2.js', function(geometry, materials) {
69
70
var matt = new THREE.MeshPhongMaterial({
71
vertexColors: THREE.FaceColors,
72
side: THREE.DoubleSide,
73
wireframe: true,
74
transparent: true,
75
opacity: .085
76
});
77
var obj = new THREE.Mesh(geometry, matt);
78
obj.scale.set(.5, .5, .5);
79
obj.rotateY(-Math.PI);
80
obj.receiveShadow = true;
81
obj.castShadow = true;
82
scene.add(obj);
83
84
});
85
86
87
//===================================================== seating rows
88
var row = new THREE.Group();
89
row.rotateY(Math.PI);
90
row.position.y = 5;
91
row.position.z = -3.75;
92
row.translateX(20);
93
scene.add(row);
94
95
96
loader = new THREE.LegacyJSONLoader();
97
loader.load('https://raw.githubusercontent.com/baronwatts/models/master/seats.js', function(geometry, materials) {
98
99
var matt = new THREE.MeshLambertMaterial({
100
vertexColors: THREE.FaceColors,
101
color: new THREE.Color('#888'),
102
side: THREE.DoubleSide
103
});
104
105
for (var i = 0; i < 30; i++) {
106
for (var j = 0; j < 2; j++) {
107
108
var obj2 = new THREE.Mesh(geometry, matt);
109
obj2.scale.set(3, 3, 3);
110
obj2.rotateY(Math.PI / 2);
111
obj2.position.z = -3 * j;
112
obj2.position.x = -1.25 * i;
113
obj2.receiveShadow = true;
114
obj2.castShadow = true;
115
row.add(obj2);
116
117
}
118
}
119
120
});
121
122
123
124
//==================================================== number
125
var canvas = document.getElementById("number");
126
var ctx = canvas.getContext("2d");
127
var x = 32;
128
var y = 32;
129
var radius = 30;
130
var startAngle = 0;
131
var endAngle = Math.PI * 2;
132
133
ctx.fillStyle = "rgba(0, 0, 0, 0)";
134
ctx.beginPath();
135
ctx.arc(x, y, radius, startAngle, endAngle);
136
ctx.fill();
137
138
ctx.fillStyle = "#ccc";
139
ctx.font = "32px Roboto Condensed";
140
ctx.textAlign = "center";
141
ctx.textBaseline = "middle";
142
ctx.fillText("56%", x, y);
143
144
var numberTexture = new THREE.CanvasTexture(document.querySelector("#number"));
145
var spriteMaterial = new THREE.SpriteMaterial({
146
map: numberTexture,
147
alphaTest: 0.5,
148
transparent: true,
149
depthTest: false,
150
depthWrite: false
151
});
152
153
sprite = new THREE.Sprite(spriteMaterial);
154
sprite.position.set(0, 10, 0);
155
sprite.scale.set(2, 2, 2);
156
scene.add(sprite);
157
158
159
160
//==================================================== number
161
var canvas2 = document.getElementById("number2");
162
var ctx = canvas2.getContext("2d");
163
ctx.fillStyle = "#ccc";
164
ctx.font = "32px Roboto Condensed";
165
ctx.textAlign = "center";
166
ctx.textBaseline = "middle";
167
ctx.fillText("56%", x, y);
168
169
var numberTexture = new THREE.CanvasTexture(document.querySelector("#number2"));
170
var spriteMaterial = new THREE.SpriteMaterial({
171
map: numberTexture,
172
alphaTest: 0.5,
173
transparent: true,
174
depthTest: false,
175
depthWrite: false
176
});
177
178
sprite = new THREE.Sprite(spriteMaterial);
179
sprite.position.set(-10, 10, 0);
180
sprite.scale.set(2, 2, 2);
181
scene.add(sprite);
182
183
184
185
186
//==================================================== number
187
var canvas2 = document.getElementById("number4");
188
var ctx = canvas2.getContext("2d");
189
ctx.fillStyle = "gray";
190
ctx.font = "32px Roboto Condensed";
191
ctx.textAlign = "center";
192
ctx.textBaseline = "middle";
193
ctx.fillText("49%", x, y);
194
195
var numberTexture = new THREE.CanvasTexture(document.querySelector("#number4"));
196
var spriteMaterial = new THREE.SpriteMaterial({
197
map: numberTexture,
198
alphaTest: 0.5,
199
transparent: true,
200
depthTest: false,
201
depthWrite: false
202
});
203
204
sprite = new THREE.Sprite(spriteMaterial);
205
sprite.position.set(-18, 10, 0);
206
sprite.scale.set(2, 2, 2);
207
scene.add(sprite);
208
209
210
211
212
//==================================================== number
213
var canvas2 = document.getElementById("number3");
214
var ctx = canvas2.getContext("2d");
215
ctx.fillStyle = "#FF1493";
216
ctx.font = "32px Roboto Condensed";
217
ctx.textAlign = "center";
218
ctx.textBaseline = "middle";
219
ctx.fillText("69%", x, y);
220
221
var numberTexture = new THREE.CanvasTexture(document.querySelector("#number3"));
222
var spriteMaterial = new THREE.SpriteMaterial({
223
map: numberTexture,
224
alphaTest: 0.5,
225
transparent: true,
226
depthTest: false,
227
depthWrite: false
228
});
229
230
sprite = new THREE.Sprite(spriteMaterial);
231
sprite.position.set(15, 10, 0);
232
sprite.scale.set(2, 2, 2);
233
scene.add(sprite);
234
235
236
237
238
//===================================================== Circle path for the camera
239
var segmentCount = 10;
240
var radius = 50;
241
var xyzArray = new Array(segmentCount)
242
.fill(null)
243
.map(
244
(d, i) =>
245
new THREE.Vector3(
246
/*x*/
247
Math.cos(i / segmentCount * Math.PI * 2) * radius,
248
/*y*/
249
5,
250
/*z*/
251
Math.sin(i / segmentCount * Math.PI * 2) * radius
252
)
253
);
254
255
var curve = new THREE.CatmullRomCurve3(xyzArray);
256
curve.closed = true;
257
var g2 = new THREE.Geometry();
258
g2.vertices = curve.getPoints(50);
259
var m2 = new THREE.LineBasicMaterial({
260
color: new THREE.Color("skyblue")
261
});
262
var curveObject = new THREE.Line(g2, m2);
263
//scene.add(curveObject);
264
265
266
267
//===================================================== buttons
268
269
let button1 = document.querySelector('.one');
270
button1.addEventListener('click', function() {
271
zoom(-30, 10, 5);
272
});
273
274
let button2 = document.querySelector('.two');
275
button2.addEventListener('click', function() {
276
zoom(-20, 10, 20)
277
});
278
279
let button3 = document.querySelector('.three');
280
button3.addEventListener('click', function() {
281
zoom(5, 10, 30);
282
});
283
284
let button4 = document.querySelector('.four');
285
button4.addEventListener('click', function() {
286
zoom(30, 10, 5);
287
});
288
289
290
291
function zoom(x, y, z) {
292
var from = {
293
x: camera.position.x,
294
y: camera.position.y,
295
z: camera.position.z
296
};
297
var to = {
298
x: x,
299
y: y,
300
z: z
301
};
302
var tween = new TWEEN.Tween(from)
303
.to(to, 900)
304
.easing(TWEEN.Easing.Quadratic.Out)
305
.onUpdate(function() {
306
camera.position.set(this.x, this.y, this.z);
307
camera.lookAt(new THREE.Vector3(0, 0, 0));
308
})
309
.onComplete(function() {
310
camera.lookAt(new THREE.Vector3(0, 0, 0));
311
})
312
.start();
313
};
314
315
316
317
318
//===================================================== animate
319
320
spline = curve;
321
camPosIndex = 0;
322
clock = new THREE.Clock();
323
324
function animate() {
325
326
TWEEN.update();
327
328
camPosIndex++;
329
if (camPosIndex > 1000) {
330
camPosIndex = 0;
331
}
332
333
334
let camPos = spline.getPoint(camPosIndex / 1000); //move camera x,y,z
335
let camRot = spline.getTangent(camPosIndex / 1000); //rotates camera x,y,z
336
337
338
//move camera on path
339
if (document.getElementById("myCheck").checked == true) {
340
camera.position.set(camPos.x, camPos.y, camPos.z + 5);
341
camera.lookAt(new THREE.Vector3(0, 0, 0));
342
} else {
343
camera.position.set(camera.position.x, camera.position.y, camera.position.z);
344
camera.lookAt(new THREE.Vector3(0, 0, 0));
345
}
346
347
348
renderer.render(scene, camera);
349
requestAnimationFrame(animate);
350
351
}
352
353
animate();
354
zoom(1, 10, 40);
Console errors: 0