2017-07-29 05:01:33 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
#include "maplocationmodel.h"
|
2019-05-09 19:33:01 +00:00
|
|
|
#include "divelocationmodel.h"
|
2018-10-26 15:03:54 +00:00
|
|
|
#include "core/divesite.h"
|
2019-05-01 21:33:45 +00:00
|
|
|
#ifndef SUBSURFACE_MOBILE
|
|
|
|
#include "qt-models/filtermodels.h"
|
2019-08-30 13:25:59 +00:00
|
|
|
#include "desktop-widgets/mapwidget.h"
|
2019-05-01 21:33:45 +00:00
|
|
|
#endif
|
2017-07-29 05:01:33 +00:00
|
|
|
|
2019-05-08 19:35:49 +00:00
|
|
|
#include <QDebug>
|
|
|
|
#include <algorithm>
|
|
|
|
|
2017-07-29 05:01:33 +00:00
|
|
|
const char *MapLocation::PROPERTY_NAME_COORDINATE = "coordinate";
|
2018-10-26 15:03:54 +00:00
|
|
|
const char *MapLocation::PROPERTY_NAME_DIVESITE = "divesite";
|
2017-07-29 05:01:33 +00:00
|
|
|
const char *MapLocation::PROPERTY_NAME_NAME = "name";
|
2019-08-30 13:25:59 +00:00
|
|
|
const char *MapLocation::PROPERTY_NAME_PIXMAP = "pixmap";
|
2019-08-30 16:09:37 +00:00
|
|
|
const char *MapLocation::PROPERTY_NAME_Z = "z";
|
2017-07-29 05:01:33 +00:00
|
|
|
|
2019-05-01 21:33:45 +00:00
|
|
|
#define MIN_DISTANCE_BETWEEN_DIVE_SITES_M 50.0
|
|
|
|
|
2019-08-30 13:25:59 +00:00
|
|
|
MapLocation::MapLocation() : m_ds(nullptr), m_selected(false)
|
2017-07-29 05:01:33 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2019-08-30 13:25:59 +00:00
|
|
|
MapLocation::MapLocation(struct dive_site *ds, QGeoCoordinate coord, QString name, bool selected) :
|
|
|
|
m_ds(ds), m_coordinate(coord), m_name(name), m_selected(selected)
|
2017-07-29 05:01:33 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2019-08-30 13:25:59 +00:00
|
|
|
// Check whether we are in divesite-edit mode. This doesn't
|
|
|
|
// exist on mobile. And on desktop we have to access the MapWidget.
|
|
|
|
// Simplify this!
|
|
|
|
static bool inEditMode()
|
|
|
|
{
|
|
|
|
#ifdef SUBSURFACE_MOBILE
|
|
|
|
return false;
|
|
|
|
#else
|
|
|
|
return MapWidget::instance()->editMode();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-07-29 05:01:33 +00:00
|
|
|
QVariant MapLocation::getRole(int role) const
|
|
|
|
{
|
|
|
|
switch (role) {
|
2018-10-26 15:03:54 +00:00
|
|
|
case Roles::RoleDivesite:
|
2018-10-28 21:19:17 +00:00
|
|
|
return QVariant::fromValue((dive_site *)m_ds);
|
2017-07-29 05:01:33 +00:00
|
|
|
case Roles::RoleCoordinate:
|
|
|
|
return QVariant::fromValue(m_coordinate);
|
|
|
|
case Roles::RoleName:
|
|
|
|
return QVariant::fromValue(m_name);
|
2019-08-30 13:25:59 +00:00
|
|
|
case Roles::RolePixmap:
|
|
|
|
return m_selected ? QString("qrc:///dive-location-marker-selected-icon") :
|
|
|
|
inEditMode() ? QString("qrc:///dive-location-marker-inactive-icon") :
|
|
|
|
QString("qrc:///dive-location-marker-icon");
|
2019-08-30 16:09:37 +00:00
|
|
|
case Roles::RoleZ:
|
|
|
|
return m_selected ? 1 : 0;
|
2017-07-29 05:01:33 +00:00
|
|
|
default:
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QGeoCoordinate MapLocation::coordinate()
|
|
|
|
{
|
|
|
|
return m_coordinate;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MapLocation::setCoordinate(QGeoCoordinate coord)
|
|
|
|
{
|
|
|
|
m_coordinate = coord;
|
|
|
|
emit coordinateChanged();
|
|
|
|
}
|
|
|
|
|
2017-11-09 16:43:21 +00:00
|
|
|
void MapLocation::setCoordinateNoEmit(QGeoCoordinate coord)
|
|
|
|
{
|
|
|
|
m_coordinate = coord;
|
|
|
|
}
|
|
|
|
|
2018-10-26 15:03:54 +00:00
|
|
|
struct dive_site *MapLocation::divesite()
|
2017-07-29 05:01:33 +00:00
|
|
|
{
|
2018-10-26 15:03:54 +00:00
|
|
|
return m_ds;
|
2017-07-29 05:01:33 +00:00
|
|
|
}
|
|
|
|
|
2018-10-26 15:03:54 +00:00
|
|
|
QVariant MapLocation::divesiteVariant()
|
2017-07-29 05:01:33 +00:00
|
|
|
{
|
2018-10-28 21:19:17 +00:00
|
|
|
return QVariant::fromValue(m_ds);
|
2018-10-26 15:03:54 +00:00
|
|
|
}
|
|
|
|
|
2019-05-01 22:09:59 +00:00
|
|
|
MapLocationModel::MapLocationModel(QObject *parent) : QAbstractListModel(parent)
|
2018-10-26 15:03:54 +00:00
|
|
|
{
|
|
|
|
m_roles[MapLocation::Roles::RoleDivesite] = MapLocation::PROPERTY_NAME_DIVESITE;
|
2017-07-29 05:01:33 +00:00
|
|
|
m_roles[MapLocation::Roles::RoleCoordinate] = MapLocation::PROPERTY_NAME_COORDINATE;
|
|
|
|
m_roles[MapLocation::Roles::RoleName] = MapLocation::PROPERTY_NAME_NAME;
|
2019-08-30 13:25:59 +00:00
|
|
|
m_roles[MapLocation::Roles::RolePixmap] = MapLocation::PROPERTY_NAME_PIXMAP;
|
2019-08-30 16:09:37 +00:00
|
|
|
m_roles[MapLocation::Roles::RoleZ] = MapLocation::PROPERTY_NAME_Z;
|
2019-05-09 19:33:01 +00:00
|
|
|
connect(&diveListNotifier, &DiveListNotifier::diveSiteChanged, this, &MapLocationModel::diveSiteChanged);
|
2017-07-29 05:01:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MapLocationModel::~MapLocationModel()
|
|
|
|
{
|
2019-05-01 21:33:45 +00:00
|
|
|
qDeleteAll(m_mapLocations);
|
2017-07-29 05:01:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant MapLocationModel::data(const QModelIndex & index, int role) const
|
|
|
|
{
|
|
|
|
if (index.row() < 0 || index.row() >= m_mapLocations.size())
|
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
return m_mapLocations.at(index.row())->getRole(role);
|
|
|
|
}
|
|
|
|
|
|
|
|
QHash<int, QByteArray> MapLocationModel::roleNames() const
|
|
|
|
{
|
|
|
|
return m_roles;
|
|
|
|
}
|
|
|
|
|
2018-05-21 15:53:42 +00:00
|
|
|
int MapLocationModel::rowCount(const QModelIndex&) const
|
2017-07-29 05:01:33 +00:00
|
|
|
{
|
|
|
|
return m_mapLocations.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
int MapLocationModel::count()
|
|
|
|
{
|
|
|
|
return m_mapLocations.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
MapLocation *MapLocationModel::get(int row)
|
|
|
|
{
|
|
|
|
if (row < 0 || row >= m_mapLocations.size())
|
|
|
|
return NULL;
|
|
|
|
return m_mapLocations.at(row);
|
|
|
|
}
|
|
|
|
|
|
|
|
void MapLocationModel::add(MapLocation *location)
|
|
|
|
{
|
|
|
|
beginInsertRows(QModelIndex(), m_mapLocations.size(), m_mapLocations.size());
|
|
|
|
m_mapLocations.append(location);
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
|
2019-05-03 21:16:40 +00:00
|
|
|
const QVector<dive_site *> &MapLocationModel::selectedDs() const
|
|
|
|
{
|
|
|
|
return m_selectedDs;
|
|
|
|
}
|
|
|
|
|
2019-05-08 19:35:49 +00:00
|
|
|
static bool hasVisibleDive(const dive_site *ds)
|
|
|
|
{
|
|
|
|
return std::any_of(&ds->dives.dives[0], &ds->dives.dives[ds->dives.nr],
|
|
|
|
[] (const dive *d) { return !d->hidden_by_filter; });
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool hasSelectedDive(const dive_site *ds)
|
2017-07-29 05:01:33 +00:00
|
|
|
{
|
2019-05-08 19:35:49 +00:00
|
|
|
return std::any_of(&ds->dives.dives[0], &ds->dives.dives[ds->dives.nr],
|
|
|
|
[] (const dive *d) { return d->selected; });
|
|
|
|
}
|
2019-05-01 21:33:45 +00:00
|
|
|
|
2019-08-30 14:51:59 +00:00
|
|
|
void MapLocationModel::selectionChanged()
|
|
|
|
{
|
|
|
|
if (m_mapLocations.isEmpty())
|
|
|
|
return;
|
|
|
|
for(MapLocation *m: m_mapLocations)
|
|
|
|
m->m_selected = m_selectedDs.contains(m->divesite());
|
|
|
|
emit dataChanged(createIndex(0, 0), createIndex(m_mapLocations.size() - 1, 0));
|
|
|
|
}
|
|
|
|
|
2019-05-08 20:15:01 +00:00
|
|
|
void MapLocationModel::reload(QObject *map)
|
2019-05-08 19:35:49 +00:00
|
|
|
{
|
2019-05-01 21:33:45 +00:00
|
|
|
beginResetModel();
|
2017-07-29 05:01:33 +00:00
|
|
|
|
|
|
|
qDeleteAll(m_mapLocations);
|
|
|
|
m_mapLocations.clear();
|
2019-05-01 22:09:59 +00:00
|
|
|
m_selectedDs.clear();
|
2019-05-01 21:33:45 +00:00
|
|
|
|
|
|
|
QMap<QString, MapLocation *> locationNameMap;
|
|
|
|
|
|
|
|
#ifdef SUBSURFACE_MOBILE
|
|
|
|
bool diveSiteMode = false;
|
|
|
|
#else
|
|
|
|
// In dive site mode (that is when either editing a dive site or on
|
|
|
|
// the dive site tab), we want to show all dive sites, not only those
|
2019-05-03 21:37:42 +00:00
|
|
|
// of the non-hidden dives. Moreover, the selected dive sites are those
|
|
|
|
// that we filter for.
|
2019-05-01 21:33:45 +00:00
|
|
|
bool diveSiteMode = MultiFilterSortModel::instance()->diveSiteMode();
|
2019-05-03 21:37:42 +00:00
|
|
|
if (diveSiteMode)
|
|
|
|
m_selectedDs = MultiFilterSortModel::instance()->filteredDiveSites();
|
2019-05-01 21:33:45 +00:00
|
|
|
#endif
|
2019-05-08 19:35:49 +00:00
|
|
|
for (int i = 0; i < dive_site_table.nr; ++i) {
|
|
|
|
struct dive_site *ds = dive_site_table.dive_sites[i];
|
2019-05-08 20:15:01 +00:00
|
|
|
QGeoCoordinate dsCoord;
|
2019-05-08 19:35:49 +00:00
|
|
|
|
|
|
|
// Don't show dive sites of hidden dives, unless we're in dive site edit mode.
|
|
|
|
if (!diveSiteMode && !hasVisibleDive(ds))
|
2019-05-01 21:33:45 +00:00
|
|
|
continue;
|
2019-05-08 20:15:01 +00:00
|
|
|
if (!dive_site_has_gps_location(ds)) {
|
|
|
|
// Dive sites that do not have a gps location are not shown in normal mode.
|
|
|
|
// In dive-edit mode, selected sites are placed at the center of the map,
|
|
|
|
// so that the user can drag them somewhere without having to enter coordinates.
|
|
|
|
if (!diveSiteMode || !m_selectedDs.contains(ds) || !map)
|
|
|
|
continue;
|
|
|
|
dsCoord = map->property("center").value<QGeoCoordinate>();
|
|
|
|
} else {
|
|
|
|
qreal latitude = ds->location.lat.udeg * 0.000001;
|
|
|
|
qreal longitude = ds->location.lon.udeg * 0.000001;
|
|
|
|
dsCoord = QGeoCoordinate(latitude, longitude);
|
|
|
|
}
|
2019-05-08 19:35:49 +00:00
|
|
|
if (!diveSiteMode && hasSelectedDive(ds) && !m_selectedDs.contains(ds))
|
2019-05-01 22:09:59 +00:00
|
|
|
m_selectedDs.append(ds);
|
2019-05-01 21:33:45 +00:00
|
|
|
QString name(ds->name);
|
2019-05-08 19:35:49 +00:00
|
|
|
if (!diveSiteMode) {
|
|
|
|
// don't add dive locations with the same name, unless they are
|
|
|
|
// at least MIN_DISTANCE_BETWEEN_DIVE_SITES_M apart
|
|
|
|
if (locationNameMap.contains(name)) {
|
|
|
|
MapLocation *existingLocation = locationNameMap[name];
|
|
|
|
QGeoCoordinate coord = existingLocation->coordinate();
|
|
|
|
if (dsCoord.distanceTo(coord) < MIN_DISTANCE_BETWEEN_DIVE_SITES_M)
|
|
|
|
continue;
|
|
|
|
}
|
2019-05-01 21:33:45 +00:00
|
|
|
}
|
2019-08-30 13:25:59 +00:00
|
|
|
bool selected = m_selectedDs.contains(ds);
|
|
|
|
MapLocation *location = new MapLocation(ds, dsCoord, name, selected);
|
2019-05-01 21:33:45 +00:00
|
|
|
m_mapLocations.append(location);
|
2019-07-27 20:30:09 +00:00
|
|
|
if (!diveSiteMode)
|
|
|
|
locationNameMap[name] = location;
|
2019-05-01 21:33:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
endResetModel();
|
2017-07-29 05:01:33 +00:00
|
|
|
}
|
|
|
|
|
2019-08-30 10:38:25 +00:00
|
|
|
void MapLocationModel::setSelected(struct dive_site *ds)
|
2018-10-26 15:03:54 +00:00
|
|
|
{
|
2019-05-01 22:09:59 +00:00
|
|
|
m_selectedDs.clear();
|
2019-08-30 10:38:25 +00:00
|
|
|
if (ds)
|
|
|
|
m_selectedDs.append(ds);
|
2018-10-26 15:03:54 +00:00
|
|
|
}
|
|
|
|
|
2019-08-30 15:38:54 +00:00
|
|
|
void MapLocationModel::setSelected(const QVector<dive_site *> &divesites)
|
|
|
|
{
|
|
|
|
m_selectedDs = divesites;
|
|
|
|
}
|
|
|
|
|
2019-05-01 22:09:59 +00:00
|
|
|
bool MapLocationModel::isSelected(const QVariant &dsVariant) const
|
2017-07-29 05:01:33 +00:00
|
|
|
{
|
2019-05-01 22:09:59 +00:00
|
|
|
dive_site *ds = dsVariant.value<dive_site *>();
|
|
|
|
return ds && m_selectedDs.contains(ds);
|
2017-07-29 05:01:33 +00:00
|
|
|
}
|
|
|
|
|
2018-10-26 15:03:54 +00:00
|
|
|
MapLocation *MapLocationModel::getMapLocation(const struct dive_site *ds)
|
2017-07-29 05:01:33 +00:00
|
|
|
{
|
|
|
|
MapLocation *location;
|
|
|
|
foreach(location, m_mapLocations) {
|
2018-10-26 15:03:54 +00:00
|
|
|
if (ds == location->divesite())
|
2017-07-29 05:01:33 +00:00
|
|
|
return location;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
2017-11-09 16:43:21 +00:00
|
|
|
|
2019-05-09 19:33:01 +00:00
|
|
|
void MapLocationModel::diveSiteChanged(struct dive_site *ds, int field)
|
2017-11-09 16:43:21 +00:00
|
|
|
{
|
2019-05-09 19:33:01 +00:00
|
|
|
// Find dive site
|
|
|
|
int row;
|
|
|
|
for (row = 0; row < m_mapLocations.size(); ++row) {
|
|
|
|
if (m_mapLocations[row]->divesite() == ds)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (row == m_mapLocations.size())
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (field) {
|
|
|
|
case LocationInformationModel::LOCATION:
|
|
|
|
if (has_location(&ds->location)) {
|
|
|
|
const qreal latitude_r = ds->location.lat.udeg * 0.000001;
|
|
|
|
const qreal longitude_r = ds->location.lon.udeg * 0.000001;
|
|
|
|
QGeoCoordinate coord(latitude_r, longitude_r);
|
|
|
|
m_mapLocations[row]->setCoordinateNoEmit(coord);
|
2017-11-09 16:43:21 +00:00
|
|
|
}
|
2019-05-09 19:33:01 +00:00
|
|
|
break;
|
|
|
|
case LocationInformationModel::NAME:
|
|
|
|
m_mapLocations[row]->setProperty("name", ds->name);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2017-11-09 16:43:21 +00:00
|
|
|
}
|
2019-05-09 19:33:01 +00:00
|
|
|
|
|
|
|
emit dataChanged(createIndex(row, 0), createIndex(row, 0));
|
2017-11-09 16:43:21 +00:00
|
|
|
}
|