Skip to content

Commit

Permalink
Add 'gdal raster clip' and 'gdal raster pipeline read ... ! clip ... …
Browse files Browse the repository at this point in the history
  • Loading branch information
rouault committed Jan 10, 2025
1 parent ee04fce commit 0f31531
Show file tree
Hide file tree
Showing 11 changed files with 508 additions and 0 deletions.
1 change: 1 addition & 0 deletions apps/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ add_library(
gdalalg_raster.cpp
gdalalg_raster_buildvrt.cpp
gdalalg_raster_info.cpp
gdalalg_raster_clip.cpp
gdalalg_raster_convert.cpp
gdalalg_raster_edit.cpp
gdalalg_raster_pipeline.cpp
Expand Down
2 changes: 2 additions & 0 deletions apps/gdalalg_raster.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@

#include "gdalalg_raster_buildvrt.h"
#include "gdalalg_raster_info.h"
#include "gdalalg_raster_clip.h"
#include "gdalalg_raster_convert.h"
#include "gdalalg_raster_edit.h"
#include "gdalalg_raster_pipeline.h"
Expand All @@ -39,6 +40,7 @@ class GDALRasterAlgorithm final : public GDALAlgorithm
{
RegisterSubAlgorithm<GDALRasterInfoAlgorithm>();
RegisterSubAlgorithm<GDALRasterConvertAlgorithm>();
RegisterSubAlgorithm<GDALRasterClipAlgorithmStandalone>();
RegisterSubAlgorithm<GDALRasterEditAlgorithmStandalone>();
RegisterSubAlgorithm<GDALRasterPipelineAlgorithm>();
RegisterSubAlgorithm<GDALRasterReprojectAlgorithmStandalone>();
Expand Down
145 changes: 145 additions & 0 deletions apps/gdalalg_raster_clip.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,145 @@
/******************************************************************************
*
* Project: GDAL
* Purpose: "clip" step of "raster pipeline", or "gdal raster clip" standalone
* Author: Even Rouault <even dot rouault at spatialys.com>
*
******************************************************************************
* Copyright (c) 2025, Even Rouault <even dot rouault at spatialys.com>
*
* SPDX-License-Identifier: MIT
****************************************************************************/

#include "gdalalg_raster_clip.h"

#include "gdal_priv.h"
#include "gdal_utils.h"

#include <algorithm>

//! @cond Doxygen_Suppress

#ifndef _
#define _(x) (x)
#endif

/************************************************************************/
/* GDALRasterClipAlgorithm::GDALRasterClipAlgorithm() */
/************************************************************************/

GDALRasterClipAlgorithm::GDALRasterClipAlgorithm(bool standaloneStep)
: GDALRasterPipelineStepAlgorithm(NAME, DESCRIPTION, HELP_URL,
standaloneStep)
{
AddBBOXArg(&m_bbox, _("Clipping bounding box as xmin,ymin,xmax,ymax"))
.SetMutualExclusionGroup("exclusion-group");
AddArg("bbox-crs", 0, _("CRS of clipping bounding box"), &m_bboxCrs)
.SetIsCRSArg()
.AddHiddenAlias("bbox_srs");
AddArg("like", 0, _("Raster dataset to use as a template for bounds"),
&m_likeDataset, GDAL_OF_RASTER)
.SetMetaVar("DATASET")
.SetMutualExclusionGroup("exclusion-group");
}

/************************************************************************/
/* GDALRasterClipAlgorithm::RunStep() */
/************************************************************************/

bool GDALRasterClipAlgorithm::RunStep(GDALProgressFunc, void *)
{
CPLAssert(m_inputDataset.GetDatasetRef());
CPLAssert(m_outputDataset.GetName().empty());
CPLAssert(!m_outputDataset.GetDatasetRef());

CPLStringList aosOptions;
aosOptions.AddString("-of");
aosOptions.AddString("VRT");
if (!m_bbox.empty())
{
aosOptions.AddString("-projwin");
aosOptions.AddString(CPLSPrintf("%.17g", m_bbox[0])); // minx
aosOptions.AddString(CPLSPrintf("%.17g", m_bbox[3])); // maxy
aosOptions.AddString(CPLSPrintf("%.17g", m_bbox[2])); // maxx
aosOptions.AddString(CPLSPrintf("%.17g", m_bbox[1])); // miny

if (!m_bboxCrs.empty())
{
aosOptions.AddString("-projwin_srs");
aosOptions.AddString(m_bboxCrs.c_str());
}
}
else if (auto poLikeDS = m_likeDataset.GetDatasetRef())
{
double adfGT[6];
if (poLikeDS->GetGeoTransform(adfGT) != CE_None)
{
ReportError(CE_Failure, CPLE_AppDefined,
"Dataset '%s' has no geotransform matrix. Its bounds "
"cannot be established.",
poLikeDS->GetDescription());
return false;
}
auto poLikeSRS = poLikeDS->GetSpatialRef();
if (!poLikeSRS)
{
ReportError(CE_Failure, CPLE_AppDefined,
"Dataset '%s' has no SRS. Its bounds cannot be used.",
poLikeDS->GetDescription());
return false;
}
const double dfTLX = adfGT[0];
const double dfTLY = adfGT[3];
const double dfTRX = adfGT[0] + poLikeDS->GetRasterXSize() * adfGT[1];
const double dfTRY = adfGT[3] + poLikeDS->GetRasterXSize() * adfGT[4];
const double dfBLX = adfGT[0] + poLikeDS->GetRasterYSize() * adfGT[2];
const double dfBLY = adfGT[3] + poLikeDS->GetRasterYSize() * adfGT[5];
const double dfBRX = adfGT[0] + poLikeDS->GetRasterXSize() * adfGT[1] +
poLikeDS->GetRasterYSize() * adfGT[2];
const double dfBRY = adfGT[3] + poLikeDS->GetRasterXSize() * adfGT[4] +
poLikeDS->GetRasterYSize() * adfGT[5];
const double dfMinX =
std::min(std::min(dfTLX, dfTRX), std::min(dfBLX, dfBRX));
const double dfMinY =
std::min(std::min(dfTLY, dfTRY), std::min(dfBLY, dfBRY));
const double dfMaxX =
std::max(std::max(dfTLX, dfTRX), std::max(dfBLX, dfBRX));
const double dfMaxY =
std::max(std::max(dfTLY, dfTRY), std::max(dfBLY, dfBRY));

aosOptions.AddString("-projwin");
aosOptions.AddString(CPLSPrintf("%.17g", dfMinX));
aosOptions.AddString(CPLSPrintf("%.17g", dfMaxY));
aosOptions.AddString(CPLSPrintf("%.17g", dfMaxX));
aosOptions.AddString(CPLSPrintf("%.17g", dfMinY));

const char *const apszOptions[] = {"FORMAT=WKT2", nullptr};
const std::string osWKT = poLikeSRS->exportToWkt(apszOptions);
aosOptions.AddString("-projwin_srs");
aosOptions.AddString(osWKT.c_str());
}
else
{
ReportError(CE_Failure, CPLE_AppDefined,
"Either --bbox or --like must be specified");
return false;
}

GDALTranslateOptions *psOptions =
GDALTranslateOptionsNew(aosOptions.List(), nullptr);

GDALDatasetH hSrcDS = GDALDataset::ToHandle(m_inputDataset.GetDatasetRef());
auto poRetDS =
GDALDataset::FromHandle(GDALTranslate("", hSrcDS, psOptions, nullptr));
GDALTranslateOptionsFree(psOptions);

const bool bOK = poRetDS != nullptr;
if (bOK)
{
m_outputDataset.Set(std::unique_ptr<GDALDataset>(poRetDS));
}

return bOK;
}

//! @endcond
62 changes: 62 additions & 0 deletions apps/gdalalg_raster_clip.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
/******************************************************************************
*
* Project: GDAL
* Purpose: "clip" step of "raster pipeline", or "gdal raster clip" standalone
* Author: Even Rouault <even dot rouault at spatialys.com>
*
******************************************************************************
* Copyright (c) 2025, Even Rouault <even dot rouault at spatialys.com>
*
* SPDX-License-Identifier: MIT
****************************************************************************/

#ifndef GDALALG_RASTER_CLIP_INCLUDED
#define GDALALG_RASTER_CLIP_INCLUDED

#include "gdalalg_raster_pipeline.h"

//! @cond Doxygen_Suppress

/************************************************************************/
/* GDALRasterClipAlgorithm */
/************************************************************************/

class GDALRasterClipAlgorithm /* non final */
: public GDALRasterPipelineStepAlgorithm
{
public:
static constexpr const char *NAME = "clip";
static constexpr const char *DESCRIPTION = "Clip a raster dataset.";
static constexpr const char *HELP_URL = "/programs/gdal_raster_clip.html";

static std::vector<std::string> GetAliases()
{
return {};
}

explicit GDALRasterClipAlgorithm(bool standaloneStep = false);

private:
bool RunStep(GDALProgressFunc pfnProgress, void *pProgressData) override;

std::vector<double> m_bbox{};
std::string m_bboxCrs{};
GDALArgDatasetValue m_likeDataset{};
};

/************************************************************************/
/* GDALRasterClipAlgorithmStandalone */
/************************************************************************/

class GDALRasterClipAlgorithmStandalone final : public GDALRasterClipAlgorithm
{
public:
GDALRasterClipAlgorithmStandalone()
: GDALRasterClipAlgorithm(/* standaloneStep = */ true)
{
}
};

//! @endcond

#endif /* GDALALG_RASTER_CLIP_INCLUDED */
2 changes: 2 additions & 0 deletions apps/gdalalg_raster_pipeline.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@

#include "gdalalg_raster_pipeline.h"
#include "gdalalg_raster_read.h"
#include "gdalalg_raster_clip.h"
#include "gdalalg_raster_edit.h"
#include "gdalalg_raster_reproject.h"
#include "gdalalg_raster_write.h"
Expand Down Expand Up @@ -161,6 +162,7 @@ GDALRasterPipelineAlgorithm::GDALRasterPipelineAlgorithm(

m_stepRegistry.Register<GDALRasterReadAlgorithm>();
m_stepRegistry.Register<GDALRasterWriteAlgorithm>();
m_stepRegistry.Register<GDALRasterClipAlgorithm>();
m_stepRegistry.Register<GDALRasterEditAlgorithm>();
m_stepRegistry.Register<GDALRasterReprojectAlgorithm>();
}
Expand Down
Loading

0 comments on commit 0f31531

Please sign in to comment.