-
Notifications
You must be signed in to change notification settings - Fork 302
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for other data types in potree pointclouds #767
Changes from all commits
a250f90
b319777
79b08d8
e2f7e92
276c504
d2e1f55
93b3e27
bb7e431
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -3,7 +3,7 @@ import Fetcher from './Fetcher'; | |
import PointCloudProcessing from '../Process/PointCloudProcessing'; | ||
import PotreeBinParser from '../Parser/PotreeBinParser'; | ||
import PotreeCinParser from '../Parser/PotreeCinParser'; | ||
import PointsMaterial from '../Renderer/PointsMaterial'; | ||
import PointsMaterial, { MODE } from '../Renderer/PointsMaterial'; | ||
import Picking from '../Core/Picking'; | ||
import Extent from '../Core/Geographic/Extent'; | ||
|
||
|
@@ -142,6 +142,60 @@ function addPickingAttribute(points) { | |
return points; | ||
} | ||
|
||
function computeBbox(layer) { | ||
let bbox; | ||
if (layer.isFromPotreeConverter) { | ||
bbox = new THREE.Box3( | ||
new THREE.Vector3(layer.metadata.boundingBox.lx, layer.metadata.boundingBox.ly, layer.metadata.boundingBox.lz), | ||
new THREE.Vector3(layer.metadata.boundingBox.ux, layer.metadata.boundingBox.uy, layer.metadata.boundingBox.uz)); | ||
} else { | ||
// lopocs | ||
let idx = 0; | ||
for (const entry of layer.metadata) { | ||
if (entry.table == layer.table) { | ||
break; | ||
} | ||
idx++; | ||
} | ||
bbox = new THREE.Box3( | ||
new THREE.Vector3(layer.metadata[idx].bbox.xmin, layer.metadata[idx].bbox.ymin, layer.metadata[idx].bbox.zmin), | ||
new THREE.Vector3(layer.metadata[idx].bbox.xmax, layer.metadata[idx].bbox.ymax, layer.metadata[idx].bbox.zmax)); | ||
} | ||
return bbox; | ||
} | ||
|
||
function parseMetadata(metadata, layer) { | ||
layer.metadata = metadata; | ||
|
||
var customBinFormat = true; | ||
|
||
// Lopocs pointcloud server can expose the same file structure as PotreeConverter output. | ||
// The only difference is the metadata root file (cloud.js vs infos/sources), and we can | ||
// check for the existence of a `scale` field. | ||
// (if `scale` is defined => we're fetching files from PotreeConverter) | ||
if (layer.metadata.scale != undefined) { | ||
layer.isFromPotreeConverter = true; | ||
// PotreeConverter format | ||
customBinFormat = layer.metadata.pointAttributes === 'CIN'; | ||
// do we have normal information | ||
const normal = Array.isArray(layer.metadata.pointAttributes) && | ||
layer.metadata.pointAttributes.find(elem => elem.startsWith('NORMAL')); | ||
if (normal) { | ||
layer.material.defines[normal] = 1; | ||
} | ||
} else { | ||
// Lopocs | ||
layer.metadata.scale = 1; | ||
layer.metadata.octreeDir = `itowns/${layer.table}.points`; | ||
layer.metadata.hierarchyStepSize = 1000000; // ignore this with lopocs | ||
customBinFormat = true; | ||
} | ||
|
||
layer.parse = customBinFormat ? PotreeCinParser.parse : PotreeBinParser.parse; | ||
layer.extension = customBinFormat ? 'cin' : 'bin'; | ||
layer.supportsProgressiveDisplay = customBinFormat; | ||
} | ||
|
||
export default { | ||
preprocessDataLayer(layer, view) { | ||
if (!layer.file) { | ||
|
@@ -169,6 +223,8 @@ export default { | |
layer.type = 'geometry'; | ||
layer.material = layer.material || {}; | ||
layer.material = layer.material.isMaterial ? layer.material : new PointsMaterial(layer.material); | ||
layer.material.defines = layer.material.defines || {}; | ||
layer.mode = MODE.COLOR; | ||
|
||
// default update methods | ||
layer.preUpdate = PointCloudProcessing.preUpdate; | ||
|
@@ -178,58 +234,21 @@ export default { | |
// this probably needs to be moved to somewhere else | ||
layer.pickObjectsAt = (view, mouse, radius) => Picking.pickPointsAt(view, mouse, radius, layer); | ||
|
||
return Fetcher.json(`${layer.url}/${layer.file}`, layer.fetchOptions).then((cloud) => { | ||
layer.metadata = cloud; | ||
|
||
let bbox; | ||
var customBinFormat = true; | ||
|
||
// Lopocs pointcloud server can expose the same file structure as PotreeConverter output. | ||
// The only difference is the metadata root file (cloud.js vs infos/sources), and we can | ||
// check for the existence of a `scale` field. | ||
// (if `scale` is defined => we're fetching files from PotreeConverter) | ||
if (layer.metadata.scale != undefined) { | ||
// PotreeConverter format | ||
customBinFormat = layer.metadata.pointAttributes === 'CIN'; | ||
bbox = new THREE.Box3( | ||
new THREE.Vector3(cloud.boundingBox.lx, cloud.boundingBox.ly, cloud.boundingBox.lz), | ||
new THREE.Vector3(cloud.boundingBox.ux, cloud.boundingBox.uy, cloud.boundingBox.uz)); | ||
} else { | ||
// Lopocs | ||
layer.metadata.scale = 1; | ||
layer.metadata.octreeDir = `itowns/${layer.table}.points`; | ||
layer.metadata.hierarchyStepSize = 1000000; // ignore this with lopocs | ||
customBinFormat = true; | ||
|
||
let idx = 0; | ||
for (const entry of cloud) { | ||
if (entry.table == layer.table) { | ||
break; | ||
} | ||
idx++; | ||
} | ||
bbox = new THREE.Box3( | ||
new THREE.Vector3(cloud[idx].bbox.xmin, cloud[idx].bbox.ymin, cloud[idx].bbox.zmin), | ||
new THREE.Vector3(cloud[idx].bbox.xmax, cloud[idx].bbox.ymax, cloud[idx].bbox.zmax)); | ||
} | ||
|
||
layer.parse = customBinFormat ? PotreeCinParser.parse : PotreeBinParser.parse; | ||
layer.extension = customBinFormat ? 'cin' : 'bin'; | ||
layer.supportsProgressiveDisplay = customBinFormat; | ||
|
||
return parseOctree( | ||
layer, | ||
layer.metadata.hierarchyStepSize, | ||
{ baseurl: `${layer.url}/${cloud.octreeDir}/r`, name: '', bbox }); | ||
}).then((root) => { | ||
// eslint-disable-next-line no-console | ||
console.log('LAYER metadata:', root); | ||
layer.root = root; | ||
root.findChildrenByName = findChildrenByName.bind(root, root); | ||
layer.extent = Extent.fromBox3(view.referenceCrs, root.bbox); | ||
|
||
return layer; | ||
}); | ||
return Fetcher.json(`${layer.url}/${layer.file}`, layer.fetchOptions) | ||
.then((metadata) => { | ||
parseMetadata(metadata, layer); | ||
const bbox = computeBbox(layer); | ||
return parseOctree(layer, layer.metadata.hierarchyStepSize, { baseurl: `${layer.url}/${layer.metadata.octreeDir}/r`, name: '', bbox }); | ||
}) | ||
.then((root) => { | ||
// eslint-disable-next-line no-console | ||
console.log('LAYER metadata:', root); | ||
layer.root = root; | ||
root.findChildrenByName = findChildrenByName.bind(root, root); | ||
layer.extent = Extent.fromBox3(view.referenceCrs, root.bbox); | ||
|
||
return layer; | ||
}); | ||
}, | ||
|
||
executeCommand(command) { | ||
|
@@ -247,7 +266,7 @@ export default { | |
// when we request .hrc files) | ||
const url = `${node.baseurl}/r${node.name}.${layer.extension}?isleaf=${command.isLeaf ? 1 : 0}`; | ||
|
||
return Fetcher.arrayBuffer(url, layer.fetchOptions).then(layer.parse).then((geometry) => { | ||
return Fetcher.arrayBuffer(url, layer.fetchOptions).then(buffer => layer.parse(buffer, layer.metadata.pointAttributes)).then((geometry) => { | ||
const points = new THREE.Points(geometry, layer.material.clone()); | ||
addPickingAttribute(points); | ||
points.frustumCulled = false; | ||
|
@@ -263,3 +282,7 @@ export default { | |
}); | ||
}, | ||
}; | ||
|
||
export const _testing = { | ||
parseMetadata, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm not okay with this, please move it to the default export, even if I understand that this is for testing purpose only. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Thanks for your review. Can you elaborate on the reasons you don't like this? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. To elaborate on my position, I'm pretty against exposing this to the default export, because it's a private method that has no reason to be exposed. Once you expose such a method, it becomes part of the public api, which means we have an implicit commitment to it. I know that this way of thinking is not very common on this project, but I'd like not to introduce api breakage when we can avoid it. For me There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't really know, maybe in fact it is a good idea. We could agree on a special exported namespace we could use to test internal methods. So in fact maybe keep it and find a more suitable name ? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Providers are not in the public API so no problems here, but see my change of pov above ;) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Just a quick note on this: I'm afraid they are. Everything importable from a ES6 configuration is part of the public API and we should treat it that way. Every transpiled file in
I don't know, There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ok I see one possible solution: surround testing exports with There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. (and then rename There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. (sorry for being so undecided) yeah |
||
}; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe you should add a comment to explain why you need this? (AFAIU it's because
Float16Array
doesn't exist so you emulate it)