-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy paththree.geojson.js
295 lines (204 loc) · 7.1 KB
/
three.geojson.js
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
//Globals
var json, camera, scene, renderer, mesh, group, groupGeometry, mouse,
//All zero or false height values will be ignored
fast = false,
width = window.innerWidth,
height = window.innerHeight;
$(document).ready(function() {
log("start loading");
$.getJSON( jsonFile, function( data ) {
log("loading complete");
json = data;
init();
});
});
function log(m){
if(log){
console.log(m);
}
}
function init() {
//Initiate THREE.js
camera = new THREE.PerspectiveCamera( 50, (width/height), 1, 1000 );
camera.position.set( 0, 100, 500 );
scene = new THREE.Scene();
mouse = new THREE.Vector2();
//Initiate Renderer
renderer = new THREE.WebGLRenderer( { antialias: false, preserveDrawingBuffer: true, alpha: true } );
renderer.setSize( width, height );
renderer.shadowMapEnabled = true;
renderer.shadowMapType = THREE.PCFSoftShadowMap;
renderer.setViewport( 0,0,width, height );
//This group will hold our objects for easier handling
group = new THREE.Object3D();
group.position.y = 50;
group.position.z = 0;
scene.add( group );
//This group will hold all geometries
groupGeometry = new THREE.Geometry();
log("initiation done");
//Lets go and build the objects based on the geoJson data
buildShape();
}
//Due to javascripts limitations we need to parse the data in subsets (5000)
var shapeCount = 0, shapes = [], subset_size = 5000;
function buildShape(){
log("buildShape ("+shapeCount+"/"+json.features.length+")");
if(shapeCount<json.features.length){
var shapeSession = 0;
for(var s = shapeCount; s < json.features.length && shapeSession < subset_size; s++){
shapeSession++;
shapeCount++;
var good = true;
var points = [];
//Check if the geometry has at least two coordinates
if(json.features[s].geometry.coordinates.length<1 || json.features[s].geometry.coordinates[0]<1){
good = false;
}else{
for(var i = 0; i<json.features[s].geometry.coordinates[0].length; i++){
//Check for weird values
if(json.features[s].geometry.coordinates[0][i][0] && json.features[s].geometry.coordinates[0][i][1] && json.features[s].geometry.coordinates[0][i][0]>0 && json.features[s].geometry.coordinates[0][i][1]>0){
points.push( new THREE.Vector2 ( translateLat(json.features[s].geometry.coordinates[0][i][0]), translateLng(json.features[s].geometry.coordinates[0][i][1])) );
}else{
good = false;
}
}
}
//If the geometry is safe, continue
if(good){
//Calculate the height of the current geometry for extrusion
var h = heightFn(json.features[s].properties[heightAttr]);
if(isNaN(parseFloat(json.features[s].properties[heightAttr]))){
if(fast){
good = false;
}
h = 0;
}
if(!h || h < 0){
if(fast){
good = false;
}
h = 0;
}
if(h>max){
h = max;
}
//Remove all objects that have no height information for faster rendering
if(h==0 && fast){
good = false;
}
}
//If the geometry is still safe, continue
if(good){
//Calculate the third dimension
var z = ((h/max)*z_max);
if(!z || z<1){z = 0;}
//Calculate the color of the object
//In this sample code we use a blue to red range to visualize the height of the object (blue short to red tall)
var red = Math.round((h/max)*255.0);
var blue = Math.round(255.0-(h/max)*255.0);
var color = new THREE.Color("rgb("+red+",0,"+blue+")");
addShape( new THREE.Shape( points ), z*z_rel, color, 0, 50, 0, r, 0, 0, 1 );
}
}
//If we have more geometries to add restart the whole loop
setTimeout(function(){ buildShape(); }, 100);
}else{
//We are done building our geometry
log("Geometry Done");
//Initiate the shader
var shaderMaterial = new THREE.ShaderMaterial({
attributes: {},
uniforms: {},
vertexShader: THREETUT.Shaders.Lit.vertex,
fragmentShader: THREETUT.Shaders.Lit.fragment
,side: THREE.FrontSide
});
//Initiate Material
var materials = [
new THREE.MeshLambertMaterial({vertexColors: THREE.VertexColors, color: "rgb(0.2,0.2,0.2)",ambient: "rgb(0.2,0.2,0.2)", shininess: 1, lights:true}),
new THREE.MeshLambertMaterial({vertexColors: THREE.VertexColors, color: "rgb(0.5,0.5,0.5)",ambient: "rgb(0.5,0.5,0.5)", shininess: 1, lights:true})
];
var material = new THREE.MeshFaceMaterial(materials);
//Create a mesh from the geometry
mesh = new THREE.Mesh( groupGeometry, material );
mesh.position.set( offset_x*3, offset_y*3, offset_z*3 );
mesh.rotation.set( r, 0, 0 );
mesh.scale.set(scale_factor * scale_x,scale_factor * scale_y,0);
mesh.castShadow = true;
mesh.receiveShadow = true;
scene.add( mesh );
//Too make it a little more fancy, add a directional light
var directionalLight = new THREE.DirectionalLight(0xeeeeee, 1);
directionalLight.position.set(0, 400, 200);
directionalLight.target = mesh;
directionalLight.castShadow = true;
directionalLight.shadowDarkness = 0.5;
scene.add( directionalLight );
//Now add the renderer to the DOM
document.body.appendChild( renderer.domElement );
//And start animating it
log("animate");
animate();
//For rotating the 3D object we use the mouse movement
renderer.domElement.addEventListener( 'mousemove', onDocumentMouseMove, false );
}
}
//Adding geometries to group
function addShape( shape, extrude, color, x, y, z, rx, ry, rz, s ) {
//Extrusion settings
var extrudeSettings = {
amount : extrude*50,
steps : 1,
material : 0,
extrudeMaterial : 1,
bevelEnabled : false
};
//Create the geometry
var geometry = new THREE.ExtrudeGeometry(shape, extrudeSettings);
//Set the color for the object
for(var f = 0; f<geometry.faces.length; f++){
geometry.faces[f].color.setRGB(color.r, color.g, color.b);
}
//Have a big amount of geometries will slow down THREE.js
//Instead we merge all geometries into one geometry
groupGeometry.merge( geometry, geometry.matrix );
}
//Store the current mouse position in the mouse-object
function onDocumentMouseMove( event ) {
event.preventDefault();
mouse.x = ( event.clientX / window.innerWidth ) * Math.PI*4;
mouse.y = ( event.clientY / window.innerHeight ) * Math.PI*4;
}
//Apply the mouse position on x/y rotation
function animate() {
//Animate at 30fs framerate
setTimeout( function() {
requestAnimationFrame( animate );
}, 1000/30 );
//Animate and render the mesh
mesh.rotation.x = mouse.y;
mesh.rotation.y = mouse.x;
//Animate the height of the objections
if(animateHeight){
heightScaler += 0.001;
}
mesh.scale.set(scale_factor * scale_x,scale_factor * scale_y,heightScaler);
//Render the scene
renderer.render( scene, camera );
}
//This is an experimental feature that allows us to save the generated mesh in an JSON-file, so we can skip the parsing and directly load the mesh
//So far the loading is not really working as expected
function saveObj(){
var j = JSON.stringify(mesh.toJSON());
var fcount = 100;
for(var f = 0; f<fcount; f++){
var ff = f;
if(f<10){ff = "0"+f;}
$.ajax({
type: "POST",
url: "save.php",
data: {content: j.substring((j.length/fcount)*f,(j.length/fcount)*(f+1)), name:"model_"+ff+".json"}
});
}
}