Skip to content

Commit 0f872e8

Browse files
authored
Merge GH-1793 (KMZ ground overlay support)
2 parents 2bfa408 + da74804 commit 0f872e8

28 files changed

+1233
-77
lines changed

src/core/map_printer.cpp

+34-22
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/*
22
* Copyright 2012, 2013 Thomas Schöps
3-
* Copyright 2012-2018 Kai Pastor
3+
* Copyright 2012-2020 Kai Pastor
44
*
55
* This file is part of OpenOrienteering.
66
*
@@ -391,6 +391,12 @@ const QPrinterInfo* MapPrinter::imageTarget()
391391
return &image_target;
392392
}
393393

394+
const QPrinterInfo* MapPrinter::kmzTarget()
395+
{
396+
static QPrinterInfo kmz_target; // TODO: set name and features?
397+
return &kmz_target;
398+
}
399+
394400

395401
// QPageSize (::key(), ::name()) made this list mostly obsolete.
396402
// But we keep it in v0.9 for loading maps where we used names
@@ -454,14 +460,17 @@ void MapPrinter::setTarget(const QPrinterInfo* new_target)
454460
target = new_target;
455461
else if (new_target == imageTarget())
456462
target = new_target;
463+
else if (new_target == kmzTarget())
464+
target = new_target;
457465
else
458466
{
459467
// We don't own this target, so we need to make a copy.
460468
target_copy = *new_target;
461469
target = &target_copy;
462470
}
463471

464-
if (old_target == imageTarget() || new_target == imageTarget())
472+
if (old_target == imageTarget() || new_target == imageTarget()
473+
|| old_target == kmzTarget() || new_target == kmzTarget())
465474
{
466475
// No page margins. Will emit pageFormatChanged( ).
467476
setCustomPageSize(page_format.page_rect.size());
@@ -536,7 +545,8 @@ bool MapPrinter::isPrinter() const
536545
{
537546
bool is_printer = target
538547
&& target != imageTarget()
539-
&& target != pdfTarget();
548+
&& target != pdfTarget()
549+
&& target != kmzTarget();
540550
return is_printer;
541551
}
542552

@@ -547,7 +557,7 @@ void MapPrinter::setPrintArea(const QRectF& area)
547557
{
548558
print_area = area;
549559

550-
if (target == imageTarget() && print_area.size() != page_format.paper_dimensions)
560+
if ((target == imageTarget() || target == kmzTarget()) && print_area.size() != page_format.paper_dimensions)
551561
setCustomPageSize(print_area.size() * scale_adjustment);
552562

553563
updatePageBreaks();
@@ -627,7 +637,7 @@ void MapPrinter::setOverlap(qreal h_overlap, qreal v_overlap)
627637

628638
void MapPrinter::updatePaperDimensions()
629639
{
630-
if (target == imageTarget() && page_format.page_size == QPageSize::Custom)
640+
if ((target == imageTarget() || target == kmzTarget()) && page_format.page_size == QPageSize::Custom)
631641
{
632642
// No margins, no need to query QPrinter.
633643
page_format.page_rect = QRectF(QPointF(0.0, 0.0), page_format.paper_dimensions);
@@ -639,7 +649,7 @@ void MapPrinter::updatePaperDimensions()
639649

640650
QPrinter* printer = target ? new QPrinter(*target, QPrinter::HighResolution)
641651
: new QPrinter(QPrinter::HighResolution);
642-
if (!printer->isValid() || target == imageTarget() || target == pdfTarget())
652+
if (!printer->isValid() || target == imageTarget() || target == kmzTarget() || target == pdfTarget())
643653
printer->setOutputFormat(QPrinter::PdfFormat);
644654

645655
if (page_format.page_size == QPageSize::Custom)
@@ -657,7 +667,7 @@ void MapPrinter::updatePaperDimensions()
657667
page_format.page_rect = printer->paperRect(QPrinter::Millimeter);
658668
page_format.paper_dimensions = page_format.page_rect.size();
659669

660-
if ( target != imageTarget() && target != pdfTarget() &&
670+
if ( target != imageTarget() && target != kmzTarget() && target != pdfTarget() &&
661671
page_format.page_size != QPageSize::Custom )
662672
{
663673
page_format.page_rect = printer->pageRect(QPrinter::Millimeter);
@@ -896,7 +906,7 @@ void MapPrinter::takePrinterSettings(const QPrinter* printer)
896906
if (!printer) return;
897907

898908
MapPrinterPageFormat f(*printer);
899-
if (target == pdfTarget() || target == imageTarget())
909+
if (target == pdfTarget() || target == imageTarget() || target == kmzTarget())
900910
{
901911
f.page_rect = QRectF(QPointF(0.0, 0.0), f.paper_dimensions);
902912
}
@@ -921,6 +931,20 @@ void MapPrinter::takePrinterSettings(const QPrinter* printer)
921931

922932

923933
void MapPrinter::drawPage(QPainter* device_painter, const QRectF& page_extent, QImage* page_buffer) const
934+
{
935+
// Determine transformation and clipping for page extent and region
936+
const qreal units_per_mm = options.resolution / 25.4;
937+
// Translate for top left page margin
938+
auto transform = QTransform::fromScale(units_per_mm, units_per_mm);
939+
transform.translate(page_format.page_rect.left(), page_format.page_rect.top());
940+
// Convert native map scale to print scale
941+
transform.scale(scale_adjustment, scale_adjustment);
942+
// Translate and clip for margins and print area
943+
transform.translate(-page_extent.left(), -page_extent.top());
944+
drawPage(device_painter, page_extent, transform, page_buffer);
945+
}
946+
947+
void MapPrinter::drawPage(QPainter* device_painter, const QRectF& page_extent, const QTransform& page_extent_transform, QImage* page_buffer) const
924948
{
925949
// Logical units per mm
926950
const qreal units_per_mm = options.resolution / 25.4;
@@ -930,18 +954,6 @@ void MapPrinter::drawPage(QPainter* device_painter, const QRectF& page_extent, Q
930954
| QPainter::Antialiasing
931955
| QPainter::SmoothPixmapTransform;
932956

933-
// Determine transformation and clipping for page extent and region
934-
const auto page_extent_transform = [this, units_per_mm, page_extent]() {
935-
// Translate for top left page margin
936-
auto transform = QTransform::fromScale(units_per_mm, units_per_mm);
937-
transform.translate(page_format.page_rect.left(), page_format.page_rect.top());
938-
// Convert native map scale to print scale
939-
transform.scale(scale_adjustment, scale_adjustment);
940-
// Translate and clip for margins and print area
941-
transform.translate(-page_extent.left(), -page_extent.top());
942-
return transform;
943-
}();
944-
945957
const auto page_region_used = page_extent.intersected(print_area);
946958

947959

@@ -964,7 +976,7 @@ void MapPrinter::drawPage(QPainter* device_painter, const QRectF& page_extent, Q
964976
* When the target is an image, use the temporary image to enforce the given
965977
* resolution.
966978
*/
967-
const bool use_buffer_for_map = rasterModeSelected() || target == imageTarget() || engineWillRasterize();
979+
const bool use_buffer_for_map = rasterModeSelected() || target == imageTarget() || target == kmzTarget() || engineWillRasterize();
968980
bool use_page_buffer = use_buffer_for_map;
969981

