Compare commits
29 Commits
Author | SHA1 | Date | |
---|---|---|---|
8041aa454c | |||
1ee60b4f96 | |||
ca17ec9e5f | |||
d3781e7f0b | |||
c4e44c8796 | |||
59be4aaf1a | |||
398ef2c4f5 | |||
5df43d3fb5 | |||
ac41483530 | |||
e0000d7299 | |||
9059e6d30d | |||
819a67158a | |||
610dd0a009 | |||
41e8b7094a | |||
0ee28d7f45 | |||
98acd3f75a | |||
a3459e34e0 | |||
d43f58deb9 | |||
73e7a1dbf4 | |||
319cdbf838 | |||
213ab73ceb | |||
db2d81042b | |||
e27bc437fa | |||
3dd21ade19 | |||
46ea3dd257 | |||
0ed630dd20 | |||
9cd82319e9 | |||
335df85fae | |||
f1b52c7005 |
@ -1,4 +1,4 @@
|
||||
version: 13.34.{build}
|
||||
version: 13.35.{build}
|
||||
|
||||
configuration:
|
||||
- Release
|
||||
|
@ -6,6 +6,7 @@ GPS log file formats.
|
||||
* Opens GPX, TCX, FIT, KML, NMEA, IGC, CUP, SIGMA SLF, Suunto SML, LOC, GeoJSON,
|
||||
OziExplorer (PLT, RTE, WPT), Garmin GPI&CSV, TomTom OV2&ITN, ONmove OMD/GHP,
|
||||
TwoNav (TRK, RTE, WPT), GPSDump WPT and geotagged JPEG files.
|
||||
* Opens geo URIs (RFC 5870).
|
||||
* User-definable online maps (OpenStreetMap/Google tiles, WMTS, WMS, TMS,
|
||||
QuadTiles).
|
||||
* Offline maps (MBTiles, OziExplorer maps, TrekBuddy maps/atlases,
|
||||
|
@ -3,7 +3,7 @@ unix:!macx:!android {
|
||||
} else {
|
||||
TARGET = GPXSee
|
||||
}
|
||||
VERSION = 13.34
|
||||
VERSION = 13.35
|
||||
|
||||
|
||||
QT += core \
|
||||
@ -118,6 +118,7 @@ HEADERS += src/common/config.h \
|
||||
src/data/gpsdumpparser.h \
|
||||
src/data/style.h \
|
||||
src/data/twonavparser.h \
|
||||
src/map/IMG/lights.h \
|
||||
src/map/downloader.h \
|
||||
src/map/demloader.h \
|
||||
src/map/ENC/attributes.h \
|
||||
|
@ -193,6 +193,10 @@
|
||||
<file alias="overfalls.png">icons/map/marine/overfalls.png</file>
|
||||
<file alias="boarding-place.png">icons/map/marine/boarding-place.png</file>
|
||||
<file alias="light.png">icons/map/marine/light.png</file>
|
||||
<file alias="light-red.png">icons/map/marine/light-red.png</file>
|
||||
<file alias="light-green.png">icons/map/marine/light-green.png</file>
|
||||
<file alias="light-yellow.png">icons/map/marine/light-yellow.png</file>
|
||||
<file alias="light-white.png">icons/map/marine/light-white.png</file>
|
||||
<file alias="building.png">icons/map/marine/building.png</file>
|
||||
<file alias="fog-signal.png">icons/map/marine/fog-signal.png</file>
|
||||
<file alias="construction.png">icons/map/marine/construction.png</file>
|
||||
@ -210,6 +214,8 @@
|
||||
<file alias="24h-anchorage.png">icons/map/marine/24h-anchorage.png</file>
|
||||
<file alias="dw-anchorage.png">icons/map/marine/dw-anchorage.png</file>
|
||||
<file alias="tanker-anchorage.png">icons/map/marine/tanker-anchorage.png</file>
|
||||
<file alias="nature-reserve-line.png">icons/map/marine/nature-reserve-line.png</file>
|
||||
<file alias="sanctuary-line.png">icons/map/marine/sanctuary-line.png</file>
|
||||
</qresource>
|
||||
|
||||
<!-- Patterns (Mapsforge) -->
|
||||
|
Before Width: | Height: | Size: 4.3 KiB After Width: | Height: | Size: 4.5 KiB |
Before Width: | Height: | Size: 4.4 KiB After Width: | Height: | Size: 4.5 KiB |
Before Width: | Height: | Size: 259 B After Width: | Height: | Size: 4.4 KiB |
BIN
icons/map/marine/light-green.png
Normal file
After Width: | Height: | Size: 466 B |
BIN
icons/map/marine/light-red.png
Normal file
After Width: | Height: | Size: 417 B |
BIN
icons/map/marine/light-white.png
Normal file
After Width: | Height: | Size: 330 B |
BIN
icons/map/marine/light-yellow.png
Normal file
After Width: | Height: | Size: 4.3 KiB |
BIN
icons/map/marine/nature-reserve-line.png
Normal file
After Width: | Height: | Size: 4.4 KiB |
Before Width: | Height: | Size: 139 B After Width: | Height: | Size: 212 B |
BIN
icons/map/marine/sanctuary-line.png
Normal file
After Width: | Height: | Size: 127 B |
@ -21,6 +21,12 @@
|
||||
<data android:scheme="content" android:mimeType="image/jpeg"/>
|
||||
<data android:scheme="content" android:mimeType="image/tiff"/>
|
||||
</intent-filter>
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.VIEW"/>
|
||||
<category android:name="android.intent.category.DEFAULT"/>
|
||||
<category android:name="android.intent.category.BROWSABLE"/>
|
||||
<data android:scheme="geo"/>
|
||||
</intent-filter>
|
||||
|
||||
<meta-data android:name="android.app.lib_name" android:value="-- %%INSERT_APP_LIB_NAME%% --"/>
|
||||
<meta-data android:name="android.app.arguments" android:value="-- %%INSERT_APP_ARGUMENTS%% --"/>
|
||||
|
@ -17,6 +17,7 @@
|
||||
Garmin GPI & CSV, TomTom OV2 & ITN, ONmove OMD/GHP,
|
||||
TwoNav (TRK, RTE, WPT), GPSDump WPT and geotagged JPEG
|
||||
files.</li>
|
||||
<li>Opens geo URIs (RFC 5870).</li>
|
||||
<li>User-definable online maps (OpenStreetMap/Google tiles, WMTS,
|
||||
WMS, TMS, QuadTiles).</li>
|
||||
<li>Offline maps (MBTiles, OziExplorer maps, TrekBuddy maps/atlases,
|
||||
@ -112,5 +113,6 @@
|
||||
<mimetype>application/vnd.iho.s57-catalogue</mimetype>
|
||||
<mimetype>application/vnd.gpsdump.wpt</mimetype>
|
||||
<mimetype>application/vnd.gpstuner.gmi</mimetype>
|
||||
<mimetype>x-scheme-handler/geo</mimetype>
|
||||
</mimetypes>
|
||||
</component>
|
||||
|
@ -11,9 +11,9 @@ Comment[ru]=Программа для просмотра и анализа GPS
|
||||
Comment[sv]=GPS-loggfilsläsare och analysator
|
||||
Comment[tr]=GPS günlük dosyası görüntüleyici ve analizcisi
|
||||
Comment[uk]=Переглядач та аналізатор GPS логів
|
||||
Exec=gpxsee %F
|
||||
Exec=gpxsee %U
|
||||
Icon=gpxsee
|
||||
Terminal=false
|
||||
Type=Application
|
||||
Categories=Graphics;Viewer;Education;Geography;Maps;Sports;Qt
|
||||
MimeType=application/gpx+xml;application/vnd.garmin.tcx+xml;application/vnd.ant.fit;application/vnd.google-earth.kml+xml;application/vnd.fai.igc;application/vnd.nmea.nmea;application/vnd.oziexplorer.plt;application/vnd.oziexplorer.rte;application/vnd.oziexplorer.wpt;application/vnd.groundspeak.loc+xml;application/vnd.sigma.slf+xml;application/geo+json;application/vnd.naviter.seeyou.cup;application/vnd.garmin.gpi;application/vnd.suunto.sml+xml;image/jpeg;text/csv;application/vnd.garmin.img;application/vnd.garmin.jnx;application/vnd.garmin.gmap+xml;image/vnd.maptech.kap;application/vnd.oziexplorer.map;application/vnd.mapbox.mbtiles;application/vnd.twonav.rmap;application/vnd.trekbuddy.tba;application/vnd.gpxsee.map+xml;application/x-tar;image/tiff;application/vnd.google-earth.kmz;application/vnd.alpinequest.aqm;application/vnd.cgtk.gemf;application/vnd.rmaps.sqlite;application/vnd.osmdroid.sqlite;application/vnd.mapsforge.map;application/vnd.tomtom.ov2;application/vnd.tomtom.itn;application/vnd.esri.wld;application/vnd.onmove.omd;application/vnd.onmove.ghp;application/vnd.memory-map.qct;application/vnd.twonav.trk;application/vnd.twonav.rte;application/vnd.twonav.wpt;application/vnd.orux.map+xml;application/vnd.iho.s57-data;application/vnd.iho.s57-catalogue;application/vnd.gpsdump.wpt;application/vnd.gpstuner.gmi
|
||||
MimeType=x-scheme-handler/geo;application/gpx+xml;application/vnd.garmin.tcx+xml;application/vnd.ant.fit;application/vnd.google-earth.kml+xml;application/vnd.fai.igc;application/vnd.nmea.nmea;application/vnd.oziexplorer.plt;application/vnd.oziexplorer.rte;application/vnd.oziexplorer.wpt;application/vnd.groundspeak.loc+xml;application/vnd.sigma.slf+xml;application/geo+json;application/vnd.naviter.seeyou.cup;application/vnd.garmin.gpi;application/vnd.suunto.sml+xml;image/jpeg;text/csv;application/vnd.garmin.img;application/vnd.garmin.jnx;application/vnd.garmin.gmap+xml;image/vnd.maptech.kap;application/vnd.oziexplorer.map;application/vnd.mapbox.mbtiles;application/vnd.twonav.rmap;application/vnd.trekbuddy.tba;application/vnd.gpxsee.map+xml;application/x-tar;image/tiff;application/vnd.google-earth.kmz;application/vnd.alpinequest.aqm;application/vnd.cgtk.gemf;application/vnd.rmaps.sqlite;application/vnd.osmdroid.sqlite;application/vnd.mapsforge.map;application/vnd.tomtom.ov2;application/vnd.tomtom.itn;application/vnd.esri.wld;application/vnd.onmove.omd;application/vnd.onmove.ghp;application/vnd.memory-map.qct;application/vnd.twonav.trk;application/vnd.twonav.rte;application/vnd.twonav.wpt;application/vnd.orux.map+xml;application/vnd.iho.s57-data;application/vnd.iho.s57-catalogue;application/vnd.gpsdump.wpt;application/vnd.gpstuner.gmi
|
||||
|
@ -738,6 +738,20 @@
|
||||
</dict>
|
||||
</array>
|
||||
|
||||
<key>CFBundleURLTypes</key>
|
||||
<array>
|
||||
<dict>
|
||||
<key>CFBundleURLSchemes</key>
|
||||
<array>
|
||||
<string>geo</string>
|
||||
</array>
|
||||
<key>CFBundleURLName</key>
|
||||
<string>org.geouri.geo</string>
|
||||
<key>CFBundleTypeRole</key>
|
||||
<string>Viewer</string>
|
||||
</dict>
|
||||
</array>
|
||||
|
||||
<key>UTImportedTypeDeclarations</key>
|
||||
<array>
|
||||
<dict>
|
||||
|
@ -18,6 +18,18 @@
|
||||
DeleteRegKey HKCR ".${EXT}"
|
||||
!macroend
|
||||
|
||||
; URI association
|
||||
!macro URI_ASSOCIATION_ADD PROTO
|
||||
WriteRegStr HKCR "${PROTO}" "" "URL:${PROTO}"
|
||||
WriteRegStr HKCR "${PROTO}" "URL Protocol" ""
|
||||
WriteRegStr HKCR "${PROTO}\DefaultIcon" "" "$INSTDIR\GPXSee.exe,0"
|
||||
WriteRegStr HKCR "${PROTO}\shell\open\command" "" "$\"$INSTDIR\GPXSee.exe$\" $\"%1$\""
|
||||
!macroend
|
||||
|
||||
!macro URI_ASSOCIATION_REMOVE PROTO
|
||||
DeleteRegKey HKCR "${PROTO}"
|
||||
!macroend
|
||||
|
||||
; Translations
|
||||
!macro LOCALIZATION LANG CODE
|
||||
Section "${LANG}"
|
||||
@ -37,7 +49,7 @@ Unicode true
|
||||
; The name of the installer
|
||||
Name "GPXSee"
|
||||
; Program version
|
||||
!define VERSION "13.34"
|
||||
!define VERSION "13.35"
|
||||
|
||||
; The file to write
|
||||
OutFile "GPXSee-${VERSION}_x64.exe"
|
||||
@ -200,6 +212,8 @@ Section "GPXSee" SEC_APP
|
||||
!insertmacro FILE_ASSOCIATION_ADD "plt" "OziExplorer Track File" 33
|
||||
!insertmacro FILE_ASSOCIATION_ADD "rte" "OziExplorer Route File" 34
|
||||
|
||||
!insertmacro URI_ASSOCIATION_ADD "geo"
|
||||
|
||||
WriteRegStr HKCR "Applications\GPXSee.exe\shell\open\command" "" "$\"$INSTDIR\GPXSee.exe$\" $\"%1$\""
|
||||
WriteRegStr HKCR ".gpx\OpenWithList" "GPXSee.exe" ""
|
||||
WriteRegStr HKCR ".tcx\OpenWithList" "GPXSee.exe" ""
|
||||
@ -426,6 +440,8 @@ Section "Uninstall"
|
||||
!insertmacro FILE_ASSOCIATION_REMOVE "000"
|
||||
!insertmacro FILE_ASSOCIATION_REMOVE "031"
|
||||
|
||||
!insertmacro URI_ASSOCIATION_REMOVE "geo"
|
||||
|
||||
DeleteRegValue HKCR ".gpx\OpenWithList" "GPXSee.exe"
|
||||
DeleteRegValue HKCR ".tcx\OpenWithList" "GPXSee.exe"
|
||||
DeleteRegValue HKCR ".kml\OpenWithList" "GPXSee.exe"
|
||||
|
@ -1069,18 +1069,33 @@ void GUI::openDir()
|
||||
|
||||
bool GUI::openFile(const QString &fileName, bool tryUnknown, int &showError)
|
||||
{
|
||||
QFileInfo fi(fileName);
|
||||
QString canonicalFileName(fi.canonicalFilePath());
|
||||
QString path;
|
||||
|
||||
if (_files.contains(canonicalFileName))
|
||||
QUrl url(fileName);
|
||||
if (url.scheme() == "geo") {
|
||||
if (loadURL(url, showError)) {
|
||||
_fileActionGroup->setEnabled(true);
|
||||
_reloadFileAction->setEnabled(false);
|
||||
return true;
|
||||
} else if (showError)
|
||||
return false;
|
||||
} else if (url.isLocalFile())
|
||||
path = url.toLocalFile();
|
||||
else
|
||||
path = fileName;
|
||||
|
||||
QFileInfo fi(path);
|
||||
QString canonicalPath(fi.canonicalFilePath());
|
||||
|
||||
if (_files.contains(canonicalPath))
|
||||
return true;
|
||||
|
||||
if (!loadFile(fileName, tryUnknown, showError))
|
||||
if (!loadFile(path, tryUnknown, showError))
|
||||
return false;
|
||||
|
||||
_files.append(canonicalFileName);
|
||||
_files.append(canonicalPath);
|
||||
#ifndef Q_OS_ANDROID
|
||||
_browser->setCurrent(fileName);
|
||||
_browser->setCurrent(path);
|
||||
#endif // Q_OS_ANDROID
|
||||
_fileActionGroup->setEnabled(true);
|
||||
// Explicitly enable the reload action as it may be disabled by loadMapDir()
|
||||
@ -1093,12 +1108,42 @@ bool GUI::openFile(const QString &fileName, bool tryUnknown, int &showError)
|
||||
if (_files.count() > 1)
|
||||
_mapView->showExtendedInfo(true);
|
||||
#ifndef Q_OS_ANDROID
|
||||
updateRecentFiles(canonicalFileName);
|
||||
updateRecentFiles(canonicalPath);
|
||||
#endif // Q_OS_ANDROID
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool GUI::loadURL(const QUrl &url, int &showError)
|
||||
{
|
||||
Data data(url);
|
||||
|
||||
if (data.isValid()) {
|
||||
loadData(data);
|
||||
return true;
|
||||
} else {
|
||||
if (showError) {
|
||||
QString error = tr("Error loading geo URI:") + "\n" + url.toString()
|
||||
+ ": " + data.errorString();
|
||||
|
||||
if (showError > 1) {
|
||||
QMessageBox message(QMessageBox::Critical, APP_NAME, error,
|
||||
QMessageBox::Ok, this);
|
||||
QCheckBox checkBox(tr("Don't show again"));
|
||||
message.setCheckBox(&checkBox);
|
||||
message.exec();
|
||||
if (checkBox.isChecked())
|
||||
showError = 0;
|
||||
} else
|
||||
QMessageBox::critical(this, APP_NAME, error);
|
||||
} else
|
||||
qWarning("%s: %s", qUtf8Printable(url.toString()),
|
||||
qUtf8Printable(data.errorString()));
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
bool GUI::loadFile(const QString &fileName, bool tryUnknown, int &showError)
|
||||
{
|
||||
Data data(fileName, tryUnknown);
|
||||
@ -1850,7 +1895,13 @@ bool GUI::loadMapNode(const TreeNode<Map*> &node, MapAction *&action,
|
||||
|
||||
bool GUI::loadMap(const QString &fileName, MapAction *&action, int &showError)
|
||||
{
|
||||
TreeNode<Map*> maps(MapList::loadMaps(fileName, _mapView->inputProjection()));
|
||||
QString path;
|
||||
QUrl url(fileName);
|
||||
|
||||
path = url.isLocalFile() ? url.toLocalFile() : fileName;
|
||||
|
||||
|
||||
TreeNode<Map*> maps(MapList::loadMaps(path, _mapView->inputProjection()));
|
||||
QList<QAction*> existingActions(_mapsActionGroup->actions());
|
||||
|
||||
return loadMapNode(maps, action, existingActions, showError);
|
||||
|
@ -158,6 +158,7 @@ private:
|
||||
#endif // Q_OS_ANDROID
|
||||
bool openPOIFile(const QString &fileName);
|
||||
bool loadFile(const QString &fileName, bool tryUnknown, int &showError);
|
||||
bool loadURL(const QUrl &url, int &showError);
|
||||
void loadData(const Data &data);
|
||||
bool loadMapNode(const TreeNode<Map*> &node, MapAction *&action,
|
||||
const QList<QAction*> &existingActions, int &showError);
|
||||
|
@ -2,6 +2,7 @@
|
||||
#include <QFile>
|
||||
#include <QFileInfo>
|
||||
#include "common/util.h"
|
||||
#include "map/crs.h"
|
||||
#include "gpxparser.h"
|
||||
#include "tcxparser.h"
|
||||
#include "csvparser.h"
|
||||
@ -154,6 +155,70 @@ Data::Data(const QString &fileName, bool tryUnknown)
|
||||
}
|
||||
}
|
||||
|
||||
Data::Data(const QUrl &url)
|
||||
{
|
||||
bool caOk, cbOk, ccOk;
|
||||
Projection proj(GCS::WGS84());
|
||||
|
||||
_valid = false;
|
||||
|
||||
QStringList parts(url.path().split(';'));
|
||||
if (parts.size() < 1) {
|
||||
_errorString = "Syntax error";
|
||||
return;
|
||||
}
|
||||
QStringList coords(parts.at(0).split(','));
|
||||
if (coords.size() < 2 || coords.size() > 3) {
|
||||
_errorString = "Syntax error";
|
||||
return;
|
||||
}
|
||||
double ca = coords.at(0).toDouble(&caOk);
|
||||
double cb = coords.at(1).toDouble(&cbOk);
|
||||
double cc = NAN;
|
||||
if (!(caOk && cbOk)) {
|
||||
_errorString = "Invalid coordinates";
|
||||
return;
|
||||
}
|
||||
if (coords.size() > 2) {
|
||||
cc = coords.at(2).toDouble(&ccOk);
|
||||
if (!ccOk) {
|
||||
_errorString = "Invalid elevation";
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (parts.size() > 1) {
|
||||
QStringList crsp(parts.at(1).split('='));
|
||||
if (crsp.size() != 2) {
|
||||
_errorString = "Syntax error";
|
||||
return;
|
||||
}
|
||||
if (!crsp.at(0).compare("crs", Qt::CaseInsensitive)) {
|
||||
if (crsp.at(1).compare("wgs84", Qt::CaseInsensitive)) {
|
||||
proj = CRS::projection(crsp.at(1));
|
||||
if (!proj.isValid()) {
|
||||
_errorString = "Unknown CRS";
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
CoordinateSystem::AxisOrder ao = proj.coordinateSystem().axisOrder();
|
||||
PointD p(ao == CoordinateSystem::XY ? PointD(ca, cb) : PointD(cb, ca));
|
||||
Coordinates c(proj.xy2ll(p));
|
||||
if (!c.isValid()) {
|
||||
_errorString = "Invalid coordinates";
|
||||
return;
|
||||
}
|
||||
|
||||
Waypoint w(c);
|
||||
w.setElevation(cc);
|
||||
_waypoints.append(w);
|
||||
|
||||
_valid = true;
|
||||
}
|
||||
|
||||
QString Data::formats()
|
||||
{
|
||||
return
|
||||
|
@ -14,6 +14,7 @@ class Data
|
||||
{
|
||||
public:
|
||||
Data(const QString &fileName, bool tryUnknown = true);
|
||||
Data(const QUrl &url);
|
||||
|
||||
bool isValid() const {return _valid;}
|
||||
const QString &errorString() const {return _errorString;}
|
||||
|
@ -6,6 +6,7 @@
|
||||
#define CATDIS 21
|
||||
#define CATHAF 30
|
||||
#define CATLMK 35
|
||||
#define CATLIT 37
|
||||
#define CATMFA 38
|
||||
#define CATMOR 40
|
||||
#define CATTRK 54
|
||||
@ -16,16 +17,21 @@
|
||||
#define CATWAT 69
|
||||
#define CATWED 70
|
||||
#define CATWRK 71
|
||||
#define COLOUR 75
|
||||
#define COMCHA 77
|
||||
#define CURVEL 84
|
||||
#define DRVAL1 87
|
||||
#define ELEVAT 90
|
||||
#define FUNCTN 94
|
||||
#define LITVIS 108
|
||||
#define OBJNAM 116
|
||||
#define ORIENT 117
|
||||
#define RESTRN 131
|
||||
#define SECTR1 136
|
||||
#define SECTR2 137
|
||||
#define TRAFIC 172
|
||||
#define VALDCO 174
|
||||
#define VALNMR 178
|
||||
#define VERCLR 181
|
||||
#define WATLEV 187
|
||||
|
||||
|
@ -173,19 +173,41 @@ static bool polygonCb(const MapData::Poly *polygon, void *context)
|
||||
static bool polygonPointCb(const MapData::Poly *polygon, void *context)
|
||||
{
|
||||
QList<MapData::Point> *points = (QList<MapData::Point>*)context;
|
||||
uint baseType = polygon->type()>>16;
|
||||
uint type = polygon->type();
|
||||
uint baseType = type>>16;
|
||||
|
||||
if (!polygon->label().isEmpty() || baseType == TSSLPT || baseType == RCTLPT
|
||||
|| baseType == I_TRNBSN
|
||||
|| polygon->type() == SUBTYPE(ACHARE, 2)
|
||||
|| polygon->type() == SUBTYPE(ACHARE, 3)
|
||||
|| polygon->type() == SUBTYPE(ACHARE, 9)
|
||||
|| polygon->type() == SUBTYPE(I_ACHARE, 2)
|
||||
|| polygon->type() == SUBTYPE(I_ACHARE, 3)
|
||||
|| polygon->type() == SUBTYPE(I_ACHARE, 9)
|
||||
|| polygon->type() == SUBTYPE(I_BERTHS, 6))
|
||||
points->append(MapData::Point(polygon->type(), polygon->bounds().center(),
|
||||
polygon->label(), polygon->param()));
|
||||
if (baseType == TSSLPT || baseType == RCTLPT || baseType == I_TRNBSN
|
||||
|| baseType == BRIDGE || baseType == I_BRIDGE || baseType == BUAARE
|
||||
|| baseType == LNDARE || baseType == LNDRGN
|
||||
|| type == SUBTYPE(ACHARE, 2) || type == SUBTYPE(I_ACHARE, 2)
|
||||
|| type == SUBTYPE(ACHARE, 3) || type == SUBTYPE(I_ACHARE, 3)
|
||||
|| type == SUBTYPE(ACHARE, 9) || type == SUBTYPE(I_ACHARE, 9)
|
||||
|| type == SUBTYPE(I_BERTHS, 6)
|
||||
|| type == SUBTYPE(RESARE, 1) || type == SUBTYPE(I_RESARE, 1)
|
||||
|| type == SUBTYPE(RESARE, 2) || type == SUBTYPE(I_RESARE, 2)
|
||||
|| type == SUBTYPE(RESARE, 4) || type == SUBTYPE(I_RESARE, 4)
|
||||
|| type == SUBTYPE(RESARE, 5) || type == SUBTYPE(I_RESARE, 5)
|
||||
|| type == SUBTYPE(RESARE, 6) || type == SUBTYPE(I_RESARE, 6)
|
||||
|| type == SUBTYPE(RESARE, 7) || type == SUBTYPE(I_RESARE, 7)
|
||||
|| type == SUBTYPE(RESARE, 9) || type == SUBTYPE(I_RESARE, 9)
|
||||
|| type == SUBTYPE(RESARE, 12) || type == SUBTYPE(I_RESARE, 12)
|
||||
|| type == SUBTYPE(RESARE, 17) || type == SUBTYPE(I_RESARE, 17)
|
||||
|| type == SUBTYPE(RESARE, 22) || type == SUBTYPE(I_RESARE, 22)
|
||||
|| type == SUBTYPE(RESARE, 23) || type == SUBTYPE(I_RESARE, 23))
|
||||
points->append(MapData::Point(baseType, polygon->bounds().center(),
|
||||
polygon->attributes(), polygon->HUNI(), true));
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool linePointCb(const MapData::Line *line, void *context)
|
||||
{
|
||||
QList<MapData::Point> *points = (QList<MapData::Point>*)context;
|
||||
uint baseType = line->type()>>16;
|
||||
|
||||
if (baseType == RDOCAL || baseType == I_RDOCAL)
|
||||
points->append(MapData::Point(baseType, line->bounds().center(),
|
||||
line->attributes(), 1));
|
||||
|
||||
return true;
|
||||
}
|
||||
@ -207,10 +229,8 @@ static Coordinates point(const ISO8211::Record &r, uint COMF)
|
||||
return coordinates(x, y, COMF);
|
||||
}
|
||||
|
||||
static uint depthLevel(const QByteArray &ba)
|
||||
static uint depthLevel(double minDepth)
|
||||
{
|
||||
double minDepth = ba.isEmpty() ? -1 : ba.toDouble();
|
||||
|
||||
if (minDepth < 0)
|
||||
return 0;
|
||||
else if (minDepth < 2)
|
||||
@ -275,72 +295,171 @@ static QString weed(uint type)
|
||||
}
|
||||
}
|
||||
|
||||
MapData::Point::Point(uint type, const Coordinates &c, const QString &label,
|
||||
const QVector<QByteArray> ¶ms) : _type(type), _pos(c), _label(label)
|
||||
static uint restrictionCategory(uint type, const MapData::Attributes &attr)
|
||||
{
|
||||
_id = ((quint64)order(type))<<32 | (uint)qHash(c);
|
||||
uint catrea = attr.value(CATREA).toUInt();
|
||||
|
||||
if (type>>16 == I_DISMAR && params.size()) {
|
||||
_label = hUnits((type>>8)&0xFF) + " " + QString::fromLatin1(params.at(0));
|
||||
_type = SUBTYPE(I_DISMAR, type & 0xFF);
|
||||
} else if ((type>>16 == I_RDOCAL || type>>16 == RDOCAL) && params.size() > 1) {
|
||||
if (!params.at(1).isEmpty())
|
||||
_label = QString("VHF ") + QString::fromLatin1(params.at(1));
|
||||
_param = QVariant(params.at(0).toDouble());
|
||||
} else if (type>>16 == CURENT && params.size() > 1) {
|
||||
if (!params.at(1).isEmpty())
|
||||
_label = QString::fromLatin1(params.at(1))
|
||||
+ QString::fromUtf8("\xE2\x80\x89kt");
|
||||
_param = QVariant(params.at(0).toDouble());
|
||||
} else if (type>>16 == I_SISTAT || type>>16 == SISTAT) {
|
||||
if (!catrea) {
|
||||
uint restrn = attr.value(
|
||||
(type == RESARE) ? RESTRN : I_RESTRN).toUInt();
|
||||
|
||||
if (restrn == 1)
|
||||
return 2;
|
||||
else if (restrn == 7)
|
||||
return 17;
|
||||
else
|
||||
return 0;
|
||||
} else
|
||||
return catrea;
|
||||
}
|
||||
|
||||
MapData::Point::Point(uint type, const Coordinates &c, const QString &label)
|
||||
: _type(SUBTYPE(type, 0)), _pos(c), _label(label), _polygon(false)
|
||||
{
|
||||
_id = ((quint64)order(_type))<<32 | (uint)qHash(c);
|
||||
}
|
||||
|
||||
MapData::Point::Point(uint type, const Coordinates &c, const Attributes &attr,
|
||||
uint HUNI, bool polygon) : _pos(c), _attr(attr), _polygon(polygon)
|
||||
{
|
||||
uint subtype = 0;
|
||||
|
||||
if (type == HRBFAC)
|
||||
subtype = CATHAF;
|
||||
else if (type == I_HRBFAC)
|
||||
subtype = I_CATHAF;
|
||||
else if (type == LNDMRK)
|
||||
subtype = CATLMK;
|
||||
else if (type == WRECKS)
|
||||
subtype = CATWRK;
|
||||
else if (type == MORFAC)
|
||||
subtype = CATMOR;
|
||||
else if (type == UWTROC)
|
||||
subtype = WATLEV;
|
||||
else if (type == BUAARE)
|
||||
subtype = CATBUA;
|
||||
else if (type == SMCFAC)
|
||||
subtype = CATSCF;
|
||||
else if (type == BUISGL)
|
||||
subtype = FUNCTN;
|
||||
else if (type == WATTUR)
|
||||
subtype = CATWAT;
|
||||
else if (type == RDOCAL)
|
||||
subtype = TRAFIC;
|
||||
else if (type == I_RDOCAL)
|
||||
subtype = TRAFIC;
|
||||
else if (type == SILTNK)
|
||||
subtype = CATSIL;
|
||||
else if (type == WEDKLP)
|
||||
subtype = CATWED;
|
||||
else if (type == LIGHTS)
|
||||
subtype = CATLIT;
|
||||
else if (type == I_DISMAR)
|
||||
subtype = CATDIS;
|
||||
else if (type == I_BERTHS)
|
||||
subtype = I_CATBRT;
|
||||
else if (type == ACHARE)
|
||||
subtype = CATACH;
|
||||
else if (type == I_ACHARE)
|
||||
subtype = I_CATACH;
|
||||
|
||||
QList<QByteArray> list(_attr.value(subtype).split(','));
|
||||
std::sort(list.begin(), list.end());
|
||||
_type = (type == RESARE || type == I_RESARE)
|
||||
? SUBTYPE(type, restrictionCategory(type, _attr))
|
||||
: SUBTYPE(type, list.first().toUInt());
|
||||
_id = ((quint64)order(_type))<<32 | (uint)qHash(c);
|
||||
_label = QString::fromLatin1(_attr.value(OBJNAM));
|
||||
|
||||
if (type == I_DISMAR) {
|
||||
if (_attr.contains(I_WTWDIS) && _attr.contains(I_HUNITS))
|
||||
_label = hUnits(_attr.value(I_HUNITS).toUInt()) + " "
|
||||
+ QString::fromLatin1(_attr.value(I_WTWDIS));
|
||||
} else if (type == I_RDOCAL || type == RDOCAL) {
|
||||
QByteArray cc(_attr.value(COMCHA));
|
||||
if (!cc.isEmpty())
|
||||
_label = QString("VHF ") + QString::fromLatin1(cc);
|
||||
} else if (type == CURENT) {
|
||||
QByteArray cv(_attr.value(CURVEL));
|
||||
if (!cv.isEmpty())
|
||||
_label = QString::fromLatin1(cv) + QString::fromUtf8("\xE2\x80\x89kt");
|
||||
} else if (type == SISTAT) {
|
||||
if (_label.isEmpty() && _attr.contains(CATSIT))
|
||||
_label = sistat(_attr.value(CATSIT).toUInt());
|
||||
} else if (type == I_SISTAT) {
|
||||
if (_label.isEmpty() && _attr.contains(I_CATSIT))
|
||||
_label = sistat(_attr.value(I_CATSIT).toUInt());
|
||||
} else if (type == WEDKLP) {
|
||||
if (_label.isEmpty())
|
||||
_label = sistat(type & 0xFF);
|
||||
_type = TYPE(SISTAT);
|
||||
} else if (type>>16 == WEDKLP) {
|
||||
_label = weed(_type & 0xFF);
|
||||
} else if (type == LNDELV) {
|
||||
if (_label.isEmpty())
|
||||
_label = weed(type & 0xFF);
|
||||
} else if (type>>16 == LNDELV && params.size()) {
|
||||
if (_label.isEmpty())
|
||||
_label = QString::fromLatin1(params.at(0))
|
||||
_label = QString::fromLatin1(_attr.value(ELEVAT))
|
||||
+ QString::fromUtf8("\xE2\x80\x89m");
|
||||
else
|
||||
_label += "\n(" + QString::fromLatin1(params.at(0))
|
||||
_label += "\n(" + QString::fromLatin1(_attr.value(ELEVAT))
|
||||
+ "\xE2\x80\x89m)";
|
||||
} else if ((type == TYPE(TSSLPT) || type == TYPE(RCTLPT)) && params.size())
|
||||
_param = QVariant(params.at(0).toDouble());
|
||||
}
|
||||
|
||||
MapData::Point::Point(uint type, const Coordinates &c, const QString &label,
|
||||
const QVariant ¶m) : _type(type), _pos(c), _label(label), _param(param)
|
||||
{
|
||||
_id = ((quint64)order(type))<<32 | (uint)qHash(c);
|
||||
}
|
||||
|
||||
MapData::Poly::Poly(uint type, const Polygon &path, const QString &label,
|
||||
const QVector<QByteArray> ¶ms, uint HUNI) : _type(type), _path(path)
|
||||
{
|
||||
if (type == TYPE(DEPARE) && params.size())
|
||||
_type = SUBTYPE(DEPARE, depthLevel(params.at(0)));
|
||||
else if ((type == TYPE(TSSLPT) || type == TYPE(RCTLPT)) && params.size())
|
||||
_param = QVariant(params.at(0).toDouble());
|
||||
else if ((type == TYPE(BRIDGE) || type == TYPE(I_BRIDGE))
|
||||
&& params.size()) {
|
||||
double clr = params.at(0).toDouble();
|
||||
} else if (type == BRIDGE || type == I_BRIDGE) {
|
||||
double clr = _attr.value(VERCLR).toDouble();
|
||||
if (clr > 0) {
|
||||
_label = QString::fromUtf8("\xE2\x86\x95") + UNIT_SPACE
|
||||
+ QString::number(clr) + UNIT_SPACE + hUnits(HUNI);
|
||||
}
|
||||
} else if (type>>16 == LNDARE || type>>16 == SEAARE || type>>16 == BERTHS
|
||||
|| type>>16 == I_BERTHS || type>>16 == BUAARE)
|
||||
_label = label;
|
||||
}
|
||||
}
|
||||
|
||||
MapData::Poly::Poly(uint type, const Polygon &path, const Attributes &attr,
|
||||
uint HUNI) : _path(path), _attr(attr), _HUNI(HUNI)
|
||||
{
|
||||
uint subtype = 0;
|
||||
|
||||
if (type == ACHARE)
|
||||
subtype = CATACH;
|
||||
else if (type == I_ACHARE)
|
||||
subtype = I_CATACH;
|
||||
else if (type == HRBFAC)
|
||||
subtype = CATHAF;
|
||||
else if (type == MARKUL)
|
||||
subtype = CATMFA;
|
||||
else if (type == I_BERTHS)
|
||||
subtype = I_CATBRT;
|
||||
|
||||
switch (type) {
|
||||
case DEPARE:
|
||||
_type = SUBTYPE(type, depthLevel(_attr.value(DRVAL1).toDouble()));
|
||||
break;
|
||||
case RESARE:
|
||||
case I_RESARE:
|
||||
_type = SUBTYPE(type, restrictionCategory(type, attr));
|
||||
break;
|
||||
default:
|
||||
_type = SUBTYPE(type, _attr.value(subtype).toUInt());
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
MapData::Line::Line(uint type, const QVector<Coordinates> &path,
|
||||
const QString &label, const QVector<QByteArray> ¶ms)
|
||||
: _type(type), _path(path), _label(label)
|
||||
const Attributes &attr) : _path(path), _attr(attr)
|
||||
{
|
||||
if ((type == TYPE(DEPCNT) || type == TYPE(LNDELV)) && params.size())
|
||||
_label = QString::fromLatin1(params.at(0));
|
||||
uint subtype = 0;
|
||||
|
||||
if (type == RECTRC)
|
||||
subtype = CATTRK;
|
||||
else if (type == RCRTCL)
|
||||
subtype = CATTRK;
|
||||
else if (type == RDOCAL)
|
||||
subtype = TRAFIC;
|
||||
else if (type == I_RDOCAL)
|
||||
subtype = TRAFIC;
|
||||
|
||||
_type = SUBTYPE(type, _attr.value(subtype).toUInt());
|
||||
|
||||
if (type == DEPCNT)
|
||||
_label = QString::fromLatin1(_attr.value(VALDCO));
|
||||
else if (type == LNDELV)
|
||||
_label = QString::fromLatin1(_attr.value(ELEVAT));
|
||||
else
|
||||
_label = QString::fromLatin1(_attr.value(OBJNAM));
|
||||
}
|
||||
|
||||
RectC MapData::Line::bounds() const
|
||||
@ -581,169 +700,46 @@ Polygon MapData::polyGeometry(const ISO8211::Record &r, const RecordMap &vc,
|
||||
return path;
|
||||
}
|
||||
|
||||
MapData::Attr MapData::pointAttr(const ISO8211::Record &r, uint OBJL)
|
||||
MapData::Attributes MapData::attributes(const ISO8211::Record &r)
|
||||
{
|
||||
QString label;
|
||||
QVector<QByteArray> params(2);
|
||||
uint subtype = 0;
|
||||
Attributes attr;
|
||||
|
||||
const ISO8211::Field *ATTF = ISO8211::field(r, "ATTF");
|
||||
if (!(ATTF && ATTF->data().at(0).size() == 2))
|
||||
return Attr();
|
||||
return attr;
|
||||
|
||||
for (int i = 0; i < ATTF->data().size(); i++) {
|
||||
const QVector<QVariant> &av = ATTF->data().at(i);
|
||||
uint key = av.at(0).toUInt();
|
||||
|
||||
if (key == OBJNAM)
|
||||
label = QString::fromLatin1(av.at(1).toByteArray());
|
||||
|
||||
if ((OBJL == HRBFAC && key == CATHAF)
|
||||
|| (OBJL == I_HRBFAC && key == I_CATHAF)
|
||||
|| (OBJL == LNDMRK && key == CATLMK)
|
||||
|| (OBJL == WRECKS && key == CATWRK)
|
||||
|| (OBJL == MORFAC && key == CATMOR)
|
||||
|| (OBJL == UWTROC && key == WATLEV)
|
||||
|| (OBJL == BUAARE && key == CATBUA)
|
||||
|| (OBJL == SMCFAC && key == CATSCF)
|
||||
|| (OBJL == BUISGL && key == FUNCTN)
|
||||
|| (OBJL == WATTUR && key == CATWAT)
|
||||
|| (OBJL == SISTAT && key == CATSIT)
|
||||
|| (OBJL == I_SISTAT && key == I_CATSIT)
|
||||
|| (OBJL == RDOCAL && key == TRAFIC)
|
||||
|| (OBJL == I_RDOCAL && key == TRAFIC)
|
||||
|| (OBJL == SILTNK && key == CATSIL)
|
||||
|| (OBJL == WEDKLP && key == CATWED))
|
||||
subtype = av.at(1).toByteArray().toUInt();
|
||||
else if (OBJL == I_DISMAR && key == CATDIS)
|
||||
subtype |= av.at(1).toByteArray().toUInt();
|
||||
else if (OBJL == I_DISMAR && key == I_HUNITS)
|
||||
subtype |= av.at(1).toByteArray().toUInt() << 8;
|
||||
|
||||
if ((OBJL == I_DISMAR && key == I_WTWDIS)
|
||||
|| (OBJL == RDOCAL && key == ORIENT)
|
||||
|| (OBJL == I_RDOCAL && key == ORIENT)
|
||||
|| (OBJL == CURENT && key == ORIENT)
|
||||
|| (OBJL == LNDELV && key == ELEVAT)
|
||||
|| (OBJL == TSSLPT && key == ORIENT)
|
||||
|| (OBJL == RCTLPT && key == ORIENT))
|
||||
params[0] = av.at(1).toByteArray();
|
||||
if ((OBJL == I_RDOCAL && key == COMCHA)
|
||||
|| (OBJL == RDOCAL && key == COMCHA)
|
||||
|| (OBJL == CURENT && key == CURVEL))
|
||||
params[1] = av.at(1).toByteArray();
|
||||
attr.insert(av.at(0).toUInt(), av.at(1).toByteArray());
|
||||
}
|
||||
|
||||
return Attr(subtype, label, params);
|
||||
}
|
||||
|
||||
MapData::Attr MapData::lineAttr(const ISO8211::Record &r, uint OBJL)
|
||||
{
|
||||
QString label;
|
||||
QVector<QByteArray> params(1);
|
||||
uint subtype = 0;
|
||||
|
||||
const ISO8211::Field *ATTF = ISO8211::field(r, "ATTF");
|
||||
if (!(ATTF && ATTF->data().at(0).size() == 2))
|
||||
return Attr();
|
||||
|
||||
for (int i = 0; i < ATTF->data().size(); i++) {
|
||||
const QVector<QVariant> &av = ATTF->data().at(i);
|
||||
uint key = av.at(0).toUInt();
|
||||
|
||||
if (key == OBJNAM)
|
||||
label = QString::fromLatin1(av.at(1).toByteArray());
|
||||
|
||||
if ((OBJL == RECTRC || OBJL == RCRTCL) && key == CATTRK)
|
||||
subtype = av.at(1).toByteArray().toUInt();
|
||||
|
||||
if ((OBJL == DEPCNT && key == VALDCO)
|
||||
|| (OBJL == LNDELV && key == ELEVAT))
|
||||
params[0] = av.at(1).toByteArray();
|
||||
}
|
||||
|
||||
return Attr(subtype, label, params);
|
||||
}
|
||||
|
||||
MapData::Attr MapData::polyAttr(const ISO8211::Record &r, uint OBJL)
|
||||
{
|
||||
QString label;
|
||||
QVector<QByteArray> params(1);
|
||||
uint subtype = 0;
|
||||
|
||||
const ISO8211::Field *ATTF = ISO8211::field(r, "ATTF");
|
||||
if (!(ATTF && ATTF->data().at(0).size() == 2))
|
||||
return Attr();
|
||||
|
||||
for (int i = 0; i < ATTF->data().size(); i++) {
|
||||
const QVector<QVariant> &av = ATTF->data().at(i);
|
||||
uint key = av.at(0).toUInt();
|
||||
|
||||
if (key == OBJNAM)
|
||||
label = QString::fromLatin1(av.at(1).toByteArray());
|
||||
|
||||
if ((OBJL == RESARE && key == CATREA)
|
||||
|| (OBJL == I_RESARE && key == CATREA)
|
||||
|| (OBJL == ACHARE && key == CATACH)
|
||||
|| (OBJL == I_ACHARE && key == I_CATACH)
|
||||
|| (OBJL == HRBFAC && key == CATHAF)
|
||||
|| (OBJL == MARKUL && key == CATMFA)
|
||||
|| (OBJL == I_BERTHS && key == I_CATBRT))
|
||||
subtype = av.at(1).toByteArray().toUInt();
|
||||
else if ((OBJL == RESARE && key == RESTRN)
|
||||
|| (OBJL == I_RESARE && key == I_RESTRN)) {
|
||||
if (av.at(1).toByteArray().toUInt() == 1)
|
||||
subtype = 2;
|
||||
if (av.at(1).toByteArray().toUInt() == 7)
|
||||
subtype = 17;
|
||||
}
|
||||
|
||||
if ((OBJL == TSSLPT && key == ORIENT)
|
||||
|| (OBJL == RCTLPT && key == ORIENT)
|
||||
|| (OBJL == DEPARE && key == DRVAL1))
|
||||
params[0] = av.at(1).toByteArray();
|
||||
if ((OBJL == BRIDGE && key == VERCLR)
|
||||
|| (OBJL == I_BRIDGE && key == VERCLR))
|
||||
params[0] = av.at(1).toByteArray();
|
||||
}
|
||||
|
||||
return Attr(subtype, label, params);
|
||||
return attr;
|
||||
}
|
||||
|
||||
MapData::Point *MapData::pointObject(const Sounding &s)
|
||||
{
|
||||
return new Point(TYPE(SOUNDG), s.c, QString::number(s.depth),
|
||||
QVector<QByteArray>());
|
||||
return new Point(SOUNDG, s.c, QString::number(s.depth));
|
||||
}
|
||||
|
||||
MapData::Point *MapData::pointObject(const ISO8211::Record &r,
|
||||
const RecordMap &vi, const RecordMap &vc, uint COMF, uint OBJL)
|
||||
const RecordMap &vi, const RecordMap &vc, uint COMF, uint OBJL, uint HUNI)
|
||||
{
|
||||
Coordinates c(pointGeometry(r, vi, vc, COMF));
|
||||
Attr attr(pointAttr(r, OBJL));
|
||||
|
||||
return (c.isNull() ? 0 : new Point(SUBTYPE(OBJL,attr.subtype()), c,
|
||||
attr.label(), attr.params()));
|
||||
return (c.isNull() ? 0 : new Point(OBJL, c, attributes(r), HUNI));
|
||||
}
|
||||
|
||||
MapData::Line *MapData::lineObject(const ISO8211::Record &r,
|
||||
const RecordMap &vc, const RecordMap &ve, uint COMF, uint OBJL)
|
||||
{
|
||||
QVector<Coordinates> path(lineGeometry(r, vc, ve, COMF));
|
||||
Attr attr(lineAttr(r, OBJL));
|
||||
|
||||
return (path.isEmpty() ? 0 : new Line(SUBTYPE(OBJL, attr.subtype()), path,
|
||||
attr.label(), attr.params()));
|
||||
return (path.isEmpty() ? 0 : new Line(OBJL, path, attributes(r)));
|
||||
}
|
||||
|
||||
MapData::Poly *MapData::polyObject(const ISO8211::Record &r,
|
||||
const RecordMap &vc, const RecordMap &ve, uint COMF, uint OBJL, uint HUNI)
|
||||
{
|
||||
Polygon path(polyGeometry(r, vc, ve, COMF));
|
||||
Attr attr(polyAttr(r, OBJL));
|
||||
|
||||
return (path.isEmpty() ? 0 : new Poly(SUBTYPE(OBJL, attr.subtype()), path,
|
||||
attr.label(), attr.params(), HUNI));
|
||||
return (path.isEmpty() ? 0 : new Poly(OBJL, path, attributes(r), HUNI));
|
||||
}
|
||||
|
||||
bool MapData::processRecord(const ISO8211::Record &record,
|
||||
@ -828,7 +824,7 @@ MapData::MapData(const QString &path)
|
||||
_points.Insert(min, max, point);
|
||||
}
|
||||
} else {
|
||||
if ((point = pointObject(r, vi, vc, COMF, OBJL))) {
|
||||
if ((point = pointObject(r, vi, vc, COMF, OBJL, HUNI))) {
|
||||
pointBounds(point->pos(), min, max);
|
||||
_points.Insert(min, max, point);
|
||||
} else
|
||||
@ -875,6 +871,7 @@ void MapData::points(const RectC &rect, QList<Point> *points) const
|
||||
rectcBounds(rect, min, max);
|
||||
_points.Search(min, max, pointCb, points);
|
||||
_areas.Search(min, max, polygonPointCb, points);
|
||||
_lines.Search(min, max, linePointCb, points);
|
||||
}
|
||||
|
||||
void MapData::lines(const RectC &rect, QList<Line> *lines) const
|
||||
|
@ -11,51 +11,53 @@ namespace ENC {
|
||||
class MapData
|
||||
{
|
||||
public:
|
||||
typedef QMap<uint, QByteArray> Attributes;
|
||||
|
||||
class Poly {
|
||||
public:
|
||||
Poly(uint type, const Polygon &path, const QString &label,
|
||||
const QVector<QByteArray> ¶ms, uint HUNI);
|
||||
Poly(uint type, const Polygon &path, const Attributes &attr, uint HUNI);
|
||||
|
||||
RectC bounds() const {return _path.boundingRect();}
|
||||
const Polygon &path() const {return _path;}
|
||||
uint type() const {return _type;}
|
||||
const QString &label() const {return _label;}
|
||||
const QVariant ¶m() const {return _param;}
|
||||
const Attributes &attributes() const {return _attr;}
|
||||
uint HUNI() const {return _HUNI;}
|
||||
|
||||
private:
|
||||
uint _type;
|
||||
Polygon _path;
|
||||
QString _label;
|
||||
QVariant _param;
|
||||
Attributes _attr;
|
||||
uint _HUNI;
|
||||
};
|
||||
|
||||
class Line {
|
||||
public:
|
||||
Line(uint type, const QVector<Coordinates> &path, const QString &label,
|
||||
const QVector<QByteArray> ¶ms);
|
||||
Line(uint type, const QVector<Coordinates> &path, const Attributes &attr);
|
||||
|
||||
RectC bounds() const;
|
||||
const QVector<Coordinates> &path() const {return _path;}
|
||||
uint type() const {return _type;}
|
||||
const QString &label() const {return _label;}
|
||||
const Attributes &attributes() const {return _attr;}
|
||||
|
||||
private:
|
||||
uint _type;
|
||||
QVector<Coordinates> _path;
|
||||
QString _label;
|
||||
Attributes _attr;
|
||||
};
|
||||
|
||||
class Point {
|
||||
public:
|
||||
Point(uint type, const Coordinates &c, const QString &label,
|
||||
const QVector<QByteArray> ¶ms);
|
||||
Point(uint type, const Coordinates &c, const QString &label,
|
||||
const QVariant ¶m);
|
||||
Point(uint type, const Coordinates &c, const Attributes &attr,
|
||||
uint HUNI, bool polygon = false);
|
||||
Point(uint type, const Coordinates &s, const QString &label);
|
||||
|
||||
const Coordinates &pos() const {return _pos;}
|
||||
uint type() const {return _type;}
|
||||
const QString &label() const {return _label;}
|
||||
const QVariant ¶m() const {return _param;}
|
||||
const Attributes &attributes() const {return _attr;}
|
||||
bool polygon() const {return _polygon;}
|
||||
|
||||
bool operator<(const Point &other) const
|
||||
{return _id < other._id;}
|
||||
@ -65,7 +67,8 @@ public:
|
||||
Coordinates _pos;
|
||||
QString _label;
|
||||
quint64 _id;
|
||||
QVariant _param;
|
||||
Attributes _attr;
|
||||
bool _polygon;
|
||||
};
|
||||
|
||||
MapData(const QString &path);
|
||||
@ -76,23 +79,6 @@ public:
|
||||
void points(const RectC &rect, QList<Point> *points) const;
|
||||
|
||||
private:
|
||||
class Attr {
|
||||
public:
|
||||
Attr() : _subtype(0) {}
|
||||
Attr(uint subtype, const QString &label,
|
||||
const QVector<QByteArray> ¶ms)
|
||||
: _subtype(subtype), _label(label), _params(params) {}
|
||||
|
||||
unsigned subtype() const {return _subtype;}
|
||||
const QString &label() const {return _label;}
|
||||
const QVector<QByteArray> ¶ms() const {return _params;}
|
||||
|
||||
private:
|
||||
unsigned _subtype;
|
||||
QString _label;
|
||||
QVector<QByteArray> _params;
|
||||
};
|
||||
|
||||
struct Sounding {
|
||||
Sounding() : depth(NAN) {}
|
||||
Sounding(const Coordinates &c, double depth) : c(c), depth(depth) {}
|
||||
@ -117,12 +103,10 @@ private:
|
||||
const RecordMap &vc, const RecordMap &ve, uint COMF);
|
||||
static Polygon polyGeometry(const ISO8211::Record &r, const RecordMap &vc,
|
||||
const RecordMap &ve, uint COMF);
|
||||
static Attr pointAttr(const ISO8211::Record &r, uint OBJL);
|
||||
static Attr lineAttr(const ISO8211::Record &r, uint OBJL);
|
||||
static Attr polyAttr(const ISO8211::Record &r, uint OBJL);
|
||||
static Attributes attributes(const ISO8211::Record &r);
|
||||
static Point *pointObject(const Sounding &s);
|
||||
static Point *pointObject(const ISO8211::Record &r, const RecordMap &vi,
|
||||
const RecordMap &vc, uint COMF, uint OBJL);
|
||||
const RecordMap &vc, uint COMF, uint OBJL, uint HUNI);
|
||||
static Line *lineObject(const ISO8211::Record &r, const RecordMap &vc,
|
||||
const RecordMap &ve, uint COMF, uint OBJL);
|
||||
static Poly *polyObject(const ISO8211::Record &r, const RecordMap &vc,
|
||||
|
@ -51,6 +51,7 @@
|
||||
#define LAKARE 69
|
||||
#define LNDARE 71
|
||||
#define LNDELV 72
|
||||
#define LNDRGN 73
|
||||
#define LNDMRK 74
|
||||
#define LIGHTS 75
|
||||
#define LOKBSN 79
|
||||
|
@ -5,6 +5,7 @@
|
||||
#include "map/textpointitem.h"
|
||||
#include "map/rectd.h"
|
||||
#include "objects.h"
|
||||
#include "attributes.h"
|
||||
#include "style.h"
|
||||
#include "rastertile.h"
|
||||
|
||||
@ -12,8 +13,7 @@ using namespace ENC;
|
||||
|
||||
#define TEXT_EXTENT 160
|
||||
#define TSSLPT_SIZE 24
|
||||
|
||||
typedef QSet<Coordinates> PointSet;
|
||||
#define RANGE_FACTOR 4
|
||||
|
||||
static const float C1 = 0.866025f; /* sqrt(3)/2 */
|
||||
static const QColor tsslptPen = QColor(0xeb, 0x49, 0xeb);
|
||||
@ -138,21 +138,21 @@ static void drawArrow(QPainter *painter, const QPolygonF &polygon, uint type)
|
||||
}
|
||||
|
||||
void RasterTile::drawArrows(QPainter *painter,
|
||||
const QList<MapData::Point> &points)
|
||||
const QList<MapData::Point> &points) const
|
||||
{
|
||||
for (int i = 0; i < points.size(); i++) {
|
||||
const MapData::Point &point = points.at(i);
|
||||
|
||||
if (point.type()>>16 == TSSLPT || point.type()>>16 == RCTLPT) {
|
||||
QPolygonF polygon(tsslptArrow(ll2xy(point.pos()),
|
||||
deg2rad(point.param().toDouble())));
|
||||
deg2rad(point.attributes().value(ORIENT).toDouble())));
|
||||
drawArrow(painter, polygon, point.type());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void RasterTile::drawPolygons(QPainter *painter,
|
||||
const QList<MapData::Poly> &polygons)
|
||||
const QList<MapData::Poly> &polygons) const
|
||||
{
|
||||
for (int n = 0; n < _style->drawOrder().size(); n++) {
|
||||
for (int i = 0; i < polygons.size(); i++) {
|
||||
@ -184,7 +184,7 @@ void RasterTile::drawPolygons(QPainter *painter,
|
||||
}
|
||||
}
|
||||
|
||||
void RasterTile::drawLines(QPainter *painter, const QList<MapData::Line> &lines)
|
||||
void RasterTile::drawLines(QPainter *painter, const QList<MapData::Line> &lines) const
|
||||
{
|
||||
painter->setBrush(Qt::NoBrush);
|
||||
|
||||
@ -202,7 +202,7 @@ void RasterTile::drawLines(QPainter *painter, const QList<MapData::Line> &lines)
|
||||
}
|
||||
|
||||
void RasterTile::drawTextItems(QPainter *painter,
|
||||
const QList<TextItem*> &textItems)
|
||||
const QList<TextItem*> &textItems) const
|
||||
{
|
||||
QRectF rect(_rect);
|
||||
|
||||
@ -213,10 +213,56 @@ void RasterTile::drawTextItems(QPainter *painter,
|
||||
}
|
||||
}
|
||||
|
||||
void RasterTile::processPoints(QList<MapData::Point> &points,
|
||||
QList<TextItem*> &textItems, QList<TextItem*> &lights)
|
||||
static QRectF lightRect(const QPointF &pos, double range)
|
||||
{
|
||||
PointSet lightsSet, signalsSet;
|
||||
return QRectF(pos.x() - range * RANGE_FACTOR, pos.y() - range * RANGE_FACTOR,
|
||||
2*range * RANGE_FACTOR, 2*range * RANGE_FACTOR);
|
||||
}
|
||||
|
||||
void RasterTile::drawSectorLights(QPainter *painter,
|
||||
const QList<SectorLight> &lights) const
|
||||
{
|
||||
for (int i = 0; i < lights.size(); i++) {
|
||||
const SectorLight &l = lights.at(i);
|
||||
QPointF pos(ll2xy(l.pos));
|
||||
QRectF rect(lightRect(pos, (l.range == 0) ? 6 : l.range));
|
||||
double a1 = -(l.end + 90);
|
||||
double a2 = -(l.start + 90);
|
||||
if (a1 > a2)
|
||||
a2 += 360;
|
||||
double as = (a2 - a1);
|
||||
if (as == 0)
|
||||
as = 360;
|
||||
|
||||
if (l.visibility == 3 || l.visibility >= 6)
|
||||
painter->setPen(QPen(Qt::black, 1, Qt::DashLine));
|
||||
else {
|
||||
painter->setPen(QPen(Qt::black, 6, Qt::SolidLine, Qt::FlatCap));
|
||||
painter->drawArc(rect, a1 * 16, as * 16);
|
||||
painter->setPen(QPen(Style::color(l.color), 4, Qt::SolidLine,
|
||||
Qt::FlatCap));
|
||||
}
|
||||
|
||||
painter->drawArc(rect, a1 * 16, as * 16);
|
||||
|
||||
if (a2 - a1 != 0) {
|
||||
QLineF ln(pos, QPointF(pos.x() + rect.width(), pos.y()));
|
||||
ln.setAngle(a1);
|
||||
painter->setPen(QPen(Qt::black, 1, Qt::DashLine));
|
||||
painter->drawLine(ln);
|
||||
ln.setAngle(a2);
|
||||
painter->drawLine(ln);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void RasterTile::processPoints(QList<MapData::Point> &points,
|
||||
QList<TextItem*> &textItems, QList<TextItem*> &lights,
|
||||
QList<SectorLight> §orLights)
|
||||
{
|
||||
LightMap lightsMap;
|
||||
SignalSet signalsSet;
|
||||
QSet<Coordinates> slMap;
|
||||
int i;
|
||||
|
||||
std::sort(points.begin(), points.end());
|
||||
@ -224,9 +270,21 @@ void RasterTile::processPoints(QList<MapData::Point> &points,
|
||||
/* Lights & Signals */
|
||||
for (i = 0; i < points.size(); i++) {
|
||||
const MapData::Point &point = points.at(i);
|
||||
if (point.type()>>16 == LIGHTS)
|
||||
lightsSet.insert(point.pos());
|
||||
else if (point.type()>>16 == FOGSIG)
|
||||
|
||||
if (point.type()>>16 == LIGHTS) {
|
||||
const MapData::Attributes &attr = point.attributes();
|
||||
Style::Color color = (Style::Color)(attr.value(COLOUR).toUInt());
|
||||
double range = attr.value(VALNMR).toDouble();
|
||||
|
||||
if (attr.contains(SECTR1)
|
||||
|| (range > 6 && !(point.type() & 0xFFFF))) {
|
||||
sectorLights.append(SectorLight(point.pos(), color,
|
||||
attr.value(LITVIS).toUInt(), range,
|
||||
attr.value(SECTR1).toDouble(), attr.value(SECTR2).toDouble()));
|
||||
slMap.insert(point.pos());
|
||||
} else
|
||||
lightsMap.insert(point.pos(), color);
|
||||
} else if (point.type()>>16 == FOGSIG)
|
||||
signalsSet.insert(point.pos());
|
||||
else
|
||||
break;
|
||||
@ -245,7 +303,7 @@ void RasterTile::processPoints(QList<MapData::Point> &points,
|
||||
const QColor *color = &style.textColor();
|
||||
const QColor *hColor = style.haloColor().isValid()
|
||||
? &style.haloColor() : 0;
|
||||
double rotate = angle(point.type(), point.param());
|
||||
double rotate = angle(point.type(), point.attributes().value(ORIENT));
|
||||
|
||||
if ((!label || !fnt) && !img)
|
||||
continue;
|
||||
@ -254,11 +312,12 @@ void RasterTile::processPoints(QList<MapData::Point> &points,
|
||||
|
||||
TextPointItem *item = new TextPointItem(pos + offset, label, fnt, img,
|
||||
color, hColor, 0, 2, rotate);
|
||||
if (item->isValid() && !item->collides(textItems)) {
|
||||
if (item->isValid() && (slMap.contains(point.pos())
|
||||
|| (point.polygon() && img) || !item->collides(textItems))) {
|
||||
textItems.append(item);
|
||||
if (lightsSet.contains(point.pos()))
|
||||
if (lightsMap.contains(point.pos()))
|
||||
lights.append(new TextPointItem(pos + _style->lightOffset(),
|
||||
0, 0, _style->light(), 0, 0, 0, 0));
|
||||
0, 0, _style->light(lightsMap.value(point.pos())), 0, 0, 0, 0));
|
||||
if (signalsSet.contains(point.pos()))
|
||||
lights.append(new TextPointItem(pos + _style->signalOffset(),
|
||||
0, 0, _style->signal(), 0, 0, 0, 0));
|
||||
@ -326,13 +385,14 @@ void RasterTile::render()
|
||||
QList<MapData::Poly> polygons;
|
||||
QList<MapData::Point> points;
|
||||
QList<TextItem*> textItems, lights;
|
||||
QList<SectorLight> sectorLights;
|
||||
|
||||
img.setDevicePixelRatio(_ratio);
|
||||
img.fill(Qt::transparent);
|
||||
|
||||
fetchData(polygons, lines, points);
|
||||
|
||||
processPoints(points, textItems, lights);
|
||||
processPoints(points, textItems, lights, sectorLights);
|
||||
processLines(lines, textItems);
|
||||
|
||||
QPainter painter(&img);
|
||||
@ -344,7 +404,9 @@ void RasterTile::render()
|
||||
drawLines(&painter, lines);
|
||||
drawArrows(&painter, points);
|
||||
|
||||
|
||||
drawTextItems(&painter, lights);
|
||||
drawSectorLights(&painter, sectorLights);
|
||||
drawTextItems(&painter, textItems);
|
||||
|
||||
qDeleteAll(textItems);
|
||||
|
@ -34,6 +34,23 @@ public:
|
||||
void render();
|
||||
|
||||
private:
|
||||
struct SectorLight
|
||||
{
|
||||
SectorLight(const Coordinates &pos, Style::Color color, uint visibility,
|
||||
double range, double start, double end) : pos(pos), color(color),
|
||||
visibility(visibility), range(range), start(start), end(end) {}
|
||||
|
||||
Coordinates pos;
|
||||
Style::Color color;
|
||||
uint visibility;
|
||||
double range;
|
||||
double start;
|
||||
double end;
|
||||
};
|
||||
|
||||
typedef QMap<Coordinates, Style::Color> LightMap;
|
||||
typedef QSet<Coordinates> SignalSet;
|
||||
|
||||
void fetchData(QList<MapData::Poly> &polygons, QList<MapData::Line> &lines,
|
||||
QList<MapData::Point> &points);
|
||||
QPointF ll2xy(const Coordinates &c) const
|
||||
@ -44,15 +61,15 @@ private:
|
||||
QPolygonF tsslptArrow(const QPointF &p, qreal angle) const;
|
||||
QPointF centroid(const QVector<Coordinates> &polygon) const;
|
||||
void processPoints(QList<MapData::Point> &points,
|
||||
QList<TextItem*> &textItems, QList<TextItem *> &lights);
|
||||
QList<TextItem*> &textItems, QList<TextItem *> &lights,
|
||||
QList<SectorLight> §orLights);
|
||||
void processLines(const QList<MapData::Line> &lines,
|
||||
QList<TextItem*> &textItems);
|
||||
void drawBitmapPath(QPainter *painter, const QImage &img,
|
||||
const Polygon &polygon);
|
||||
void drawArrows(QPainter *painter, const QList<MapData::Point> &points);
|
||||
void drawPolygons(QPainter *painter, const QList<MapData::Poly> &polygons);
|
||||
void drawLines(QPainter *painter, const QList<MapData::Line> &lines);
|
||||
void drawTextItems(QPainter *painter, const QList<TextItem*> &textItems);
|
||||
void drawArrows(QPainter *painter, const QList<MapData::Point> &points) const;
|
||||
void drawPolygons(QPainter *painter, const QList<MapData::Poly> &polygons) const;
|
||||
void drawLines(QPainter *painter, const QList<MapData::Line> &lines) const;
|
||||
void drawTextItems(QPainter *painter, const QList<TextItem*> &textItems) const;
|
||||
void drawSectorLights(QPainter *painter, const QList<SectorLight> &lights) const;
|
||||
|
||||
static bool polyCb(MapData *data, void *context);
|
||||
static bool pointCb(MapData *data, void *context);
|
||||
|
@ -70,17 +70,26 @@ void Style::polygonStyle()
|
||||
_polygons[TYPE(DRGARE)] = Polygon(QBrush(QColor(0xa0, 0xa0, 0xff),
|
||||
Qt::Dense4Pattern));
|
||||
_polygons[TYPE(UNSARE)] = Polygon(QBrush(QColor(0x99, 0x99, 0x99)));
|
||||
_polygons[SUBTYPE(RESARE, 1)] = Polygon(QImage(":/marine/safety-zone-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 2)] = Polygon(QImage(":/marine/noanchor-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 4)] = Polygon(QImage(":/marine/nature-reserve-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 5)] = Polygon(QImage(":/marine/sanctuary-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 6)] = Polygon(QImage(":/marine/sanctuary-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 7)] = Polygon(QImage(":/marine/sanctuary-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 9)] = Polygon(QBrush(QColor(0xff, 0x00, 0x00),
|
||||
Qt::BDiagPattern));
|
||||
_polygons[SUBTYPE(RESARE, 2)] = Polygon(QImage(":/marine/noanchor-line.png"));
|
||||
_polygons[SUBTYPE(I_RESARE, 2)] = Polygon(QImage(":/marine/noanchor-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 17)] = Polygon(
|
||||
QImage(":/marine/entry-prohibited-line.png"));
|
||||
_polygons[SUBTYPE(I_RESARE, 17)] = Polygon(
|
||||
QImage(":/marine/entry-prohibited-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 12)] = Polygon(QImage(":/marine/safety-zone-line.png"));
|
||||
_polygons[SUBTYPE(I_RESARE, 12)] = Polygon(QImage(":/marine/safety-zone-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 1)] = Polygon(QImage(":/marine/safety-zone-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 17)] = Polygon(QImage(":/marine/entry-prohibited-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 22)] = Polygon(QImage(":/marine/sanctuary-line.png"));
|
||||
_polygons[SUBTYPE(RESARE, 23)] = Polygon(QImage(":/marine/nature-reserve-line.png"));
|
||||
_polygons[SUBTYPE(I_RESARE, 1)] = _polygons[SUBTYPE(RESARE, 1)];
|
||||
_polygons[SUBTYPE(I_RESARE, 2)] = _polygons[SUBTYPE(RESARE, 2)];
|
||||
_polygons[SUBTYPE(I_RESARE, 4)] = _polygons[SUBTYPE(RESARE, 4)];
|
||||
_polygons[SUBTYPE(I_RESARE, 9)] = _polygons[SUBTYPE(RESARE, 9)];
|
||||
_polygons[SUBTYPE(I_RESARE, 12)] = _polygons[SUBTYPE(RESARE, 12)];
|
||||
_polygons[SUBTYPE(I_RESARE, 17)] = _polygons[SUBTYPE(RESARE, 17)];
|
||||
_polygons[SUBTYPE(I_RESARE, 22)] = _polygons[SUBTYPE(RESARE, 22)];
|
||||
_polygons[SUBTYPE(I_RESARE, 23)] = _polygons[SUBTYPE(RESARE, 23)];
|
||||
_polygons[SUBTYPE(ACHARE, 1)] = Polygon(QImage(":/marine/anchor-line.png"));
|
||||
_polygons[SUBTYPE(ACHARE, 2)] = _polygons[SUBTYPE(ACHARE, 1)];
|
||||
_polygons[SUBTYPE(ACHARE, 3)] = _polygons[SUBTYPE(ACHARE, 1)];
|
||||
@ -162,9 +171,14 @@ void Style::polygonStyle()
|
||||
<< SUBTYPE(I_ACHARE, 2) << SUBTYPE(I_ACHARE, 3) << SUBTYPE(I_ACHARE, 4)
|
||||
<< SUBTYPE(I_ACHARE, 5) << SUBTYPE(I_ACHARE, 6) << SUBTYPE(I_ACHARE, 7)
|
||||
<< SUBTYPE(I_ACHARE, 8) << SUBTYPE(I_ACHARE, 9) << SUBTYPE(I_ACHARE, 10)
|
||||
<< SUBTYPE(I_ACHARE, 11) << SUBTYPE(I_ACHARE, 12) << SUBTYPE(RESARE, 9)
|
||||
<< SUBTYPE(RESARE, 2) << SUBTYPE(I_RESARE, 2) << SUBTYPE(RESARE, 17)
|
||||
<< SUBTYPE(I_RESARE, 17) << SUBTYPE(RESARE, 12) << SUBTYPE(I_RESARE, 12)
|
||||
<< SUBTYPE(I_ACHARE, 11) << SUBTYPE(I_ACHARE, 12) << SUBTYPE(RESARE, 1)
|
||||
<< SUBTYPE(I_RESARE, 1) << SUBTYPE(RESARE, 2) << SUBTYPE(I_RESARE, 2)
|
||||
<< SUBTYPE(RESARE, 4) << SUBTYPE(I_RESARE, 4) << SUBTYPE(RESARE, 5)
|
||||
<< SUBTYPE(I_RESARE, 5) << SUBTYPE(RESARE, 6) << SUBTYPE(I_RESARE, 6)
|
||||
<< SUBTYPE(RESARE, 7) << SUBTYPE(I_RESARE, 7) << SUBTYPE(RESARE, 9)
|
||||
<< SUBTYPE(I_RESARE, 9) << SUBTYPE(RESARE, 12) << SUBTYPE(I_RESARE, 12)
|
||||
<< SUBTYPE(RESARE, 17) << SUBTYPE(I_RESARE, 17) << SUBTYPE(RESARE, 22)
|
||||
<< SUBTYPE(I_RESARE, 22) << SUBTYPE(RESARE, 23) << SUBTYPE(I_RESARE, 23)
|
||||
<< SUBTYPE(RESARE, 1) << TYPE(CBLARE) << TYPE(PIPARE) << TYPE(PRCARE)
|
||||
<< TYPE(I_TRNBSN) << SUBTYPE(MARKUL, 3) << TYPE(CONZNE);
|
||||
}
|
||||
@ -222,6 +236,14 @@ void Style::lineStyle(qreal ratio)
|
||||
_lines[TYPE(PIPOHD)] = Line(QImage(":/marine/pipeline-overhead.png"));
|
||||
_lines[TYPE(I_PIPOHD)] = Line(QImage(":/marine/pipeline-overhead.png"));
|
||||
_lines[TYPE(CANALS)] = Line(QPen(QColor(0x9f, 0xc4, 0xe1), 2));
|
||||
_lines[SUBTYPE(RDOCAL, 1)] = Line(QPen(QColor(0xeb, 0x49, 0xeb), 1,
|
||||
Qt::DashLine));
|
||||
_lines[SUBTYPE(RDOCAL, 2)] = Line(QPen(QColor(0xeb, 0x49, 0xeb), 1,
|
||||
Qt::DashLine));
|
||||
_lines[SUBTYPE(RDOCAL, 3)] = Line(QPen(QColor(0xeb, 0x49, 0xeb), 1,
|
||||
Qt::DashLine));
|
||||
_lines[SUBTYPE(RDOCAL, 4)] = Line(QPen(QColor(0xeb, 0x49, 0xeb), 1,
|
||||
Qt::DashLine));
|
||||
}
|
||||
|
||||
void Style::pointStyle(qreal ratio)
|
||||
@ -302,7 +324,9 @@ void Style::pointStyle(qreal ratio)
|
||||
_points[SUBTYPE(I_HRBFAC, 5)] = Point(QImage(":/marine/yacht-harbor.png"));
|
||||
_points[SUBTYPE(I_HRBFAC, 9)] = Point(QImage(":/marine/shipyard.png"));
|
||||
_points[TYPE(ACHBRT)] = Point(QImage(":/marine/anchorage.png"));
|
||||
_points[TYPE(I_ACHBRT)] = Point(QImage(":/marine/anchorage.png"));
|
||||
_points[TYPE(ACHBRT)].setTextColor(QColor(0xeb, 0x49, 0xeb));
|
||||
_points[TYPE(ACHBRT)].setHaloColor(QColor());
|
||||
_points[TYPE(I_ACHBRT)] = _points[TYPE(ACHBRT)];
|
||||
_points[TYPE(OFSPLF)] = Point(QImage(":/marine/platform.png"));
|
||||
_points[TYPE(PILPNT)] = Point(QImage(":/marine/pile.png"), Small);
|
||||
_points[SUBTYPE(MORFAC, 1)] = Point(QImage(":/marine/pile.png"), Small);
|
||||
@ -366,10 +390,14 @@ void Style::pointStyle(qreal ratio)
|
||||
_points[SUBTYPE(WEDKLP, 0)] = Point(QImage(":/marine/kelp.png"));
|
||||
_points[SUBTYPE(WEDKLP, 1)] = Point(QImage(":/marine/kelp.png"));
|
||||
_points[TYPE(SEAARE)].setHaloColor(QColor());
|
||||
_points[TYPE(LNDARE)].setHaloColor(QColor());
|
||||
_points[TYPE(LNDRGN)].setHaloColor(QColor());
|
||||
_points[TYPE(RADRFL)] = Point(QImage(":/marine/radar-reflector.png"));
|
||||
|
||||
_points[SUBTYPE(I_BERTHS, 6)] = Point(QImage(":/marine/fleeting-area.png"),
|
||||
Small);
|
||||
_points[SUBTYPE(I_BERTHS, 6)].setTextColor(QColor(0xeb, 0x49, 0xeb));
|
||||
_points[SUBTYPE(I_BERTHS, 6)].setHaloColor(QColor());
|
||||
_points[SUBTYPE(ACHARE, 2)] = Point(QImage(":/marine/dw-anchorage.png"),
|
||||
Small);
|
||||
_points[SUBTYPE(ACHARE, 3)] = Point(QImage(":/marine/tanker-anchorage.png"),
|
||||
@ -379,6 +407,30 @@ void Style::pointStyle(qreal ratio)
|
||||
_points[SUBTYPE(I_ACHARE, 2)] = _points[SUBTYPE(ACHARE, 2)];
|
||||
_points[SUBTYPE(I_ACHARE, 3)] = _points[SUBTYPE(ACHARE, 3)];
|
||||
_points[SUBTYPE(I_ACHARE, 9)] = _points[SUBTYPE(ACHARE, 9)];
|
||||
_points[SUBTYPE(RESARE, 1)].setTextColor(QColor(0xeb, 0x49, 0xeb));
|
||||
_points[SUBTYPE(RESARE, 1)].setHaloColor(QColor());
|
||||
_points[SUBTYPE(RESARE, 2)] = _points[SUBTYPE(RESARE, 1)];
|
||||
_points[SUBTYPE(RESARE, 4)].setTextColor(QColor(0x30, 0xa0, 0x1b));
|
||||
_points[SUBTYPE(RESARE, 4)].setHaloColor(QColor());
|
||||
_points[SUBTYPE(RESARE, 5)] = _points[SUBTYPE(RESARE, 4)];
|
||||
_points[SUBTYPE(RESARE, 6)] = _points[SUBTYPE(RESARE, 4)];
|
||||
_points[SUBTYPE(RESARE, 7)] = _points[SUBTYPE(RESARE, 4)];
|
||||
_points[SUBTYPE(RESARE, 9)] = _points[SUBTYPE(RESARE, 1)];
|
||||
_points[SUBTYPE(RESARE, 12)] = _points[SUBTYPE(RESARE, 1)];
|
||||
_points[SUBTYPE(RESARE, 17)] = _points[SUBTYPE(RESARE, 1)];
|
||||
_points[SUBTYPE(RESARE, 22)] = _points[SUBTYPE(RESARE, 4)];
|
||||
_points[SUBTYPE(RESARE, 23)] = _points[SUBTYPE(RESARE, 4)];
|
||||
_points[SUBTYPE(I_RESARE, 1)] = _points[SUBTYPE(RESARE, 1)];
|
||||
_points[SUBTYPE(I_RESARE, 2)] = _points[SUBTYPE(RESARE, 2)];
|
||||
_points[SUBTYPE(I_RESARE, 4)] = _points[SUBTYPE(RESARE, 4)];
|
||||
_points[SUBTYPE(I_RESARE, 5)] = _points[SUBTYPE(RESARE, 5)];
|
||||
_points[SUBTYPE(I_RESARE, 6)] = _points[SUBTYPE(RESARE, 6)];
|
||||
_points[SUBTYPE(I_RESARE, 7)] = _points[SUBTYPE(RESARE, 7)];
|
||||
_points[SUBTYPE(I_RESARE, 9)] = _points[SUBTYPE(RESARE, 9)];
|
||||
_points[SUBTYPE(I_RESARE, 12)] = _points[SUBTYPE(RESARE, 12)];
|
||||
_points[SUBTYPE(I_RESARE, 17)] = _points[SUBTYPE(RESARE, 17)];
|
||||
_points[SUBTYPE(I_RESARE, 22)] = _points[SUBTYPE(RESARE, 22)];
|
||||
_points[SUBTYPE(I_RESARE, 23)] = _points[SUBTYPE(RESARE, 23)];
|
||||
|
||||
_points[SUBTYPE(SMCFAC, 7)] = Point(svg2img(":/POI/restaurant-11.svg",
|
||||
ratio), Small);
|
||||
@ -427,6 +479,9 @@ void Style::pointStyle(qreal ratio)
|
||||
Style::Style(qreal ratio)
|
||||
{
|
||||
_light = QImage(":/marine/light.png");
|
||||
_lightRed = QImage(":/marine/light-red.png");
|
||||
_lightGreen = QImage(":/marine/light-green.png");
|
||||
_lightYellow = QImage(":/marine/light-yellow.png");
|
||||
_lightOffset = QPoint(11, 11);
|
||||
_signal = QImage(":/marine/fog-signal.png");
|
||||
_signalOffset = QPoint(-9, 9);
|
||||
@ -477,3 +532,37 @@ const QFont *Style::font(Style::FontSize size) const
|
||||
return &_normal;
|
||||
}
|
||||
}
|
||||
|
||||
const QImage *Style::light(Color color) const
|
||||
{
|
||||
switch (color) {
|
||||
case Red:
|
||||
return &_lightRed;
|
||||
case Green:
|
||||
return &_lightGreen;
|
||||
case White:
|
||||
case Yellow:
|
||||
case Amber:
|
||||
case Orange:
|
||||
return &_lightYellow;
|
||||
default:
|
||||
return &_light;
|
||||
}
|
||||
}
|
||||
|
||||
QColor Style::color(Style::Color c)
|
||||
{
|
||||
switch (c) {
|
||||
case Red:
|
||||
return Qt::red;
|
||||
case Green:
|
||||
return Qt::green;
|
||||
case White:
|
||||
case Yellow:
|
||||
case Amber:
|
||||
case Orange:
|
||||
return Qt::yellow;
|
||||
default:
|
||||
return Qt::magenta;
|
||||
}
|
||||
}
|
||||
|
@ -89,6 +89,11 @@ public:
|
||||
QPoint _offset;
|
||||
};
|
||||
|
||||
enum Color {
|
||||
Unknown, White, Black, Red, Green, Blue, Yellow, Gray,
|
||||
Brown, Amber, Violet, Orange, Magenta, Pink
|
||||
};
|
||||
|
||||
Style(qreal ratio);
|
||||
|
||||
const Line &line(uint type) const;
|
||||
@ -97,11 +102,13 @@ public:
|
||||
const QVector<uint> &drawOrder() const {return _drawOrder;}
|
||||
|
||||
const QFont *font(Style::FontSize size) const;
|
||||
const QImage *light() const {return &_light;}
|
||||
const QImage *light(Color color) const;
|
||||
const QImage *signal() const {return &_signal;}
|
||||
const QPoint &lightOffset() const {return _lightOffset;}
|
||||
const QPoint &signalOffset() const {return _signalOffset;}
|
||||
|
||||
static QColor color(Color c);
|
||||
|
||||
private:
|
||||
void polygonStyle();
|
||||
void lineStyle(qreal ratio);
|
||||
@ -114,7 +121,7 @@ private:
|
||||
|
||||
/* Fonts and images must be initialized after QGuiApplication! */
|
||||
QFont _small, _normal, _large;
|
||||
QImage _light, _signal;
|
||||
QImage _light, _lightRed, _lightGreen, _lightYellow, _signal;
|
||||
QPoint _lightOffset, _signalOffset;
|
||||
};
|
||||
|
||||
|
@ -111,3 +111,13 @@ double DEMTree::elevation(const Coordinates &c) const
|
||||
|
||||
return ele;
|
||||
}
|
||||
|
||||
MatrixD DEMTree::elevation(const MatrixC &m) const
|
||||
{
|
||||
MatrixD ret(m.h(), m.w());
|
||||
|
||||
for (int i = 0; i < m.size(); i++)
|
||||
ret.at(i) = elevation(m.at(i));
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
@ -2,6 +2,7 @@
|
||||
#define IMG_DEMTREE_H
|
||||
|
||||
#include "common/rtree.h"
|
||||
#include "map/matrix.h"
|
||||
#include "mapdata.h"
|
||||
|
||||
namespace IMG {
|
||||
@ -11,6 +12,7 @@ public:
|
||||
DEMTree(const QList<MapData::Elevation> &tiles);
|
||||
|
||||
double elevation(const Coordinates &c) const;
|
||||
MatrixD elevation(const MatrixC &m) const;
|
||||
|
||||
private:
|
||||
typedef RTree<const MapData::Elevation*, double, 2> Tree;
|
||||
|
52
src/map/IMG/lights.h
Normal file
@ -0,0 +1,52 @@
|
||||
#ifndef IMG_LIGHTS_H
|
||||
#define IMG_LIGHTS_H
|
||||
|
||||
#include <QVector>
|
||||
#include <QDebug>
|
||||
|
||||
namespace IMG {
|
||||
|
||||
class Lights
|
||||
{
|
||||
public:
|
||||
enum Color {None, Red, Green, White, Blue, Yellow, Violet, Amber};
|
||||
|
||||
struct Sector
|
||||
{
|
||||
Sector() : color(None), angle(0), range(0) {}
|
||||
Sector(Color color, quint32 angle, quint32 range)
|
||||
: color(color), angle(angle), range(range) {}
|
||||
|
||||
Color color;
|
||||
quint32 angle;
|
||||
quint32 range;
|
||||
};
|
||||
|
||||
Lights() : color(None), range(0) {}
|
||||
bool isSectorLight() const
|
||||
{return ((color && range > 6) || !sectors.isEmpty());}
|
||||
|
||||
Color color;
|
||||
quint32 range;
|
||||
QVector<Sector> sectors;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#ifndef QT_NO_DEBUG
|
||||
inline QDebug operator<<(QDebug dbg, const IMG::Lights::Sector §or)
|
||||
{
|
||||
dbg.nospace() << "Sector(" << sector.color << ", " << sector.angle
|
||||
<< ", " << sector.range << ")";
|
||||
return dbg.space();
|
||||
}
|
||||
|
||||
inline QDebug operator<<(QDebug dbg, const IMG::Lights &lights)
|
||||
{
|
||||
dbg.nospace() << "Lights(" << lights.color << ", " << lights.range << ", "
|
||||
<< lights.sectors << ")";
|
||||
return dbg.space();
|
||||
}
|
||||
#endif // QT_NO_DEBUG
|
||||
|
||||
#endif // IMG_LIGHTS_H
|
@ -14,6 +14,7 @@
|
||||
#include "map/matrix.h"
|
||||
#include "label.h"
|
||||
#include "raster.h"
|
||||
#include "lights.h"
|
||||
#include "zoom.h"
|
||||
|
||||
namespace IMG {
|
||||
@ -51,11 +52,11 @@ public:
|
||||
enum Flags {
|
||||
NoFlag = 0,
|
||||
ClassLabel = 1,
|
||||
Light = 2
|
||||
};
|
||||
|
||||
Coordinates coordinates;
|
||||
Label label;
|
||||
Lights lights;
|
||||
quint64 id;
|
||||
quint32 type;
|
||||
quint32 flags;
|
||||
|
@ -18,13 +18,13 @@ using namespace IMG;
|
||||
|
||||
#define TEXT_EXTENT 160
|
||||
#define ICON_PADDING 2
|
||||
#define RANGE_FACTOR 4
|
||||
#define ROAD 0
|
||||
#define WATER 1
|
||||
|
||||
#define AREA(rect) \
|
||||
(rect.size().width() * rect.size().height())
|
||||
|
||||
#define ROAD 0
|
||||
#define WATER 1
|
||||
|
||||
static const QColor textColor(Qt::black);
|
||||
static const QColor haloColor(Qt::white);
|
||||
static const QColor shieldColor(Qt::white);
|
||||
@ -226,6 +226,65 @@ void RasterTile::drawTextItems(QPainter *painter,
|
||||
}
|
||||
}
|
||||
|
||||
static QRect lightRect(const QPoint &pos, quint32 range)
|
||||
{
|
||||
return QRect(pos.x() - range * RANGE_FACTOR, pos.y() - range * RANGE_FACTOR,
|
||||
2*range * RANGE_FACTOR, 2*range * RANGE_FACTOR);
|
||||
}
|
||||
|
||||
void RasterTile::drawSectorLights(QPainter *painter,
|
||||
const QList<const MapData::Point *> &lights) const
|
||||
{
|
||||
for (int i = 0; i < lights.size(); i++) {
|
||||
const MapData::Point *p = lights.at(i);
|
||||
QPoint pos(p->coordinates.lon(), p->coordinates.lat());
|
||||
|
||||
if (p->lights.sectors.size()) {
|
||||
for (int j = 0; j < p->lights.sectors.size(); j++) {
|
||||
const Lights::Sector &start = p->lights.sectors.at(j);
|
||||
const Lights::Sector &end = (j == p->lights.sectors.size() - 1)
|
||||
? p->lights.sectors.at(0) : p->lights.sectors.at(j+1);
|
||||
|
||||
if (start.color) {
|
||||
double a1 = -(end.angle / 10.0 + 90.0);
|
||||
double a2 = -(start.angle / 10.0 + 90.0);
|
||||
if (a1 > a2)
|
||||
a2 += 360;
|
||||
double as = (a2 - a1);
|
||||
if (as == 0)
|
||||
as = 360;
|
||||
|
||||
QRect rect(lightRect(pos, start.range ? start.range : 6));
|
||||
|
||||
painter->setPen(QPen(Qt::black, 6, Qt::SolidLine,
|
||||
Qt::FlatCap));
|
||||
painter->drawArc(rect, a1 * 16, as * 16);
|
||||
painter->setPen(QPen(Style::color(start.color), 4,
|
||||
Qt::SolidLine, Qt::FlatCap));
|
||||
painter->drawArc(rect, a1 * 16, as * 16);
|
||||
|
||||
if (a2 - a1 != 0) {
|
||||
QLineF ln(pos, QPointF(pos.x() + rect.width(), pos.y()));
|
||||
ln.setAngle(a1);
|
||||
painter->setPen(QPen(Qt::black, 1, Qt::DashLine));
|
||||
painter->drawLine(ln);
|
||||
ln.setAngle(a2);
|
||||
painter->drawLine(ln);
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
QRect rect(lightRect(pos, p->lights.range));
|
||||
|
||||
painter->setPen(QPen(Qt::black, 6, Qt::SolidLine, Qt::FlatCap));
|
||||
painter->drawArc(rect, 0, 360 * 16);
|
||||
painter->setPen(QPen(Style::color(p->lights.color), 4,
|
||||
Qt::SolidLine, Qt::FlatCap));
|
||||
painter->drawArc(rect, 0, 360 * 16);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void removeDuplicitLabel(QList<TextItem *> &labels, const QString &text,
|
||||
const QRectF &tileRect)
|
||||
{
|
||||
@ -398,7 +457,7 @@ void RasterTile::processShields(const QList<MapData::Poly> &lines,
|
||||
}
|
||||
|
||||
void RasterTile::processPoints(QList<MapData::Point> &points,
|
||||
QList<TextItem*> &textItems)
|
||||
QList<TextItem*> &textItems, QList<const MapData::Point*> &lights)
|
||||
{
|
||||
std::sort(points.begin(), points.end());
|
||||
|
||||
@ -408,6 +467,9 @@ void RasterTile::processPoints(QList<MapData::Point> &points,
|
||||
const Style::Point &ps = style->point(point.type);
|
||||
bool poi = Style::isPOI(point.type);
|
||||
|
||||
if (point.lights.isSectorLight())
|
||||
lights.append(&point);
|
||||
|
||||
const QString *label = point.label.text().isEmpty()
|
||||
? 0 : &(point.label.text());
|
||||
const QImage *img = ps.img().isNull() ? 0 : &ps.img();
|
||||
@ -428,11 +490,12 @@ void RasterTile::processPoints(QList<MapData::Point> &points,
|
||||
|
||||
TextPointItem *item = new TextPointItem(pos + offset, label, fnt, img,
|
||||
color, hcolor, 0, ICON_PADDING);
|
||||
if (item->isValid() && !item->collides(textItems)) {
|
||||
if (item->isValid() && (point.lights.isSectorLight()
|
||||
|| !item->collides(textItems))) {
|
||||
textItems.append(item);
|
||||
if (Style::isLight(point.type) || point.flags & MapData::Point::Light)
|
||||
if (point.lights.color && !point.lights.isSectorLight())
|
||||
textItems.append(new TextPointItem(pos + style->lightOffset(),
|
||||
0, 0, style->light(), 0, 0, 0, 0));
|
||||
0, 0, style->light(point.lights.color), 0, 0, 0, 0));
|
||||
} else
|
||||
delete item;
|
||||
}
|
||||
@ -483,22 +546,18 @@ MatrixD RasterTile::elevation(int extend) const
|
||||
|
||||
if (_data->hasDEM()) {
|
||||
RectC rect;
|
||||
QList<MapData::Elevation> tiles;
|
||||
|
||||
for (int i = 0; i < ll.size(); i++)
|
||||
rect = rect.united(ll.at(i));
|
||||
// Extra margin for always including the next DEM tile on the map tile
|
||||
// edges (the DEM tile resolution is usally 0.5-15% of the map tile)
|
||||
/* Extra margin for always including the next DEM tile on the map tile
|
||||
edges (the DEM tile resolution is usally 0.5-15% of the map tile) */
|
||||
double factor = 6 - (_zoom - 24) * 1.7;
|
||||
_data->elevations(_file, rect.adjusted(0, 0, rect.width() / factor,
|
||||
-rect.height() / factor), _zoom, &tiles);
|
||||
RectC br(rect.adjusted(0, 0, rect.width() / factor, -rect.height()
|
||||
/ factor));
|
||||
|
||||
DEMTree tree(tiles);
|
||||
MatrixD m(ll.h(), ll.w());
|
||||
for (int i = 0; i < ll.size(); i++)
|
||||
m.at(i) = tree.elevation(ll.at(i));
|
||||
QList<MapData::Elevation> tiles;
|
||||
_data->elevations(_file, br, _zoom, &tiles);
|
||||
|
||||
return m;
|
||||
return DEMTree(tiles).elevation(ll);
|
||||
} else
|
||||
return DEM::elevation(ll);
|
||||
}
|
||||
@ -525,6 +584,7 @@ void RasterTile::render()
|
||||
QList<MapData::Poly> lines;
|
||||
QList<MapData::Point> points;
|
||||
QList<TextItem*> textItems;
|
||||
QList<const MapData::Point*> lights;
|
||||
QImage arrows[2];
|
||||
|
||||
arrows[ROAD] = Util::svg2img(":/symbols/oneway.svg", _ratio);
|
||||
@ -535,7 +595,7 @@ void RasterTile::render()
|
||||
ll2xy(lines);
|
||||
ll2xy(points);
|
||||
|
||||
processPoints(points, textItems);
|
||||
processPoints(points, textItems, lights);
|
||||
processPolygons(polygons, textItems);
|
||||
processLines(lines, textItems, arrows);
|
||||
|
||||
@ -550,6 +610,7 @@ void RasterTile::render()
|
||||
drawPolygons(&painter, polygons);
|
||||
drawHillShading(&painter);
|
||||
drawLines(&painter, lines);
|
||||
drawSectorLights(&painter, lights);
|
||||
drawTextItems(&painter, textItems);
|
||||
|
||||
qDeleteAll(textItems);
|
||||
|
@ -34,7 +34,8 @@ public:
|
||||
private:
|
||||
typedef RTree<const MapData::Elevation*, double, 2> DEMTRee;
|
||||
|
||||
struct ElevationCTX {
|
||||
struct ElevationCTX
|
||||
{
|
||||
ElevationCTX(const DEMTRee &tree, const Coordinates &c, double &ele)
|
||||
: tree(tree), c(c), ele(ele) {}
|
||||
|
||||
@ -42,7 +43,8 @@ private:
|
||||
const Coordinates &c;
|
||||
double &ele;
|
||||
};
|
||||
struct EdgeCTX {
|
||||
struct EdgeCTX
|
||||
{
|
||||
EdgeCTX(const Coordinates &c, double &ele) : c(c), ele(ele) {}
|
||||
|
||||
const Coordinates &c;
|
||||
@ -62,13 +64,14 @@ private:
|
||||
void drawLines(QPainter *painter, const QList<MapData::Poly> &lines) const;
|
||||
void drawTextItems(QPainter *painter, const QList<TextItem*> &textItems) const;
|
||||
void drawHillShading(QPainter *painter) const;
|
||||
void drawSectorLights(QPainter *painter, const QList<const MapData::Point*> &lights) const;
|
||||
|
||||
void processPolygons(const QList<MapData::Poly> &polygons,
|
||||
QList<TextItem *> &textItems);
|
||||
void processLines(QList<MapData::Poly> &lines, QList<TextItem*> &textItems,
|
||||
const QImage (&arrows)[2]);
|
||||
void processPoints(QList<MapData::Point> &points,
|
||||
QList<TextItem*> &textItems);
|
||||
void processPoints(QList<MapData::Point> &points, QList<TextItem*> &textItems,
|
||||
QList<const MapData::Point *> &lights);
|
||||
void processShields(const QList<MapData::Poly> &lines,
|
||||
QList<TextItem*> &textItems);
|
||||
void processStreetNames(const QList<MapData::Poly> &lines,
|
||||
|
@ -107,7 +107,8 @@ bool RGNFile::readObstructionInfo(Handle &hdl, quint8 flags, quint32 size,
|
||||
return true;
|
||||
}
|
||||
|
||||
bool RGNFile::readBuoyInfo(Handle &hdl, quint8 flags, MapData::Point *point) const
|
||||
bool RGNFile::readBuoyInfo(Handle &hdl, quint8 flags, quint32 size,
|
||||
MapData::Point *point) const
|
||||
{
|
||||
quint16 val;
|
||||
quint8 lc;
|
||||
@ -115,7 +116,7 @@ bool RGNFile::readBuoyInfo(Handle &hdl, quint8 flags, MapData::Point *point) con
|
||||
if ((flags & 0xe0) != 0xe0)
|
||||
return true;
|
||||
|
||||
if (!readUInt16(hdl, val))
|
||||
if (!(size >= 2 && readUInt16(hdl, val)))
|
||||
return false;
|
||||
|
||||
lc = (val >> 10) & 0x0f;
|
||||
@ -123,7 +124,105 @@ bool RGNFile::readBuoyInfo(Handle &hdl, quint8 flags, MapData::Point *point) con
|
||||
lc = (val >> 6) & 7;
|
||||
|
||||
if (lc)
|
||||
point->flags |= MapData::Point::Light;
|
||||
point->lights.color = (Lights::Color)lc;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool RGNFile::readLightInfo(Handle &hdl, quint8 flags, quint32 size,
|
||||
MapData::Point *point) const
|
||||
{
|
||||
quint16 flags1;
|
||||
quint8 flags2;
|
||||
quint32 unused;
|
||||
|
||||
if (!(size >= 3 && readUInt16(hdl, flags1) && readUInt8(hdl, flags2)))
|
||||
return false;
|
||||
size -= 3;
|
||||
if (flags2 >> 6) {
|
||||
if (!(size >= (flags2 >> 6) && readVUInt32(hdl, (flags2 >> 6), unused)))
|
||||
return false;
|
||||
size -= (flags2 >> 6);
|
||||
}
|
||||
if (flags2 >> 2 & 3) {
|
||||
if (!(size >= (flags2 >> 2 & 3)
|
||||
&& readVUInt32(hdl, (flags2 >> 2 & 3), unused)))
|
||||
return false;
|
||||
size -= (flags2 >> 2 & 3);
|
||||
}
|
||||
if (flags1 & 0xc0) {
|
||||
if (flags1 & 0x80) {
|
||||
if (!(size >= 1 && readUInt8(hdl, unused)))
|
||||
return false;
|
||||
unused |= ((flags1 & 0x40) << 2);
|
||||
size--;
|
||||
} else {
|
||||
if (!(size >= 2 && readUInt16(hdl, unused)))
|
||||
return false;
|
||||
size -= 2;
|
||||
}
|
||||
}
|
||||
if (flags & 2) {
|
||||
if (!(size >= 3 && readUInt24(hdl, unused)))
|
||||
return false;
|
||||
size -= 3;
|
||||
}
|
||||
if (flags & 4) {
|
||||
if (!(size >= 3 && readUInt24(hdl, unused)))
|
||||
return false;
|
||||
size -= 3;
|
||||
}
|
||||
if (flags & 8) {
|
||||
if (!(size >= 3 && readUInt24(hdl, unused)))
|
||||
return false;
|
||||
size -= 3;
|
||||
}
|
||||
if (flags1 & 0x200) {
|
||||
if (!(size >= 2 && readUInt16(hdl, unused)))
|
||||
return false;
|
||||
size -= 2;
|
||||
}
|
||||
if (flags1 & 0x400) {
|
||||
if (!(size >= 1 && readUInt8(hdl, unused)))
|
||||
return false;
|
||||
size--;
|
||||
}
|
||||
if (flags1 & 0x800) {
|
||||
quint16 la;
|
||||
quint8 cf, range = 0;
|
||||
|
||||
do {
|
||||
if (!(size >= 2 && readUInt16(hdl, la)))
|
||||
return false;
|
||||
size -= 2;
|
||||
|
||||
cf = la >> 8;
|
||||
Lights::Color c = (Lights::Color)(cf >> 4 & 7);
|
||||
if (c) {
|
||||
if (!(size >= 1 && readUInt8(hdl, range)))
|
||||
return false;
|
||||
size--;
|
||||
}
|
||||
point->lights.sectors.append(Lights::Sector(c, la & 0xfff, range));
|
||||
} while (!(cf >> 7));
|
||||
} else {
|
||||
quint8 v1, v2, range;
|
||||
|
||||
if (!(size >= 1 && readUInt8(hdl, v1)))
|
||||
return false;
|
||||
size--;
|
||||
|
||||
range = v1 & 0x1f;
|
||||
if ((v1 & 0x1f) == 0x1f) {
|
||||
if (!(size >= 1 && readUInt8(hdl, v2)))
|
||||
return false;
|
||||
size--;
|
||||
range += v2;
|
||||
}
|
||||
|
||||
point->lights.color = (Lights::Color)(v1 >> 5);
|
||||
point->lights.range = range;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
@ -183,7 +282,9 @@ bool RGNFile::readClassFields(Handle &hdl, SegmentType segmentType,
|
||||
if (point && Style::isObstructionPoint(point->type))
|
||||
readObstructionInfo(hdl, flags, rs, point);
|
||||
if (point && Style::isBuoy(point->type))
|
||||
readBuoyInfo(hdl, flags, point);
|
||||
readBuoyInfo(hdl, flags, rs, point);
|
||||
if (point && Style::isLight(point->type))
|
||||
readLightInfo(hdl, flags, rs, point);
|
||||
|
||||
return seek(hdl, off + rs);
|
||||
}
|
||||
|
@ -64,7 +64,10 @@ private:
|
||||
MapData::Point *point) const;
|
||||
bool readObstructionInfo(Handle &hdl, quint8 flags, quint32 size,
|
||||
MapData::Point *point) const;
|
||||
bool readBuoyInfo(Handle &hdl, quint8 flags, MapData::Point *point) const;
|
||||
bool readBuoyInfo(Handle &hdl, quint8 flags, quint32 size,
|
||||
MapData::Point *point) const;
|
||||
bool readLightInfo(Handle &hdl, quint8 flags, quint32 size,
|
||||
MapData::Point *point) const;
|
||||
bool readLabel(Handle &hdl, LBLFile *lbl, Handle &lblHdl,
|
||||
quint8 flags, quint32 size, MapData::Point *point) const;
|
||||
|
||||
|
@ -1319,6 +1319,10 @@ Style::Style(qreal ratio, SubFile *typ)
|
||||
_extraSmall = pixelSizeFont(10);
|
||||
|
||||
_light = QImage(":/marine/light.png");
|
||||
_lightRed = QImage(":/marine/light-red.png");
|
||||
_lightGreen = QImage(":/marine/light-green.png");
|
||||
_lightYellow = QImage(":/marine/light-yellow.png");
|
||||
_lightWhite = QImage(":/marine/light-white.png");
|
||||
_lightOffset = QPoint(11, 11);
|
||||
|
||||
defaultLineStyle(ratio);
|
||||
@ -1373,6 +1377,40 @@ const QFont *Style::font(Style::FontSize size, Style::FontSize defaultSize) cons
|
||||
}
|
||||
}
|
||||
|
||||
const QImage *Style::light(Lights::Color color) const
|
||||
{
|
||||
switch (color) {
|
||||
case Lights::Red:
|
||||
return &_lightRed;
|
||||
case Lights::Green:
|
||||
return &_lightGreen;
|
||||
case Lights::White:
|
||||
return &_lightWhite;
|
||||
case Lights::Yellow:
|
||||
case Lights::Amber:
|
||||
return &_lightYellow;
|
||||
default:
|
||||
return &_light;
|
||||
}
|
||||
}
|
||||
|
||||
QColor Style::color(Lights::Color c)
|
||||
{
|
||||
switch (c) {
|
||||
case Lights::Red:
|
||||
return Qt::red;
|
||||
case Lights::Green:
|
||||
return Qt::green;
|
||||
case Lights::White:
|
||||
return Qt::white;
|
||||
case Lights::Yellow:
|
||||
case Lights::Amber:
|
||||
return Qt::yellow;
|
||||
default:
|
||||
return Qt::magenta;
|
||||
}
|
||||
}
|
||||
|
||||
#ifndef QT_NO_DEBUG
|
||||
static QString penColor(const QPen &pen)
|
||||
{
|
||||
|
@ -5,6 +5,7 @@
|
||||
#include <QBrush>
|
||||
#include <QFont>
|
||||
#include <QDebug>
|
||||
#include "lights.h"
|
||||
#include "subfile.h"
|
||||
|
||||
#define TYPE(t) ((t)<<8)
|
||||
@ -112,8 +113,7 @@ public:
|
||||
const QList<quint32> &drawOrder() const {return _drawOrder;}
|
||||
const QFont *font(Style::FontSize size, Style::FontSize defaultSize
|
||||
= Style::Normal) const;
|
||||
|
||||
const QImage *light() const {return &_light;}
|
||||
const QImage *light(Lights::Color color) const;
|
||||
const QPoint &lightOffset() const {return _lightOffset;}
|
||||
|
||||
static bool isPOI(quint32 type)
|
||||
@ -157,6 +157,8 @@ public:
|
||||
static bool isMarina(quint32 type)
|
||||
{return type == 0x10703;}
|
||||
|
||||
static QColor color(Lights::Color c);
|
||||
|
||||
private:
|
||||
struct Section {
|
||||
quint32 offset;
|
||||
@ -203,7 +205,7 @@ private:
|
||||
/* Fonts and images must be initialized after QGuiApplication! */
|
||||
QFont _large, _normal, _small, _extraSmall;
|
||||
|
||||
QImage _light;
|
||||
QImage _light, _lightRed, _lightGreen, _lightYellow, _lightWhite;
|
||||
QPoint _lightOffset;
|
||||
};
|
||||
|
||||
|
@ -63,7 +63,7 @@ MapList::ParserMap MapList::parsers()
|
||||
return map;
|
||||
}
|
||||
|
||||
MapList::ParserMap MapList::_parsers = parsers();
|
||||
MapList::ParserMap MapList::_parsers = MapList::parsers();
|
||||
|
||||
Map *MapList::loadFile(const QString &path, const Projection &proj, bool *isDir)
|
||||
{
|
||||
|