forked from peterqliu/threebox
-
Notifications
You must be signed in to change notification settings - Fork 148
/
11-animation.html
363 lines (303 loc) · 9.37 KB
/
11-animation.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
<!doctype html>
<head>
<title>Threebox animated soldier</title>
<script src="../dist/threebox.js" type="text/javascript"></script>
<link href="../dist/threebox.css" rel="stylesheet" />
<script src="config.js"></script>
<link href="https://api.mapbox.com/mapbox-gl-js/v2.2.0/mapbox-gl.css" rel="stylesheet">
<script src="https://api.mapbox.com/mapbox-gl-js/v2.2.0/mapbox-gl.js"></script>
<script src="https://d3js.org/d3.v5.min.js"></script>
<script src="plugins/jquery.min.js"></script>
<!-- Place your kit's code here -->
<script src="css/fontawesome.js"></script>
<link href="css/free.min.css" rel="stylesheet" />
<link href="css/free-v4-font-face-min.css" rel="stylesheet" />
<link href="css/free-v4-shims.min.css" rel="stylesheet" />
<style>
body, html {
width: 100%;
height: 100%;
margin: 0;
background: black;
}
#map {
width: 100%;
height: 100%;
}
#explainer {
z-index: 99;
}
.top-right-tools {
display: flex;
right: 40px;
z-index: 10;
}
.tools-i.mapboxgl-ctrl {
margin: 10px 20px 0 0;
display: inline-flex;
}
.helpDiv {
width: auto;
left: 50%;
top: 0px;
z-index: 2;
position: absolute;
}
.help {
background: rgba(0, 0, 0, 0.5);
color: #fff;
position: relative;
text-align: center;
top: 10px;
left: -50%;
padding: 5px 10px;
margin: 0;
font-size: 11px;
line-height: 18px;
border-radius: 3px;
z-index: 1;
display: block;
}
/*these 3 clases will provide mapbox-like style for labels*/
.toolTip {
border: 0.5px black solid;
display: inline-block;
background: white;
padding: 1px 6px;
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
font-size: 11px !important;
}
.marker {
max-width: 240px;
display: flex;
margin-bottom: 5em;
text-align: center;
}
.mapboxgl-popup-tip {
margin-top: -1px;
}
</style>
</head>
<body>
<div id='map' class='map'>
<div class="mapboxgl-ctrl-top-right top-right-tools" aria-label="Map Tools Control">
<div class="tools-i mapboxgl-ctrl mapboxgl-ctrl-group">
<button type="button" name="wireButton" id="wireButton" title="Wireframe/Texture selected item [W]" alt="Wireframe/Texture selected item [W]" disabled><i class="fas fa-border-none"></i></button>
<button type="button" name="playButton" id="playButton" title="Play/Pause animations" alt="Play/Pause animations" disabled><i class="fas fa-play"></i></button>
</div>
</div>
<div class="mapboxgl-ctrl helpDiv">
<div id="help" class="help">
Add or Select an object
</div>
</div>
</div>
<div id='explainer'>Select the soldier, drag, rotate, wireframe and play animation</div>
<script type="module">
// This example downloads a soldier model from an external OBJ/MTL file, adds it to the map, and drives it around via paths fetched from the Mapbox Directions API
if (!config) console.error("Config not set! Make a copy of 'config_template.js', add in your access token, and save the file as 'config.js'.");
mapboxgl.accessToken = config.accessToken;
var origin = [-122.47920912, 37.716351775];
var destination, line;
var soldier;
var map = new mapboxgl.Map({
container: 'map',
style: 'mapbox://styles/mapbox/outdoors-v11',
center: origin,
zoom: 18,
pitch: 60,
antialias: true,
bearing: 0
});
map.addControl(new mapboxgl.NavigationControl());
let stats;
import Stats from 'https://threejs.org/examples/jsm/libs/stats.module.js';
function animate() {
requestAnimationFrame(animate);
stats.update();
}
map.on('style.load', function () {
// stats
stats = new Stats();
map.getContainer().appendChild(stats.dom);
animate();
map.addLayer({
id: 'custom_layer',
type: 'custom',
renderingMode: '3d',
onAdd: function (map, mbxContext) {
window.tb = new Threebox(
map,
mbxContext,
{
defaultLights: true,
enableSelectingFeatures: true,
enableSelectingObjects: true,
enableDraggingObjects: true,
enableRotatingObjects: true,
enableTooltips: true
}
);
// import soldier from an external glb file, scaling up its size 20x
// IMPORTANT: .glb is not a standard MIME TYPE, you'll have to add it to your web server config,
// otherwise you'll receive a 404 error
// Attribution: Soldier animated model by T. Choonyung at https://www.mixamo.com
// from https://www.mixamo.com/#/?page=1&query=vanguard&type=Character
var options = {
obj: 'models/soldier.glb',
type: 'gltf',
scale: 20,
units: 'meters',
rotation: { x: 90, y: 0, z: 0 },
anchor: 'center'//default rotation
}
tb.loadObj(options, function (model) {
soldier = model.setCoords(origin);
// Listening to the events
soldier.addEventListener('SelectedChange', onSelectedChange, false);
soldier.addEventListener('Wireframed', onWireframed, false);
soldier.addEventListener('IsPlayingChanged', onIsPlayingChanged, false);
soldier.addEventListener('ObjectDragged', onDraggedObject, false);
soldier.addEventListener('ObjectMouseOver', onObjectMouseOver, false);
soldier.addEventListener('ObjectMouseOut', onObjectMouseOut, false);
soldier.addEventListener('ObjectChanged', onObjectChanged, false);
tb.add(soldier);
})
},
render: function (gl, matrix) {
tb.update();
}
});
}).on('click', function (e) {
var pt = [e.lngLat.lng, e.lngLat.lat];
travelPath(pt);
})
function travelPath(destination) {
// request directions. See https://docs.mapbox.com/api/navigation/#directions for details
var url = "https://api.mapbox.com/directions/v5/mapbox/driving/" + [origin, destination].join(';') + "?geometries=geojson&access_token=" + config.accessToken
fetchFunction(url, function (data) {
let duration = 10000;
// extract path geometry from callback geojson, and set duration of travel
var options = {
animation: 1,
path: data.routes[0].geometry.coordinates,
duration: duration
}
// start the soldier animation with above options, and remove the line when animation ends
soldier.followPath(
options,
function () {
tb.remove(line);
}
);
soldier.playAnimation(options);
// set up geometry for a line to be added to map, lofting it up a bit for *style*
var lineGeometry = options.path
.map(function (coordinate) {
return coordinate.concat([15])
})
// create and add line object
line = tb.line({
geometry: lineGeometry,
width: 5,
color: 'steelblue'
})
tb.add(line);
// set destination as the new origin, for the next trip
origin = destination;
})
}
//convenience function for fetch
function fetchFunction(url, cb) {
fetch(url)
.then(
function (response) {
if (response.status === 200) {
response.json()
.then(function (data) {
cb(data)
})
}
}
)
}
let selectedObject;
//actions to execute onSelectedChange
function onSelectedChange(e) {
let selected = e.detail.selected;
console.log("onSelectedChange: " + selected);
$('#wireButton')[0].disabled = !selected;
if (e.detail.hasDefaultAnimation) { $('#playButton')[0].disabled = !selected; }
if (selected) {
selectedObject = e.detail;
$('#help')[0].innerHTML = "Press 'Shift' to drag or ress 'Alt' to rotate <br>\ Tap 'Wireframe' or 'Play' buttons ";
}
else {
$('#help')[0].innerHTML = "Add or Select an object";
}
tb.update();
map.repaint = true;
}
//actions to execute onWireFramed
function onWireframed(e) {
if (e.detail.wireframe) {
$('#wireButton').children(".fas").removeClass("fa-border-none").addClass("fa-border-all");
}
else {
$('#wireButton').children(".fas").removeClass("fa-border-all").addClass("fa-border-none");
}
console.log("onWireframed: " + e.detail.wireframe);
tb.update();
map.repaint = true;
}
//actions to execute onIsPlayingChanged
function onIsPlayingChanged(e) {
if (e.detail.isPlaying) {
$('#playButton').children(".fas").removeClass("fa-play").addClass("fa-pause");
}
else {
$('#playButton').children(".fas").removeClass("fa-pause").addClass("fa-play");
}
}
//actions to execute onDraggedObject
function onDraggedObject(e) {
let draggedObject = e.detail.draggedObject;
let draggedAction = e.detail.draggedAction;
console.log("onDraggedObject" + draggedAction);
}
//actions to execute onObjectMouseOver
function onObjectMouseOver(e) {
console.log("onObjectMouseOver");
}
//actions to execute onObjectMouseOut
function onObjectMouseOut(e) {
console.log("onObjectMouseOut");
}
function onObjectChanged(e) {
let model = e.detail.object; //here's the object already modified
let action = e.detail.action; //here's the action that changed the object
console.log(action);
}
//wire / unwire object
$('#wireButton').on('click', function () {
if (selectedObject) {
if (selectedObject.wireframe) {
selectedObject.wireframe = false;
} else {
selectedObject.wireframe = true;
}
}
});
//play/pause default animation
$('#playButton').on('click', function () {
if (selectedObject) {
if (selectedObject.isPlaying) {
selectedObject.stop();
} else {
// play animation 3, for 10 seconds
selectedObject.playAnimation( { animation: 1, duration: 10000 });
}
}
});
</script>
</body>