2018-01-02 17:38:15 +00:00
|
|
|
import ecto, os
|
2015-11-30 15:53:44 +00:00
|
|
|
|
|
|
|
from opendm import log
|
|
|
|
from opendm import io
|
|
|
|
from opendm import system
|
|
|
|
from opendm import context
|
2018-06-10 18:57:16 +00:00
|
|
|
from opendm import mesh
|
2015-11-30 15:53:44 +00:00
|
|
|
|
2015-12-10 11:01:41 +00:00
|
|
|
|
2016-02-26 18:50:12 +00:00
|
|
|
class ODMeshingCell(ecto.Cell):
|
2015-12-10 11:01:41 +00:00
|
|
|
def declare_params(self, params):
|
|
|
|
params.declare("max_vertex", 'The maximum vertex count of the output '
|
2016-02-26 18:50:12 +00:00
|
|
|
'mesh', 100000)
|
2015-12-10 11:01:41 +00:00
|
|
|
params.declare("oct_tree", 'Oct-tree depth used in the mesh reconstruction, '
|
2016-02-26 18:50:12 +00:00
|
|
|
'increase to get more vertices, recommended '
|
|
|
|
'values are 8-12', 9)
|
2015-12-10 11:01:41 +00:00
|
|
|
params.declare("samples", 'Number of points per octree node, recommended '
|
2016-02-26 18:50:12 +00:00
|
|
|
'value: 1.0', 1)
|
2015-12-10 11:01:41 +00:00
|
|
|
params.declare("solver", 'Oct-tree depth at which the Laplacian equation '
|
2016-02-26 18:50:12 +00:00
|
|
|
'is solved in the surface reconstruction step. '
|
|
|
|
'Increasing this value increases computation '
|
|
|
|
'times slightly but helps reduce memory usage.', 9)
|
2016-12-11 22:16:11 +00:00
|
|
|
params.declare("verbose", 'print additional messages to console', False)
|
2015-11-30 15:53:44 +00:00
|
|
|
|
|
|
|
def declare_io(self, params, inputs, outputs):
|
2015-12-10 11:01:41 +00:00
|
|
|
inputs.declare("tree", "Struct with paths", [])
|
2015-11-30 15:53:44 +00:00
|
|
|
inputs.declare("args", "The application arguments.", {})
|
2015-12-10 11:01:41 +00:00
|
|
|
inputs.declare("reconstruction", "Clusters output. list of ODMReconstructions", [])
|
|
|
|
outputs.declare("reconstruction", "Clusters output. list of ODMReconstructions", [])
|
2015-11-30 15:53:44 +00:00
|
|
|
|
|
|
|
def process(self, inputs, outputs):
|
2016-02-26 18:50:12 +00:00
|
|
|
|
2016-02-29 14:45:00 +00:00
|
|
|
# Benchmarking
|
|
|
|
start_time = system.now_raw()
|
|
|
|
|
2016-03-08 18:26:58 +00:00
|
|
|
log.ODM_INFO('Running ODM Meshing Cell')
|
2015-11-30 15:53:44 +00:00
|
|
|
|
|
|
|
# get inputs
|
2018-01-26 19:38:26 +00:00
|
|
|
args = inputs.args
|
|
|
|
tree = inputs.tree
|
|
|
|
reconstruction = inputs.reconstruction
|
2016-12-11 22:16:11 +00:00
|
|
|
verbose = '-verbose' if self.params.verbose else ''
|
2015-11-30 15:53:44 +00:00
|
|
|
|
|
|
|
# define paths and create working directories
|
2015-12-10 11:01:41 +00:00
|
|
|
system.mkdir_p(tree.odm_meshing)
|
2015-11-30 15:53:44 +00:00
|
|
|
|
2015-12-01 16:52:18 +00:00
|
|
|
# check if we rerun cell or not
|
2016-03-08 18:26:58 +00:00
|
|
|
rerun_cell = (args.rerun is not None and
|
|
|
|
args.rerun == 'odm_meshing') or \
|
|
|
|
(args.rerun_all) or \
|
|
|
|
(args.rerun_from is not None and
|
|
|
|
'odm_meshing' in args.rerun_from)
|
2015-12-10 11:01:41 +00:00
|
|
|
|
2018-06-27 18:32:49 +00:00
|
|
|
infile = tree.smvs_model
|
|
|
|
if args.use_opensfm_dense:
|
2018-06-30 23:54:05 +00:00
|
|
|
infile = tree.opensfm_model
|
2018-01-02 17:38:15 +00:00
|
|
|
elif args.fast_orthophoto:
|
|
|
|
infile = os.path.join(tree.opensfm, 'reconstruction.ply')
|
2017-04-05 17:56:48 +00:00
|
|
|
|
2018-07-03 16:37:39 +00:00
|
|
|
# Create full 3D model unless --skip-3dmodel is set
|
|
|
|
if not args.skip_3dmodel:
|
2018-01-02 17:38:15 +00:00
|
|
|
if not io.file_exists(tree.odm_mesh) or rerun_cell:
|
|
|
|
log.ODM_DEBUG('Writing ODM Mesh file in: %s' % tree.odm_mesh)
|
|
|
|
|
2018-07-03 15:24:47 +00:00
|
|
|
mesh.screened_poisson_reconstruction(infile,
|
|
|
|
tree.odm_mesh,
|
|
|
|
depth=self.params.oct_tree,
|
|
|
|
samples=self.params.samples,
|
|
|
|
maxVertexCount=self.params.max_vertex,
|
|
|
|
verbose=verbose)
|
|
|
|
|
2018-01-02 17:38:15 +00:00
|
|
|
else:
|
|
|
|
log.ODM_WARNING('Found a valid ODM Mesh file in: %s' %
|
|
|
|
tree.odm_mesh)
|
2016-02-26 18:50:12 +00:00
|
|
|
|
2018-07-03 16:37:39 +00:00
|
|
|
# Always generate a 2.5D mesh
|
|
|
|
# unless --use-3dmesh is set.
|
|
|
|
if not args.use_3dmesh:
|
2017-04-05 17:56:48 +00:00
|
|
|
if not io.file_exists(tree.odm_25dmesh) or rerun_cell:
|
2018-06-10 18:57:16 +00:00
|
|
|
|
2017-04-05 17:56:48 +00:00
|
|
|
log.ODM_DEBUG('Writing ODM 2.5D Mesh file in: %s' % tree.odm_25dmesh)
|
2018-07-04 18:39:21 +00:00
|
|
|
dsm_resolution = 1.0 / float(args.orthophoto_resolution)
|
|
|
|
|
|
|
|
# Sparse point clouds benefits from using
|
|
|
|
# a larger resolution value (more radius interolation, less holes)
|
|
|
|
if args.fast_orthophoto:
|
|
|
|
dsm_resolution *= 2
|
2017-04-05 17:56:48 +00:00
|
|
|
|
2018-07-02 19:21:30 +00:00
|
|
|
mesh.create_25dmesh(infile, tree.odm_25dmesh,
|
2018-07-04 18:39:21 +00:00
|
|
|
dsm_resolution=dsm_resolution,
|
2018-07-02 19:21:30 +00:00
|
|
|
depth=self.params.oct_tree,
|
2018-07-03 15:24:47 +00:00
|
|
|
maxVertexCount=self.params.max_vertex,
|
2018-07-03 17:01:18 +00:00
|
|
|
verbose=self.params.verbose,
|
|
|
|
max_workers=args.max_concurrency)
|
2017-04-05 17:56:48 +00:00
|
|
|
else:
|
|
|
|
log.ODM_WARNING('Found a valid ODM 2.5D Mesh file in: %s' %
|
|
|
|
tree.odm_25dmesh)
|
|
|
|
|
2018-01-26 19:38:26 +00:00
|
|
|
outputs.reconstruction = reconstruction
|
|
|
|
|
2016-03-08 18:26:58 +00:00
|
|
|
if args.time:
|
2016-02-29 14:45:00 +00:00
|
|
|
system.benchmark(start_time, tree.benchmarking, 'Meshing')
|
|
|
|
|
2016-03-08 18:26:58 +00:00
|
|
|
log.ODM_INFO('Running ODM Meshing Cell - Finished')
|
|
|
|
return ecto.OK if args.end_with != 'odm_meshing' else ecto.QUIT
|