970982
auto first_front_template = map.getFirstFrontTemplate();
@@ -1232,7 +1244,7 @@ void MapPrinter::drawSeparationPages(QPrinter* printer, QPainter* device_painter
12321244

12331245
// Translate and clip for margins and print area
12341246
device_painter->translate(-page_extent.left(), -page_extent.top());
1235-
device_painter->setClipRect(page_extent.intersected(print_area), Qt::ReplaceClip);
1247+
device_painter->setClipRect(page_extent.intersected(print_area).adjusted(-10, 10, 10, 10), Qt::ReplaceClip);
12361248

12371249
bool need_new_page = false;
12381250
for (int i = map.getNumColors() - 1; i >= 0; --i)

src/core/map_printer.h

+6-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/*
22
* Copyright 2012, 2013 Thomas Schöps
3-
* Copyright 2012-2019 Kai Pastor
3+
* Copyright 2012-2020 Kai Pastor
44
*
55
* This file is part of OpenOrienteering.
66
*
@@ -229,6 +229,9 @@ Q_OBJECT
229229
/** Returns a QPrinterInfo pointer which signals printing to a raster file. */
230230
static const QPrinterInfo* imageTarget();
231231

232+
/** Returns a QPrinterInfo pointer which signals printing to a KML ground overlay. */
233+
static const QPrinterInfo* kmzTarget();
234+
232235
/** Returns a reference to a hash which maps paper sizes to names as C strings.
233236
* These strings are not translated.
234237
*
@@ -361,6 +364,8 @@ Q_OBJECT
361364
* buffer but refers to the logical coordinates of device_painter. */
362365
void drawPage(QPainter* device_painter, const QRectF& page_extent, QImage* page_buffer = nullptr) const;
363366

367+
void drawPage(QPainter* device_painter, const QRectF& page_extent, const QTransform& page_extent_transform, QImage* page_buffer = nullptr) const;
368+
364369
/** Draws the separations as distinct pages to the printer. */
365370
void drawSeparationPages(QPrinter* printer, QPainter* device_painter, const QRectF& page_extent) const;
366371

src/gdal/CMakeLists.txt

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
#
2-
# Copyright 2016-2017 Kai Pastor
2+
# Copyright 2016-2020 Kai Pastor
33
#
44
# This file is part of OpenOrienteering.
55
#
@@ -28,10 +28,12 @@ set(MAPPER_GDAL_HEADERS
2828
)
2929

3030
set(MAPPER_GDAL_SOURCES
31+
gdal_file.cpp
3132
gdal_image_reader.cpp
3233
gdal_manager.cpp
3334
gdal_settings_page.cpp
3435
gdal_template.cpp
36+
kmz_groundoverlay_export.cpp
3537
ogr_file_format.cpp
3638
ogr_template.cpp
3739
mapper-osmconf.ini

src/gdal/gdal_file.cpp

+73
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
/*
2+
* Copyright 2020 Kai Pastor
3+
*
4+
* This file is part of OpenOrienteering.
5+
*
6+
* OpenOrienteering is free software: you can redistribute it and/or modify
7+
* it under the terms of the GNU General Public License as published by
8+
* the Free Software Foundation, either version 3 of the License, or
9+
* (at your option) any later version.
10+
*
11+
* OpenOrienteering is distributed in the hope that it will be useful,
12+
* but WITHOUT ANY WARRANTY; without even the implied warranty of
13+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14+
* GNU General Public License for more details.
15+
*
16+
* You should have received a copy of the GNU General Public License
17+
* along with OpenOrienteering. If not, see <http://www.gnu.org/licenses/>.
18+
*/
19+
20+
#include "gdal_file.h"
21+
22+
#include <cpl_conv.h>
23+
#include <cpl_vsi.h>
24+
#include <gdal.h>
25+
26+
#include <QByteArray>
27+
#include <QString>
28+
29+
30+
namespace OpenOrienteering {
31+
32+
namespace GdalFile {
33+
34+
bool exists(const QByteArray& filepath)
35+
{
36+
VSIStatBufL stat_buf;
37+
return VSIStatExL(filepath, &stat_buf, VSI_STAT_EXISTS_FLAG) == 0;
38+
}
39+
40+
bool isRelative(const QByteArray& filepath)
41+
{
42+
return CPLIsFilenameRelative(filepath) == TRUE;
43+
}
44+
45+
46+
bool isDir(const QByteArray& filepath)
47+
{
48+
VSIStatBufL stat_buf;
49+
return VSIStatExL(filepath, &stat_buf, VSI_STAT_EXISTS_FLAG | VSI_STAT_NATURE_FLAG) == 0
50+
&& VSI_ISDIR(stat_buf.st_mode);
51+
}
52+
53+
bool mkdir(const QByteArray& filepath)
54+
{
55+
// 0777 would be right for POSIX mkdir with umask, but we
56+
// cannot rely on umask for all paths supported by VSIMkdir.
57+
return VSIMkdir(filepath, 0755) == 0;
58+
}
59+
60+
61+
QByteArray tryToFindRelativeTemplateFile(const QByteArray& template_path, const QByteArray& map_path)
62+
{
63+
QByteArray filepath = map_path + '/' + template_path;
64+
if (!exists(filepath))
65+
filepath = QByteArray(CPLGetDirname(map_path)) + '/' + template_path;
66+
if (!exists(filepath))
67+
filepath.clear();
68+
return filepath;
69+
}
70+
71+
} // namespace GdalUtil
72+
73+
} // namespace OpenOrienteering

src/gdal/gdal_file.h

+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
/*
2+
* Copyright 2020 Kai Pastor
3+
*
4+
* This file is part of OpenOrienteering.
5+
*
6+
* OpenOrienteering is free software: you can redistribute it and/or modify
7+
* it under the terms of the GNU General Public License as published by
8+
* the Free Software Foundation, either version 3 of the License, or
9+
* (at your option) any later version.
10+
*
11+
* OpenOrienteering is distributed in the hope that it will be useful,
12+
* but WITHOUT ANY WARRANTY; without even the implied warranty of
13+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14+
* GNU General Public License for more details.
15+
*
16+
* You should have received a copy of the GNU General Public License
17+
* along with OpenOrienteering. If not, see <http://www.gnu.org/licenses/>.
18+
*/
19+
20+
#ifndef OPENORIENTEERING_GDAL_FILE_H
21+
#define OPENORIENTEERING_GDAL_FILE_H
22+
23+
class QByteArray;
24+
25+
namespace OpenOrienteering {
26+
27+
28+
/**
29+
* Utility functions which use GDAL's VSI-aware file API.
30+
*
31+
* Paths must be passed as, and are returned as, UTF-8.
32+
*
33+
* \see https://gdal.org/user/virtual_file_systems.html
34+
*/
35+
namespace GdalFile {
36+
37+
/**
38+
* Checks if a file exists.
39+
*/
40+
bool exists(const QByteArray& filepath);
41+
42+
/**
43+
* Checks if a path is regarded as relative.
44+
*/
45+
bool isRelative(const QByteArray& filepath);
46+
47+
/**
48+
* Checks if a path is an existing directory.
49+
*/
50+
bool isDir(const QByteArray& filepath);
51+
52+
/**
53+
* Creates a directory.
54+
*/
55+
bool mkdir(const QByteArray& filepath);
56+
57+
58+
/**
59+
* GDAL-based implementation of Template::tryToFindRelativeTemplateFile().
60+
*
61+
* Returns an absolute path when the given template path identifies an existing
62+
* file relative to the map path, or an empty value otherwise.
63+
*/
64+
QByteArray tryToFindRelativeTemplateFile(const QByteArray& template_path, const QByteArray& map_path);
65+
66+
67+
} // namespace GdalUtil
68+
69+
} // namespace OpenOrienteering
70+
71+
#endif // OPENORIENTEERING_GDAL_FILE_H

0 commit comments

Comments
 (0)