lidR v2.2.5 (Release date: 2020-05-07)
ENHANCEMENTS
- Clear unrelevant message about OpenMP support when using the LAScatalog processing engine with a version of
lidRnot compiled with OpenMP support (i.e. on MacOS)
lidR v2.2.4 (Release date: 2020-04-24)
FIXES
-
Fix segfault on Windows 64 bits when constructing a proj4 from some specific modern WTK strings using
doCheckCRSArgs = FALSE. #323 sp #75 -
Fix wrong gpstime matching in
lasrangecorrection()at the edge of flightlines #327. -
Fix error when building the clusters with a partial processing and a realignment #332.
-
Fix error in
lasclip()andlasmergespatial()withsfobjects when the coordinates are not stored in a column namedgeometry. Thank to Michael Koontz in #335. -
lasrangecorrection()no longer mess-up the original sensor data. See #336
ENHANCEMENTS
-
Enhancements made here and there to improve the support of the CRS when reading and checking a LAS file.
-
crs not foundmessage is no longer displayed when building a LAS object. This message appeared with an update ofrgdalorsp. It is now gone. -
sensor_tracking()now throws an error for the invalid case reported in #327 -
lascheck()now reports problems for invalid data reported in #327 -
grid_metrics()returns a raster full of NAs instead of failing if aRasterLayeris given as a layout but this layer does not encompase the point cloud -
opt_output_file()now applies tilde-expansion to the path. -
When processing by file with an raster output, automatic chunk extension to match with a raster resolution now perform a tighter extension.
lidR v2.2.3 (Release date: 2020-03-02)
FIXES
-
This fix breaks backward compatibility. In
catalog_apply()ifautomerge = TRUEand the output contains alistof strings the list was expected to be merged into a character vector. But actually, the raw list was returned, which was not the intended behavior. This appends withSpatial*andsfobjects and withdata.frame. This bug should not have affected too many people.opt_output_files(ctg) <- paste0(tempdir(), "/{ORIGINALFILENAME}") option <- list(automerge = TRUE) ret <- catalog_apply(ctg, sptest, .options = option) # now returns a vector print(ret) #> "/tmp/RtmpV4CQll/file38f1.txt" "/tmp/RtmpV4CQll/file38g.txt" "/tmp/RtmpV4CQll/file38h.txt" "/tmp/RtmpV4CQll/file38i.txt"
-
When using a
grid_*function with aRasterLayerused as layout, if the layout was not empty or full of NAs, the values of the layout were transferred to the NA cells of the output #318. -
lascheck()no longer warns about "proj4string found but no CRS in the header". This was a false positive. Overall, CRS are better checked.
ENHANCEMENTS
-
opt_output_files()now prints a message when using theORIGINALFILENAMEtemplate with a chunk size that is not 0 to inform users that it does not make sense.opt_chunk_size(ctg) <- 800 opt_output_files(ctg) <- "{ORIGINALFILENAME}" #> ORIGINALFILENAME template has been used but the chunk size is not 0. This template makes sense only when processing by file.
-
Internally when building the chunks an informative error is now thrown when using the
ORIGINALFILENAMEtemplate with a chunk size that is not 0 to inform users that it does not make sense instead of the former uninformative error,Error in eval(parse(text = text, keep.source = FALSE), envir) : objet 'ORIGINALFILENAME' not found.#> Erreur : The template {ORIGINALFILENAME} makes sense only when processing by file (chunk size = 0). It is undefined otherwise. -
When using a "by file" processing strategy + a buffer around each file, up to 9 files may be read. Internally the chunks (
LAScluster) are now built in such a way that the first file read is the main one (and not one of the "buffer file"). This way, if the 9 files do not have the same scales and the same offsets, the main file has precedence over the other ones when rescaling and re-offsetting on-the-fly. This reduces the risk of incompatibilities and preserves the original pattern when processing a LAScatalog. -
grid_metrics()now constructs aRasterBrickin a better way and this reduces the risk of bugs with users' functions that sometimes return 0 length objects. TheRasterBrickwill now be properly filled withNAs. -
lascheck()now reports information if some points are flagged 'withheld', 'synthetic' or 'keypoint'. -
We moved the internal logic of chunk realignment with a raster from
catalog_apply()to the internal functioncatalog_makecluster(). This simplifies the source code, make it easier to maintain and test and will enable us to provide access, at the user level, to more internal functions in future releases.
lidR v2.2.2 (Release date: 2020-01-28)
FIXES
-
We introduced a bug in v2.2.0 in the catalog processing engine. Empty chunks triggered and error
i[1] is 1 which is out of range [1,nrow=0]internally. It now works again. -
Fix heap-buffer-overflow in
lasrangecorrection()when throwing an error about invalid range. -
lasunormalize()now update the header.
lidR v2.2.1 (Release date: 2020-01-21)
BREAKING CHANGE
-
imagerwas used to drive themcwatershed()algorithm.imageris an orphaned package that generated a warning on CRAN. Consequentlymcwatershed()has been removed. In attempt to provide an informative message to users, the function still exists but generates an error. Anyway this method was weak and buggy and it was a good reason to remove it... -
In version 2.2.0 we missed to put the parameter
rinpoint_metrics(). It is not yet supported but will be.
NEW FEATURES
- LAScatalog processing engine:
- In
catalog_apply()the optionsautomergenow supports automerging ofsfanddata.frameobjects. - New function
catalog_sapply()strictly equivalent tocatalog_apply()but with the optionautomerge = TRUEenforced to simplify the output whenever it is possible.
- In
ENHANCEMENTS
-
In the catalog processing engine, the graphical progression map is now able to plot the actual shape of the chunks. In the case of
lasclipit means that discs and polygons are displayed instead of bounding boxes. -
Multi-layers VRTs are returned as
RasterBrickinstead ofRasterStackfor consistency with in memory raster that are returns asRasterBrick. -
grid_functions now try to preserve the layer names when returning a VRT built from files written on disk. This works only with file formats that support to store layer name (e.g. notGTiff). -
There are now more than 900 unit tests for a coverage of 91%.
FIXES
-
Fix access to not mapped memory in one unit test (consequentless for users).
-
In
lasclip()the templateXCENTERactually gave the Y coordinate. It is now the correct X coordinate of the center of the clipped region. -
In
lasclip()the templateYCENTERwas not actually defined. It is now the correct Y coordinate of the center of the clipped region. -
Fix heap-buffer-overflow in
lasrangecorrection(). The range was likely to be badly computed for points that have a gpstime later than the last sensor position
lidR v2.2.0 (Release date: 2020-01-06)
NEW FEATURES
-
LAScatalog processing engine:
-
catalog_apply()gains an optionautomerge = TRUE.catalog_apply()used to return alistthat had to be merged by the user. This new option allows for automatic merging. This is a fail-safe feature. In the worst case, if the user-defined function returns a non-supported list of objects that cannot be merged it falls back to the former behavior i.e. it returns alist. Thus there is no risk associated with adding the optionautomerge = TRUEbut by defaut it is set toFALSEfor retrocompatibility. This might be switched toTRUEin future releases. -
opt_output_file()now interprets*as{ORIGINALFILENAME}for shorter syntax. The following is now accepted:
opt_output_file(ctg) <- "/home/user/data/norm/*_norm" # {*} is valid as well # instead of opt_output_file(ctg) <- "/home/user/data/norm/{ORIGINALFILENAME}_norm"
- The engine now supports "alternative directories". This is a very specific and undocumented feature useful in a single case of remote computing. More details on the wiki page.
ctg = readLAScatalog("~/folder/LASfiles/") ctg@input_options$alt_dir = c("/home/Alice/data/", "/home/Bob/remote/project1/data/")
- LAScatalog modification constraints are now relaxed. It is now possible to add or modify an attribute if this attribute has a name that is not reserved.
ctg$newattr <- 1 # is now allowed ctg$GUID <- TRUE # is still forbidden #> Erreur : LAScatalog data read from standard files cannot be modified
- The engine supports partial processing. It is possible to flag some files that will, or will not, be processed. These files are not removed from the LAScatalog. They are used to load a buffer, if required, for the files that are actually processed. To activate this option a new boolean attribute named
processedcan be added in the catalog.
ctg$processed <- TRUE ctg$processed[3:5] <- FALSE
-
-
3D rendering:
- The argument
colorPaletteof the functionplot()forLASobjects is now set to"auto"by default. This allows for this argument to not be specified even when plotting an attribute other than Z, and having an appropriate color palette by default. More interestingly, it will automatically apply a nice color scheme to the point cloud with the attribute 'Classification' following the ASPRS specifications. See #275.
plot(las) plot(las, color = "Intensity") plot(las, color = "ReturnNumber") plot(las, color = "Classification")
- In
plot.lasmetrics3d()the parametertrimis now set toInfby default.
- The argument
-
New function
point_metrics()- very similar togrid_metrics()but at the point level. The 'metrics' family is now complete.cloud_metrics()computes user-defined metrics at the point cloud level.grid_metrics()andhexbin_metrics()compute user-defined metrics at the pixel level.voxel_metricscomputes user-defined metrics at the voxel level.point_metrics()computes user-defined metrics at the point level. -
lasnormalize():- Gains an argument
use_classto control the points used as ground. - By default 'ground point' now includes points classified as water by default. This might be useful in regions with a lot of water because in this case
lasnormalize()can take forever to run (see #295)).
- Gains an argument
-
New function
sensor_tracking()to retrieve the position of the sensor in the sky. -
New function
lasrangecorrection()to normalize intensity using the sensor position (range correction) -
catalog_selectnow also allows files to process to be flagged interactively:ctg <- catalog_select(ctg, method = "flag_processed") ctg <- catalog_select(ctg, method = "flag_unprocessed")
-
grid_terrain()- Have a new argument
use_classto control which points are considered as ground points - With a
LAScatalogit now uses the filter-keep_classby default respecting the classes given inuse_class.
- Have a new argument
CHANGES
-
LAS()now rounds the values to 2 digits if no header is provided to fit with the default header automatically generated. This ensures that a perfectly validLASobject is built out of external data. This change is made by reference, meaning that the original dataset is also rounded.pts <- data.frame(X = runif(10), Y = runif(10), Z = runif(10)) las <- LAS(pts) # 'las' contains rounded values but 'pts' as well to avoid data copying
-
lasmetrics()is deprecated. Alllas*functions returnLASobjects exceptlasmetrics(). For consistency across the packagelasmetrics()becomescloud_metrics(). -
grid_metrics3d()andgrid_hexametrics()are deprecated. They are renamedvoxel_metrics()andhexbin_metrics()for naming consistency. -
The example dataset
Topography.lazis now larger and include attributes gpstime, PointSourceID and some classified lakes.
ENHANCEMENTS
-
Internally the package used a QuadTree as spatial index in versions <= 2.1.3. Spatial index has been rewritten and changed for a grid partition which is twice as fast as the former QuadTree. This change provides a significant boost (i.e. up to two times faster) to many algorithms of the package that rely on a spatial index. This includes
lmf(),shp_*(),wing2015(),pmf(),lassmooth(),tin(),pitfree(). Benchmark on a Intel Core i7-5600U CPU @ 2.60GHz × 2.# 1 x 1 km, 13 pts/m², 13.1 million points set_lidr_threads(n) tree_detection(las, lmf(3)) #> v2.1: 1 core: 80s - 4 cores: 38s #> v2.2: 1 core: 38s - 4 cores: 20s # 500 x 500 m, 12 pt/m², 3.2 million points lassnags(las, wing2015(neigh_radii = nr, BBPRthrsh_mat = bbpr_th)) #> v2.1: 1 core: 66s - 4 cores: 33s #> v2.2: 1 core: 43s - 4 cores: 21s # 250 x 250 m, 12 pt/m², 717.6 thousand points lasdetectshape(las3, shp_plane()) #> v2.1 - 1 cores: 12s - 4 cores: 7s #> v2.2 - 1 cores: 6s - 4 cores: 3s
-
Internally the Delaunay triangulation has been rewritten with
boostinstead of relying on thegeometrypackage. The Delaunay triangulation and the rasterization of the Delaunay triangulation are now written in C++ providing an important speed-up (up to three times faster) totin(),dsmtin()andpitfree(). However, for this to work, the point cloud must be converted to integers. This implies that the scale factors and offset in the header must be properly populated, which might not be the case if users have modified these values manually or if using a point cloud coming from a format other than las/laz. Benchmark on an Intel Core i7-5600U CPU @ 2.60GHz × 2.# 1.7 million ground points set_lidr_threads(n) grid_terrain(las, 0.5, tin()) #> v2.1: 1 core: 48s - 4 cores: 37s #> v2.2: 1 core: 22s - 4 cores: 20s # 560 thousand first returns (1.6 pts/m²) grid_canopy(las, res = 0.5, dsmtin()) #> v2.1: 1 core: 8s - 4 cores: 7s #> v2.2: 1 core: 3s - 4 cores: 3s # 560 thousand first returns (1.6 pts/m²) grid_canopy(las, res = 0.5, pitfree(c(0,2,5,10,15), c(0, 1.5))) #> v2.1: 1 core: 30s - 4 cores: 28s #> v2.2: 1 core: 11s - 4 cores: 9s
-
There are more than 100 new unit tests in
testthat. The coverage increased from 68 to 87%. -
The vignette named Speed-up the computations on a LAScatalog gains a section about the possible additional speed-up using the argument
selectfromreadLAS(). -
The vignette named LAScatalog formal class gains a section about partial processing.
-
Harmonization and review of the sections 'Supported processing options' in the man pages.
FIXES
-
Several minor fixes in
lascheck()for very improbable cases ofLASobjects likely to have been modified manually. -
Fix colorization of boolean data when plotting an object of class
lasmetrics3d(returned byvoxel_metrics()) #289 -
The LAScatalog engine now calls
raster::writeRaster()withNAflag = -999999because it seems that the default-Infgenerates a lot of trouble on windows when building a virtual raster mosaic withgdalUtils::gdalbuildvrt(). -
plot.LAS()better handles the case when coloring with an attribute that has only two values:NAand one other value. -
lasclip()was not actually able to retrieve the attributes of theSpatial*DataFrameorsfequivalent when usingopt_output_file(ctg) <- "/dir/{PLOTID}". -
opt_stop_early()was not actually working as expected. The processing was aborted without logs. It now prevent the catalog processing engine to stop even when an error occurs. -
In
tree_detection()if no tree is found (e.g. in a lake) the function crashed. It now returns an emptySpatialPointDataFrame. -
The argument
keep_lowestingrid_terrainreturned dummy output full of NAs because NAs have the precedence on actual numbers.
lidR v2.1.4 (Release date: 2019-10-15)
NEW FEATURES
-
grid_terrain()gains an argumentfull_raster = FALSE. -
lasnormalize()gains an argument...to tuneraster::extract()and use, for example,method = "bilinear".
FIXES
-
In
lasground()iflast_returns = TRUEand theLASis not properly populated i.e. no last return, the classification was not actually computed. The expected behavior was to use all the points. This is now the case. -
lasclip()is now able to clip into aLASobjects usingSpatialPointsorsf POINT. It previously worked only intoLAScatalogobjects. -
lasaddextrabyte_manual()was not actually working because thetypewas not converted to a numeric value according to the LAS specifications. -
Fix double precision floating point error in
grid_*function in some specific cases. This fix affect alsohighest()and other raster-based algorithms #273. -
lasreoffset()now checks for integer overflow and throws an error in case of invalid user request #274. -
Tolerance for internal
point_in_triangle()have been increased to fix double precision error in rasterization of a triangulation. This fixes some rareNAs inpitfree(),dsmtin()andtin(). -
The NAs are now correctly interpreted when writing a GDAL virtual raster #283.
-
Fix
lasmergespatial()with 'on disk' rasters #285. -
Fix
pitfree()with a single triangle case #288.
ENHANCEMENTS
pitfree()handles more errors and fails more nicely in some specific cases #286.
lidR v2.1.3 (Release date: 2019-09-10)
NEW FEATURES
-
New functions
lasrescale()andlasreoffset()to modify the scale factors and the offsets. The functions update the header and recompute the coordinates to get the proper rounded values in accordance with the new header. -
readLAS()throw (again) warnings for invalid files such as files with invalid scale factors, invalid bounding box, invalid attributes ReturnNumber and so on.
ENHANCEMENT
-
readLAScatalog()is 60% faster -
The progress bar of the LAScatalog processing engine has been removed in non interactive sessions and replaced by regular but more informative prints. This allows to track the state of the computation with a stream redirection to a file when running a script remotely for example.
R -f script.R &> log.txt &
FIXES
-
Fix an infinite loop in the knn search when k > number of points. This bug may affect
lasdetectectshape(),wing2012()and other functions that rely on a knn search. -
Using remote futures now works for any function that supports a
LAScataloginput. Previously remote evaluation of futures failed because of the presence ofreturn()statement in the code future#333plan(remote, workers = "132.203.41.25")
-
lasclipCircle()behaves identically forLASandLAScatalogobject. It now returns the points that are strictly inside the circle. Previously forLASobjects it also returned the point belonging on the disc. -
The bounding box is updated after
lastransform()#270 -
The offsets are updated after
lastransform()to prevent integer overflow when writing the point cloud in.lasfiles #272 -
Removed deprecated C++ functions
std::bind2ndas requested by CRAN.
NOTE
- All C++ source code has been reworked in a tidy framework to clean-up 4 years of mess. It is almost invisible for regular users but the size of the package has been reduced of several MB and many new tools will now be possible to build.
lidR v2.1.2 (Release date: 2019-08-07)
FIXES
- Fix a serious issue of uninitialized values in an internal C++ function but this issue is consequentless for the package.
lidR v2.1.1 (Release date: 2019-08-06)
NEW FEATURES
-
#266
lasmetricshas now a dispatch toLASandLASclustercluster objects. It means thatlasmetricscan be used withcatalog_applyin some specific cases where it has a meaning (see also #266):opt_chunk_buffer(ctg) <- 0 opt_chunk_size(ctg) <- 0 opt_filter(ctg) <- "-keep_first" opt_output_files(new_ctg) <- "" output <- catalog_apply(new_ctg, lasmetrics, func = .stdmetrics) output <- data.table::rbindlist(output)
ENHANCEMENT
-
lastrees()now uses S3 dispatcher system. When trying to use it with aLAScatalogobject, user will have a standard R message to state thatLAScatalogis not supported instead of an uninformative message that state that 'no slot of name "header" for this object of class "LAScatalog"' -
Internal code has been modified to drastically reduce probability of name intersection in
catalog_apply(). For example, the use of a function that have a parameterpincatalog_apply()failed because of partial matching between the true argumentpand the internal argumentprocessing_option. -
lasfilterdecimate()with algorithmhighest()is now more than 20 times faster.lasfiltersurfacepoints(), being a proxy of this algorithm, had the same speed-up -
plotforLASobjects gained the pan capability.
FIXES
- #267. A dummy character was introduced by mistake in a variable name breaking the automatic exportation of user object in
grid_metricswhen used with a parallelized plan (tree_metrics()was also affected).
lidR v2.1.0 (Release date: 2019-07-13)
VISIBLE CHANGES
Several algorithms are now natively parallelized at the C++ level with OpenMP. This has for consequences for speed-up of some computations by default but implies visible changes for users. For more details see help("lidR-parallelism"). The following only explains how to modify code to restore the exact former behavior.
In versions < 2.1.0 the catalog processing engine has R-based parallelism capabilities using the future package. The addition of C++-based parallelism introduced additional complexity. To prevent against nested parallelism and give the user the ability to use either R-based or C++-based parallelism (or a mix of the two), the function opt_cores() is no longer supported. If used it generates a message and does nothing. The strategy used to process the tiles in parallel must now be explicitly declared by users. This is anyway how it should have been designed from the beginning! For users, restoring the exact former behavior implies only one change.
In versions < 2.1.0 the following was correct:
library(lidR)
ctg <- catalog("folder/")
opt_cores(ctg) <- 4L
hmean <- grid_metrics(ctg, mean(Z))In versions >= 2.1.0 this must be explicitly declared with the future package:
library(lidR)
library(future)
plan(multisession)
ctg <- catalog("folder/")
hmean <- grid_metrics(ctg, mean(Z))NEW FEATURES
-
readLAS():- LAS 1.4 and point formats > 6 are now better supported.
lascheck()andprint()were updated to work correctly with these formats (#204) - New function
readLASheader()to read the header of a file in aLASheaderobject.
- LAS 1.4 and point formats > 6 are now better supported.
-
Coordinate Reference System:
- New function
wkt()to store a WKT CRS in a LAS 1.4 file. This function is the twin ofepsg()to store CRS. It updates theproj4stringand the header of the LAS object. This function is not expected to be used by users. Users must prefer the new functionprojection()instead. - New function
projection<-that updates both the slotproj4stringand the header with an EPSG code or a WKT string from aproj4stringor asp:CRSobject. This function supersedesepsg()andwkt()that are actually only useful internally and in specific cases. The vignetteLAS-classhas been updated accordingly.
projection(las) <- projection(raster)
- New function
-
LAScatalog processing engine:
- Progression estimation displayed on a map now handles warnings by coloring the chunks in orange.
- Progression estimation displayed on a map now colors in blue the chunks that are processing.
- The engine now returns the partial result in case of a fail.
- The engine now has a log system to help users reload the chunk that throws an error and try to understand what going wrong with this cluster specifically. If something went wrong a message like the following is displayed:
An error occurred when processing the chunk 190. Try to load this chunk with: chunk <- readRDS("/tmp/RtmpAlHUux/chunk190.rds") las <- readLAS(chunk) -
grid_metrics():- New function
stdshapemetrics()and lazy coding.stdshapemetricsto compute eigenvalue-related features (#217). - New argument
filteringrid_metrics(). This argument enables users to compute metrics on a subset of selected points such as "first returns", for example, without creating a copy of the point cloud. Such an argument is expected to be added later in several other functions.
hmean <- grid_metrics(las, ~mean(Z), 20, filter = ~ReturnNumber == 1)
- New function
-
New functions
lasdetectshape()for water and human-made structure detection with three algorithmsshp_plane(),shp_hplane(),shp_line(). -
plot():- For LAS objects
plot()gained an argumentaxis = TRUEto display axis. - For LAS objects
plot()gained an argumentlegend = TRUEto display color gradient legend (#224).
- For LAS objects
-
tree_hull():- Gained an argument
functo compute metrics for each tree, liketree_metrics()
convhulls <- tree_hulls(las, func = ~list(imean = mean(Intensity)))
- Gained an argument
-
Miscellaneous tools:
- The function
area()has been extended toLASheaderobjects. - New functions
npoints()anddensity()available forLAS,LASheaderandLAScatalogobjects that return what users may expect.
las <- readLAS("file.las", filter = "-keep_first") header <- readLASheader(file) ctg <- catalog("folder/") npoints(las) #> [1] 55756 npoints(header) #> [1] 81590 npoints(ctg) #> [1] 1257691 density(las) #> [1] 1.0483 density(header) #> [1] 1.5355 density(ctg) #> [1] 1.5123
- The function
-
Several functions are natively parallelized at the C++ level with OpenMP. See
help("lidR-parallelism")for more details. -
New function
catalog_selectfor interactive tile selection. -
lasgroundhave lost the argumentlast_returnsfor a more generic argumentfilter. Retro-compatibility as been preserved by interpreting adding an ellipsis.
NOTE
-
grid_metrics(),grid_metrics3d(),tree_metrics(),tree_hull(),grid_hexametrics()andlasmetrics()expect a formula as input. Users should not writegrid_metrics(las, mean(Z))butgrid_metrics(las, ~mean(Z)). The first syntax is still valid, for now. -
The argument named
fieldintree_metrics()is now namedattributefor consistency with all other functions. -
The documentation of supported options in
tree_*()functions was incorrect and has been fixed. -
readLAScatalog()replacescatalog().catalog()is soft-deprecated.
FIX
-
#264
grid_terrainnow filter degenerated ground points. -
#238 fix a floating point precision error in
p2ralgorithm.
ENHANCEMENT
- When reading a file that contains extrabytes attributes and these data are not loaded (e.g.
readLAS(f, select = "xyzi")) the header is updated to remove the non-loaded extrabytes. This fixes the issue #234 and enables LAS objects to be written without updating the header manually.
lidR v2.0.3 (Release date: 2019-05-02)
- Fix: in
li2012()the doc states that If R = 0 all the points are automatically considered as local maxima and the search step is skipped (much faster). This is now true. - Fix: in
lasmergespatialused with aSpatialPolygonDataFramewhen the bounding boxes do not match the full search was performed uselessly. Now the function exits early without searching anything. - Fix: #242 on Windows when using multicore options to process a LAScatalog the parameter of the algorithms were not exported to each session.
- Enhance: internally the function
tsearchthat searches in a triangulation is 25% faster giving a small speed-up topitfree()andtin()algorithms. - Enhance: in
lasmergespatialused with aSpatialPolygonDataFramethe function checks the bounding box of the polygon to speed-up the computation with complex polygons. - Doc: add a
?lidRpage to the manual. - Fix: in
li2012()the doc states that If R = 0 all the points are automatically considered as local maxima and the search step is skipped (much faster). This is now true. - Fix: in
lasmergespatialwith aSpatialPolygonDataFramewhen the bounding boxes do not match instead of exiting early without searching anything the full search was performed uselessly.
lidR v2.0.2 (Release date: 2019-03-02)
- Fix: #222
grid_*()functions return consistently aRasterLayerif there is a single layer. virtual raster mosaic were returned asRasterStackno matter the number of layers. - Fix: #223
lasmergespatial()wrongly copied shapefile attributes to each point when the parameterattributewas the name of an attribute of the shapefile. - Fix: #225
laspulse(),lasflightline(),lasscanline()were broken since v2.0.0. - Fix: #227 When processing a LAScatalog the chunks are better computed. In former version it was possible to have chunks that lie on tile only because of the buffer. These chunks are not build anymore.
- Fix: #227 When processing a LAScatalog some chunks may belong in a file/tile but when actually reading the points in the file the chunks could be empty with points only in the buffer region. In these case an empty point cloud is returned and the computation is be skipped.
- Fix: #228
lasmergespatial()andlasclip()loose precision when extracting polygons due to missing digits in the WKT string used to rebuild the polygons at C++ level.
lidR v2.0.1 (Release date: 2019-02-02)
- Change: the function
cataloghas been slightly modified in prevision of the release of the packagerlas 1.3.0to preserve future compatibility. This is invisible for the users. - New:
lasnormalizegained a parameterna.rm = TRUE - Fix: an error occured when plotting a LAScatalog with the option
chunk_pattern = TRUE: object 'ctg' not found. - Fix: examples in documentation of
tin()andknnidw()were inverted. - Fix: #213 bug when using option
keep_lowestingrid_terrain. - Fix: #212 bug when merging big rasters that exceed the memory allowed by the raster package
- Fix: bug when merging rasters when some of then only have one cell
- Fix: bug when printing a 0 point LAS object
lidR v2.0.0 (Release date: 2019-01-02)
Why versions > 2.0 are incompatible with versions 1.x.y?
The lidR package versions 1 were mainly built upon "personal R scripts" I wrote 3 years ago. These scripts were written for my own use at a time when the lidR package was much smaller (both in term of code and users). The lidR package became a relatively large framework built on top of an unstructured base so it became impossible to develop it further. Many features and functions were missing because the way lidR was built did not allow them to be written. The new release (lidR version 2) breaks the former code to build a more robust, more consistent and more scalable framework that is intended and expected to continue for years without the need to break anything more in the future.
Old binaries can still be found here for 6 months:
- Windows R-3.6
- Windows R-3.5
- Windows R-3.4
- Mac OSX R-3.5
- Mac OSX R-3.4
Overview of the main visible changes
lidR as a GIS tool
lidR versions 1 was not a GIS tool. For example, rasterization functions such as grid_metrics() or grid_canopy() returned a data.frame. Tree tops extraction with tree_detection() also returned a data.frame. Tree segmentation with lastrees() accepted RasterLayer or data.frame as input in a very inconsistent way. Moreover, the CRS of the point cloud was useless and never propagated to the outputs because outputs were not spatial objects.
lidR version 2 consistently uses Raster* and Spatial* objects everywhere. Rasterization functions such as grid_metrics() or grid_canopy() return Raster* objects. Tree tops extraction returns SpatialPointDataFrame objects. Tree segmentation methods accept SpatialPointDataFrame objects only in a consistent way across functions. The CRS of the point cloud is always propagated to the outputs. LAS objects are Spatial objects. LAScatalog objects are SpatialPolygonDataFrame objects. In short, lidR version 2 is now a GIS tool that is fully compatible with the R ecosystem.
No longer any update by reference
Several lidR functions used to update objects by reference. In lidR versions 1 the user wrote: lasnormalize(las) instead of las2 <- lasnormalize(las1). This used to make sense in R < 3.1 but now the gain is no longer as relevant because R makes shallow copies instead of deep copies.
To simplfy, let's assume that we have a 1 GB data.frame that stores the point cloud. In R < 3.1 las2 was a copy of las1 i.e. las1 + las2 = 2GB . This is why we made functions that worked by reference that implied no copy at all. This was memory optimized but not common or traditional in R. The question of memory optimization is now less relevant since R >= 3.1. In the previous example las2 is no longer a deep copy of las1, but a shallow copy. Thus lidR now consistently uses the traditional syntax y <- f(x).
Algorithm dispatch
The frame of lidR versions 1 was designed at a time when there were fewer algorithms. The increasing number of algorithms led to inconsistent ways to dispatch algorithms. For example:
grid_canopy()implemented one algorithm and a second functiongrid_tincanopy()was created to implement another algorithm. With two functions the switch was possible by using two different names (algorithms dispatched by names).grid_tincanopy()actually implemented two algorithms in one function. The switch was possible by changing the input parameters in the function (algorithm dispatched by input).lastrees()had several variants that provided access to several algorithms:lastrees_li(),lastrees_dalpontes(),lastrees_watershed(), and so on. With several functions the switch was possible by using several different names (algorithms dispatched by names).tree_detectiondid not have several variants, thus it was impossible to introduce a new algorithm (no dispatch at all).
lidR version 2 comes with a flexible and scalable dispatch method that unifies all the former functions. For example, grid_canopy() is the only function to make a CHM. There is no longer the need for a second function grid_tincanopy(). grid_canopy() unifies the two functions by accepting as input an algorithm for a digital surface model:
chm = grid_canopy(las, res = 1, algo = pitfree())
chm = grid_canopy(las, res = 1, algo = p2r(0.2))The same idea drives several other functions including lastrees, lassnags, tree_detection, grid_terrain, lasnormalize, and so on. Examples:
ttops = tree_detection(las, algo = lmf(5))
ttops = tree_detection(las, algo = lidRplugins::multichm(1,2))
lastrees(las, algo = li2012(1.5, 2))
lastrees(las, algo = watershed(chm))
lasnormalize(las, algo = tin())
lasnormalize(las, algo = knnidw(k = 10))This allows lidR to be extended with new algorithms without any restriction either in lidR or even from third-party tools. Also, how lidR functions are used is now more consistent across the package.
LAScatalog processing engine
lidR versions 1 was designed to run algorithms on medium-sized point clouds loaded in memory but not to run algorithms over a set of files covering wide areas. In addition, lidR 1 had a poorly and inconsistently designed engine to process catalogs of las files. For example:
- It was possible to extract a polygon of points from a
LAScatalogbut not multipart-polygons or polygons with holes. This was only possible withLASobjects i.e loaded in memory (inconsistent behaviors within a function). - It was possible to run
grid_metrics()on aLAScatalogi.e. over a wide area not loaded in memory, but notlasnormalize,lasgroundortree_detection(inconsistent behavior across the functions).
lidR version 2 comes with a powerful and scalable catalog processing engine. Almost all the lidR functions can be used seamlessly with either LAS or LAScatalog objects. The following chunks of code are now possible:
ctg = catalog("folfer/to/las/file")
opt_output_file(ctg) <- "folder/to/normalized/las/files/{ORIGINALFILENAME}_normalized"
new_ctg = lasnormalize(ctg, algo = tin())Complete description of visible changes
LAS class
- Change: the
LASclass is now aSpatialobject or, more technically, it inherits aSpatialobject. - Change: being a
Spatialobject, aLASobject no longer has a@crsslot. It has now a slot@proj4stringthat is accessible with the functionsraster::projectionorsp::proj4string - New: being a
Spatialobject, aLASobject inherits multiple functions fromrasterandsp, such$and[[accessors orraster::extent,sp::bbox,raster::projection, and so on. However, the replacement method$<-,[[<-have restricted capabilities to ensure aLASobject cannot be modified in a way that implies loosening the properties of the LAS specifications. - New: empty
LASobjects with 0 points are now allowed. This has repercussions for several functions includinglasfilter,lasclip, andreadLASthat do not returnNULLfor empty data but aLASobject with 0 points. This new behavior has been introduced to fix the old inconsistent behavior of functions that return eitherLASorNULLobjects.LASobjects are always returned.
LAScatalog class
- Change: the
LAScatalogclass is now aSpatialPolygonsDataFrameor, more technically, it inherits aSpatialPolygonsDataFrame. - Change: being a
SpatialPolygonsDataFrameobject, aLAScatalogno longer has a@crsslot. It has now a slot@proj4stringthat is accessible with the functionsraster::projectionorsp::proj4string. - Change: being a
SpatialPolygonsDataFrameaLAScatalogcan be plotted withsp::spplot(). - Change: there are no longer any slots
@cores,@by_file,@buffer, and so on. They are replaced by more generic and scalable slots@processing_options,@output_options,@clustering_optionsand@input_optionsthat are list of options classified by their main roles. - Change: documentation has been entirely rewritten to explain the whole potential of the class.
- Change: functions
by_file,progress,tiling_size,bufferwere replaced byopt_chunk_size,opt_chunk_buffer,opt_progress, and so on. These allow for a consistent set of functions that do not overlap with functions fromrasterorsp. - Change: standard column names were renamed to make syntactically-valid names and for compatibility with
spfunctions.
readLAS
- Change:
readLASno longer supports optionPFC. Users must use the functionslaspulse,lasflightlinesmanually.
lasclip
- New:
lasclipnow works both with aLASobject and aLAScatalogobject in a seamless and consistent way. There are no longer any differences between the capabilities of theLASversion or theLAScatalogone. - New:
lasclipsupport many geometries including multipart polygons and polygons with holes, both with aLASobject and aLAScatalogobject. - Change: The option
insidehas been removed for consistency because it cannot be safely supported both onLASandLAScatalog. - Change: The option
ofilehas been removed for consistency and this option in now managed by theLAScatalogprocessing engine. For example, one can extract ground inventories and write them inlazfiles automatically named after their center coordinates like this:
ctg = catalog(folder)
output_files(ctg) <- "path/to/a/file_{XCENTER}_{YCENTER}"
laz_compression(ctg) <- TRUE
new_ctg = lasclipCircle(ctg, xc,yc, r)- Change: documentation has been reviewed and extended
- Change:
lasclipdoes not returnNULLanymore for empty queries but an emptyLASobject. - Fix:
lasclipRectanglereturns the same output both with aLASand aLAScatalog. With aLASthe rectangle is now closed on the bottom and the left and open on the right and the top.
catalog_queries
- Change:
catalog_querieshas been removed because it is superseded bylasclip.
lasnormalize
- Change:
lasnormalize()no longer updates the original object by reference. - Change: remove the old option
copy = TRUEthat is now meaningless. - Change:
lasnormalize()now relies on lidR algorithms dispatch (see also the main new features above). - New:
lasnormalize()can be applied on aLAScatalogto write a new normalized catalog using the catalog processing engine (see also the main new features above).
lasclassify
- Change:
lasclassify()is now namedlasmergespatial()to free the namelasclassifythat should be reserved for other usage. - Change:
lasmergespatial()no longer updates the original object by reference. - Fix: the classification, when made with a
RasterLayer, preserves the data type of theRasterLayer. This also fixes the fact thatlastrees()used to classify the tree withdoubleinstead ofint.
tree_detection
- Change:
tree_detection()now relies on the new dispatch method (see also the main new features above). - New: algorithm
lmfhas user-defined variable-sized search windows and two possible search window shapes (square or disc). - New: introduction of the
manualalgorithm for manual correction of tree detection. - New:
tree_detectionalgorithms are seamlessly useable with aLAScatalogobject by using the catalog processing engine (see also the main new features above). Thus, the following just works:
ctg <- catalog(folder)
ttop <- tree_detection(ctg, lmf(5))- Change: the
lmfalgorithm, when used with aRasterLayeras input, expects parameters given in the units of the map and no longer in pixels. - Change:
tree_detection()function consistently returns aSpatialPointsDataFramewhatever the algorithm. - Change:
tree_detection()function based on a CHM no longer support alasmetricobject as input. Anyway, this class no longer exists.
tree_metrics
- Change:
tree_metrics()returns aSpatialPointsDataFrame. - Change:
tree_metrics()is seamlessly useable with aLAScatalogusing the catalog processing engine (see also the main new features above). Thus, this just works if the las file has extra bytes attributes that store the tree ids:
ctg <- catalog(folder)
metrics <- tree_metrics(ctg, list(`Mean I` = mean(Intensity)))lastrees
- Change:
lastrees()now relies on the new algorithms dispatch method (see also the main new features above). - New: introduction of the
mcwatershedalgorithm that implements a marker-controlled watershed.
grid_metrics
- Change:
grid_metrics()as well as othergrid_*functions consistently return aRasterLayeror aRasterBrickinstead of adata.table. - Change: option
splitlineshas been removed.grid_metrics()used to return adata.tablebecause of thesplitlinesoption and lidR was built on top of that feature from the very beginning. Now lidR consistently usesspandrasterand this option is no longer supported.
grid_terrain
- Change:
grid_terrain()now relies on the new algorithms dispatch method (see also the main new features above). - Change:
grid_terrain()consistently returns aRasterLayerinstead of adata.table, whatever the algorithm used.
grid_canopy
- Change:
grid_canopy()now relies on the new algorithms dispatch method (see also the main new features above). It unifies the former functionsgrid_canopy()andgrid_tincanopy(). - Change:
grid_canopy()consistently returns aRasterLayerinstead of adata.table, whatever the algorithm used. - Fix: the pitfree algorithm fails if a layer contains only 1 or 2 points.
- Fix: the p2r algorithm is five times faster with the subcircle tweak.
grid_tincanopy
- Change:
grid_tincanopy()has been removed. Digital Surface Models are consistently driven by the functiongrid_canopy()and the lidR algorithm dispatch engine. The algorithms that replacedgrid_tincanopy()aredsmtinandpitfree.
grid_hexametrics
- Change: as for
grid_metrics, the parametersplitlineshas been removed. - Change: the function returns a
hexbinobject or a list ofhexbinobjects and no longerdata.tableobjects.
grid_catalog
- Change:
grid_catalog()has been removed. The newLAScatalogprocessing engine means that this function is no longer useful.
class lasmetrics
data.tablewith a classlasmetricsno longer exists. It has been consistently replaced byRasterLayerandRasterBrickeverywhere.as.rasterno longer exists because it used to convertlasmetricsintoRasterLayerandRasterStack.as.spatialno longer convertslasmetricstoSpatialPixelsDataFramebut still convertsLAStoSpatialPointsDataFrame.plot.lasmetricshas been removed obviously.
lasroi
- Change:
lasoi()has been removed. It was not useful and 'buggy'. It might be reintroduced later inlasclipManual.
lascolor
- Change:
lascolor()has been removed. It was one of the first functions of the package and is no longer useful becauseplot()has enhanced capabilities.
lasfilterdecimate
- Change: now relies on the new algorithms dispatch method (see also the main new features above).
- New: introduction of the algorithm
highestavailable inlasfilterdecimate(). This supersedes the functionlasfiltersurfacepoints().
lassnags
- Change:
lassnags()now relies on the new algorithms dispatch method (see also the main new features above). - New:
lasnsnags()can be applied on aLAScatalogto write a new catalog using the catalog processing engine (see also the main new features above).
lidr_options
- Change:
lidr_option()has been removed. The options are now managed by regular R base options with functionoptions(). Available lidR options are named with the prefixlidR.
Example files
- New: the three example files are now georeferenced with an EPSG code that is read and converted to a
proj4string. - New: the example file
MixedConifers.lazcontains the segmented trees in extra bytes 0.
plot
- New:
plot()forLASobjects supportsRGBas a color attribute. - New: option
colorsupports lazy evaluation. This syntax is correct:plot(las, color = Classification). - New: option
clear_artifact = TRUEshifts the point cloud to (0,0) and reduces the display artifact due to the use of floating point inrgl. - New: new functions
add_treetops3d,add_dtm3dandplot_dtm3dadd elements in the point cloud. - Change:
trimdoes not trim on a percentile of values but on the values themselves.
Coordinate reference system
- New: coordinate reference system is supported everywhere and can be written in las files. See function
epsg(). - New: function
lastranformthat returns transformed coordinates of aLASobject using the CRS argument.
New functions
- New: function
lasfilterduplicates - New: function
lascheck - New: function
lasvoxelize
Other changes that are not directly visible
- Change: the code that drives the
point_in_polygonalgorithm relies onboostand drastically simplifies the former code oflasmergespatial() - Change: many memory optimizations