| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							| 
									
										
										
										
											2023-11-18 13:12:18 +01:00
										 |  |  | // Copyright (C) 2022-2023 Jon Beniston, M7RCE <jon@beniston.com>                //
 | 
					
						
							|  |  |  | // Copyright (C) 2022 Edouard Griffiths, F4EXB <f4exb06@gmail.com>               //
 | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // This program is free software; you can redistribute it and/or modify          //
 | 
					
						
							|  |  |  | // it under the terms of the GNU General Public License as published by          //
 | 
					
						
							|  |  |  | // the Free Software Foundation as version 3 of the License, or                  //
 | 
					
						
							|  |  |  | // (at your option) any later version.                                           //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // This program is distributed in the hope that it will be useful,               //
 | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of                //
 | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the                  //
 | 
					
						
							|  |  |  | // GNU General Public License V3 for more details.                               //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // You should have received a copy of the GNU General Public License             //
 | 
					
						
							|  |  |  | // along with this program. If not, see <http://www.gnu.org/licenses/>.          //
 | 
					
						
							|  |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <QGeoRectangle>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "channel/channelwebapiutils.h"
 | 
					
						
							|  |  |  | #include "maincore.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "mapmodel.h"
 | 
					
						
							|  |  |  | #include "mapgui.h"
 | 
					
						
							|  |  |  | #include "map.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "SWGTargetAzimuthElevation.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | QVariant MapModel::data(const QModelIndex &index, int role) const | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     int row = index.row(); | 
					
						
							|  |  |  |     if ((row < 0) || (row >= m_items.count())) { | 
					
						
							|  |  |  |         return QVariant(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     switch (role) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case itemSettingsRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(m_items[row]->m_itemSettings); | 
					
						
							|  |  |  |     case nameRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(m_items[row]->m_name); | 
					
						
							|  |  |  |     case labelRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(m_items[row]->m_label); | 
					
						
							|  |  |  |     case positionRole: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Coordinates to display the label at
 | 
					
						
							|  |  |  |         QGeoCoordinate coords; | 
					
						
							|  |  |  |         coords.setLatitude(m_items[row]->m_latitude); | 
					
						
							|  |  |  |         coords.setLongitude(m_items[row]->m_longitude); | 
					
						
							|  |  |  |         coords.setAltitude(m_items[row]->m_altitude); | 
					
						
							|  |  |  |         return QVariant::fromValue(coords); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case mapImageMinZoomRole: | 
					
						
							|  |  |  |         // Minimum zoom level at which this is visible
 | 
					
						
							|  |  |  |         return QVariant::fromValue(m_items[row]->m_itemSettings->m_2DMinZoom); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return QVariant(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | bool MapModel::setData(const QModelIndex &index, const QVariant& value, int role) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     (void) value; | 
					
						
							|  |  |  |     (void) role; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int row = index.row(); | 
					
						
							|  |  |  |     if ((row < 0) || (row >= m_items.count())) { | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MapModel::add(MapItem *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     beginInsertRows(QModelIndex(), rowCount(), rowCount()); | 
					
						
							|  |  |  |     m_items.append(item); | 
					
						
							|  |  |  |     m_itemsHash.insert(item->m_hashKey, item); | 
					
						
							|  |  |  |     endInsertRows(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MapModel::update(const QObject *sourcePipe, SWGSDRangel::SWGMapItem *swgMapItem, const QString &group) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QString name = *swgMapItem->getName(); | 
					
						
							|  |  |  |     // Add, update or delete and item
 | 
					
						
							|  |  |  |     MapItem *item = findMapItem(sourcePipe, name); | 
					
						
							|  |  |  |     if (item != nullptr) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QString image = *swgMapItem->getImage(); | 
					
						
							|  |  |  |         if (image.isEmpty()) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-05-15 16:47:29 +01:00
										 |  |  |             // Delete the item from 2D map
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             remove(item); | 
					
						
							| 
									
										
										
										
											2023-05-15 16:47:29 +01:00
										 |  |  |             // Delete from 3D map
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             item->update(swgMapItem); | 
					
						
							| 
									
										
										
										
											2023-05-15 16:47:29 +01:00
										 |  |  |             update3D(item); | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Update the item
 | 
					
						
							|  |  |  |             item->update(swgMapItem); | 
					
						
							|  |  |  |             update(item); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         // Make sure not a duplicate request to delete
 | 
					
						
							|  |  |  |         QString image = *swgMapItem->getImage(); | 
					
						
							|  |  |  |         if (!image.isEmpty()) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Add new item
 | 
					
						
							|  |  |  |             item = newMapItem(sourcePipe, group, m_gui->getItemSettings(group), swgMapItem); | 
					
						
							|  |  |  |             add(item); | 
					
						
							|  |  |  |             // Add to 3D Map (we don't appear to get a dataChanged signal when adding)
 | 
					
						
							|  |  |  |             update3D(item); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void MapModel::update(MapItem *item) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     int row = m_items.indexOf(item); | 
					
						
							|  |  |  |     if (row >= 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QModelIndex idx = index(row); | 
					
						
							|  |  |  |         emit dataChanged(idx, idx); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void MapModel::remove(MapItem *item) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     int row = m_items.indexOf(item); | 
					
						
							|  |  |  |     if (row >= 0) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         QString key = m_items[row]->m_hashKey; | 
					
						
							|  |  |  |         beginRemoveRows(QModelIndex(), row, row); | 
					
						
							|  |  |  |         m_items.removeAt(row); | 
					
						
							|  |  |  |         m_itemsHash.remove(key); | 
					
						
							|  |  |  |         endRemoveRows(); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MapModel::removeAll() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_items.count() > 0) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         beginRemoveRows(QModelIndex(), 0, m_items.count() - 1); | 
					
						
							|  |  |  |         m_items.clear(); | 
					
						
							|  |  |  |         m_itemsHash.clear(); | 
					
						
							|  |  |  |         endRemoveRows(); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | // After new settings are deserialised, we need to update
 | 
					
						
							|  |  |  | // pointers to item settings for all existing items
 | 
					
						
							|  |  |  | void MapModel::updateItemSettings(QHash<QString, MapSettings::MapItemSettings *> m_itemSettings) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     for (auto item : m_items) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         if (m_itemSettings.contains(item->m_group)) { | 
					
						
							|  |  |  |             item->m_itemSettings = m_itemSettings[item->m_group]; | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void MapModel::allUpdated() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_items.count() > 0) { | 
					
						
							|  |  |  |         emit dataChanged(index(0), index(m_items.count()-1)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MapItem *MapModel::findMapItem(const QObject *source, const QString& name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QString key = source->objectName() + name; | 
					
						
							|  |  |  |     if (m_itemsHash.contains(key)) { | 
					
						
							|  |  |  |         return m_itemsHash.value(key); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 16:47:29 +01:00
										 |  |  | // FIXME: This should potentially return a list, as we have have multiple items with the same name
 | 
					
						
							|  |  |  | // from different sources
 | 
					
						
							|  |  |  | MapItem *MapModel::findMapItem(const QString& name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QListIterator<MapItem *> i(m_items); | 
					
						
							|  |  |  |     while (i.hasNext()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         MapItem *item = i.next(); | 
					
						
							|  |  |  |         if (!item->m_name.compare(name, Qt::CaseInsensitive)) { | 
					
						
							|  |  |  |             return item; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QModelIndex MapModel::findMapItemIndex(const QString& name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int idx = 0; | 
					
						
							|  |  |  |     QListIterator<MapItem *> i(m_items); | 
					
						
							|  |  |  |     while (i.hasNext()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         MapItem *item = i.next(); | 
					
						
							|  |  |  |         if (item->m_name == name) { | 
					
						
							|  |  |  |             return index(idx); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         idx++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return index(-1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | QHash<int, QByteArray> MapModel::roleNames() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QHash<int, QByteArray> roles; | 
					
						
							|  |  |  |     roles[itemSettingsRole] = "itemSettings"; | 
					
						
							|  |  |  |     roles[nameRole] = "name"; | 
					
						
							|  |  |  |     roles[labelRole] = "label"; | 
					
						
							|  |  |  |     roles[positionRole] = "position"; | 
					
						
							|  |  |  |     roles[mapImageMinZoomRole] = "mapImageMinZoom"; | 
					
						
							|  |  |  |     return roles; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Slot called on dataChanged signal, to update 3D map
 | 
					
						
							|  |  |  | void MapModel::update3DMap(const QModelIndex &topLeft, const QModelIndex &bottomRight, const QVector<int> &roles) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     (void) roles; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int row = topLeft.row(); row <= bottomRight.row(); row++) { | 
					
						
							|  |  |  |         update3D(m_items[row]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MapItem *ImageMapModel::newMapItem(const QObject *sourcePipe, const QString &group, MapSettings::MapItemSettings *itemSettings, SWGSDRangel::SWGMapItem *mapItem) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return new ImageMapItem(sourcePipe, group, itemSettings, mapItem); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QVariant ImageMapModel::data(const QModelIndex &index, int role) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int row = index.row(); | 
					
						
							|  |  |  |     if ((row < 0) || (row >= m_items.count())) { | 
					
						
							|  |  |  |         return QVariant(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ImageMapItem *mapItem = (ImageMapItem*)m_items[row]; | 
					
						
							|  |  |  |     switch (role) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case imageRole: | 
					
						
							|  |  |  |          return QVariant::fromValue(mapItem->m_image); | 
					
						
							|  |  |  |     case imageZoomLevelRole: | 
					
						
							|  |  |  |          return QVariant::fromValue(mapItem->m_imageZoomLevel); | 
					
						
							|  |  |  |     case boundsRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(mapItem->m_bounds); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return MapModel::data(index, role); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ImageMapModel::update3D(MapItem *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CesiumInterface *cesium = m_gui->cesium(); | 
					
						
							|  |  |  |     if (cesium) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         ImageMapItem *imageItem = (ImageMapItem *)item; | 
					
						
							|  |  |  |         if (!imageItem->m_image.isEmpty()) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             /*qDebug() << "ImageMapModel::update3D - " << imageItem->m_name
 | 
					
						
							|  |  |  |                                                      << imageItem->m_bounds.right() | 
					
						
							|  |  |  |                                                      << imageItem->m_bounds.left() | 
					
						
							|  |  |  |                                                      << imageItem->m_bounds.top() | 
					
						
							|  |  |  |                                                      << imageItem->m_bounds.bottom() | 
					
						
							|  |  |  |                                                      ;  */ | 
					
						
							|  |  |  |             cesium->updateImage(imageItem->m_name, | 
					
						
							|  |  |  |                                 imageItem->m_bounds.topRight().longitude(), | 
					
						
							|  |  |  |                                 imageItem->m_bounds.bottomLeft().longitude(), | 
					
						
							|  |  |  |                                 imageItem->m_bounds.topRight().latitude(), | 
					
						
							|  |  |  |                                 imageItem->m_bounds.bottomLeft().latitude(), | 
					
						
							|  |  |  |                                 imageItem->m_altitude, | 
					
						
							|  |  |  |                                 imageItem->m_image); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             qDebug() << "ImageMapModel::update3D - removeImage " << imageItem->m_name; | 
					
						
							|  |  |  |             cesium->removeImage(imageItem->m_name); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | MapItem *PolygonMapModel::newMapItem(const QObject *sourcePipe, const QString &group, MapSettings::MapItemSettings *itemSettings, SWGSDRangel::SWGMapItem *mapItem) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return new PolygonMapItem(sourcePipe, group, itemSettings, mapItem); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QVariant PolygonMapModel::data(const QModelIndex &index, int role) const | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     int row = index.row(); | 
					
						
							|  |  |  |     if ((row < 0) || (row >= m_items.count())) { | 
					
						
							|  |  |  |         return QVariant(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PolygonMapItem *polygonItem = ((PolygonMapItem *)m_items[row]); | 
					
						
							|  |  |  |     switch (role) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case borderColorRole: | 
					
						
							|  |  |  |          return QVariant::fromValue(QColor(0x00, 0x00, 0x00, 0x00)); // Transparent
 | 
					
						
							|  |  |  |     case fillColorRole: | 
					
						
							| 
									
										
										
										
											2023-03-21 11:07:25 +00:00
										 |  |  |         if (polygonItem->m_itemSettings->m_display2DTrack) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (polygonItem->m_colorValid) { | 
					
						
							|  |  |  |                 return QVariant::fromValue(QColor::fromRgba(polygonItem->m_color)); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 return QVariant::fromValue(QColor::fromRgba(polygonItem->m_itemSettings->m_2DTrackColor)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             return QVariant::fromValue(QColor(0x00, 0x00, 0x00, 0x00)); // Transparent
 | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case polygonRole: | 
					
						
							|  |  |  |         return polygonItem->m_polygon; | 
					
						
							|  |  |  |     case boundsRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(polygonItem->m_bounds); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return MapModel::data(index, role); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void PolygonMapModel::update3D(MapItem *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CesiumInterface *cesium = m_gui->cesium(); | 
					
						
							|  |  |  |     if (cesium) { | 
					
						
							|  |  |  |         cesium->update((PolygonMapItem *)item); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | MapItem *PolylineMapModel::newMapItem(const QObject *sourcePipe, const QString &group, MapSettings::MapItemSettings *itemSettings, SWGSDRangel::SWGMapItem *mapItem) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return new PolylineMapItem(sourcePipe, group, itemSettings, mapItem); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QVariant PolylineMapModel::data(const QModelIndex &index, int role) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int row = index.row(); | 
					
						
							|  |  |  |     if ((row < 0) || (row >= m_items.count())) { | 
					
						
							|  |  |  |         return QVariant(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PolylineMapItem *polylineItem = (PolylineMapItem *)m_items[row]; | 
					
						
							|  |  |  |     switch (role) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case lineColorRole: | 
					
						
							| 
									
										
										
										
											2023-03-21 11:07:25 +00:00
										 |  |  |         if (polylineItem->m_colorValid) { | 
					
						
							|  |  |  |             return QVariant::fromValue(QColor::fromRgba(polylineItem->m_color)); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             return QVariant::fromValue(QColor::fromRgba(polylineItem->m_itemSettings->m_2DTrackColor)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case coordinatesRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(polylineItem->m_polyline); | 
					
						
							|  |  |  |     case boundsRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(polylineItem->m_bounds); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return MapModel::data(index, role); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void PolylineMapModel::update3D(MapItem *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CesiumInterface *cesium = m_gui->cesium(); | 
					
						
							|  |  |  |     if (cesium) { | 
					
						
							|  |  |  |         cesium->update((PolylineMapItem *)item); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ObjectMapModel::ObjectMapModel(MapGUI *gui) : | 
					
						
							|  |  |  |     MapModel(gui), | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     m_target(-1) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     //connect(this, &ObjectMapModel::dataChanged, this, &ObjectMapModel::update3DMap);
 | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | Q_INVOKABLE void ObjectMapModel::add(MapItem *item) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     m_selected.append(false); | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     MapModel::add(item); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | /*void ObjectMapModel::update(const QObject *sourcePipe, SWGSDRangel::SWGMapItem *swgMapItem, const QString &group)
 | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     QString name = *swgMapItem->getName(); | 
					
						
							|  |  |  |     // Add, update or delete and item
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     ObjectMapItem *item = (ObjectMapItem *)MapModel::findMapItem(sourcePipe, name); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     if (item != nullptr) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QString image = *swgMapItem->getImage(); | 
					
						
							|  |  |  |         if (image.isEmpty()) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Delete the item
 | 
					
						
							|  |  |  |             remove(item); | 
					
						
							|  |  |  |             // Need to call update, for it to be removed in 3D map
 | 
					
						
							|  |  |  |             // Item is set to not be available from this point in time
 | 
					
						
							| 
									
										
										
										
											2022-03-21 10:14:35 +00:00
										 |  |  |             // It will still be available if time is set in the past
 | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |             item->update(swgMapItem); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Update the item
 | 
					
						
							|  |  |  |             item->update(swgMapItem); | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             splitTracks(item);            // ******** diff from base   FIXME
 | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |             update(item); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Make sure not a duplicate request to delete
 | 
					
						
							|  |  |  |         QString image = *swgMapItem->getImage(); | 
					
						
							|  |  |  |         if (!image.isEmpty()) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Add new item
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             item = (ObjectMapItem *)newMapItem(sourcePipe, group, m_gui->getItemSettings(group), swgMapItem); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |             add(item); | 
					
						
							|  |  |  |             // Add to 3D Map (we don't appear to get a dataChanged signal when adding)
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             update3D(item); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | }*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MapItem *ObjectMapModel::newMapItem(const QObject *sourcePipe, const QString &group, MapSettings::MapItemSettings *itemSettings, SWGSDRangel::SWGMapItem *mapItem) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return new ObjectMapItem(sourcePipe, group, itemSettings, mapItem); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ObjectMapModel::update3D(MapItem *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CesiumInterface *cesium = m_gui->cesium(); | 
					
						
							|  |  |  |     if (cesium) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         ObjectMapItem *objectMapItem = (ObjectMapItem *)item; | 
					
						
							|  |  |  |         cesium->update(objectMapItem, isTarget(objectMapItem), isSelected3D(objectMapItem)); | 
					
						
							|  |  |  |         playAnimations(objectMapItem); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Slot called on dataChanged signal, to update 3D map
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | /*void ObjectMapModel::update3DMap(const QModelIndex &topLeft, const QModelIndex &bottomRight, const QVector<int> &roles)
 | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-02-04 22:42:05 +00:00
										 |  |  |     (void) roles; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     CesiumInterface *cesium = m_gui->cesium(); | 
					
						
							|  |  |  |     if (cesium) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         for (int row = topLeft.row(); row <= bottomRight.row(); row++) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             ObjectMapItem *item = (ObjectMapItem *)m_items[row]; | 
					
						
							|  |  |  |             cesium->update(item, isTarget(item), isSelected3D(item)); | 
					
						
							|  |  |  |             playAnimations(item); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | } */ | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::playAnimations(ObjectMapItem *item) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     CesiumInterface *cesium = m_gui->cesium(); | 
					
						
							|  |  |  |     if (cesium) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         for (auto animation : item->m_animations) { | 
					
						
							|  |  |  |             m_gui->cesium()->playAnimation(item->m_name, animation); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     qDeleteAll(item->m_animations); | 
					
						
							|  |  |  |     item->m_animations.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::update(MapItem *item) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     splitTracks((ObjectMapItem *)item); | 
					
						
							|  |  |  |     MapModel::update(item); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     int row = m_items.indexOf(item); | 
					
						
							|  |  |  |     if (row >= 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (row == m_target) { | 
					
						
							|  |  |  |             updateTarget(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::remove(MapItem *item) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     int row = m_items.indexOf(item); | 
					
						
							|  |  |  |     if (row >= 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_selected.removeAt(row); | 
					
						
							|  |  |  |         if (row == m_target) { | 
					
						
							|  |  |  |             m_target = -1; | 
					
						
							|  |  |  |         } else if (row < m_target) { | 
					
						
							|  |  |  |             m_target--; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         MapModel::remove(item); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::removeAll() | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     MapModel::removeAll(); | 
					
						
							|  |  |  |     m_selected.clear(); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | QHash<int, QByteArray> ObjectMapModel::roleNames() const | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     QHash<int, QByteArray> roles = MapModel::roleNames(); | 
					
						
							|  |  |  |     //roles[itemSettingsRole] = "itemSettings";
 | 
					
						
							|  |  |  |     //roles[nameRole] = "name";
 | 
					
						
							|  |  |  |     //roles[positionRole] = "position";
 | 
					
						
							|  |  |  |     //roles[mapTextRole] = "mapText";
 | 
					
						
							|  |  |  |     roles[MapModel::labelRole] = "mapText"; | 
					
						
							|  |  |  |     roles[mapTextVisibleRole] = "mapTextVisible"; | 
					
						
							|  |  |  |     roles[mapImageVisibleRole] = "mapImageVisible"; | 
					
						
							|  |  |  |     roles[mapImageRole] = "mapImage"; | 
					
						
							|  |  |  |     roles[mapImageRotationRole] = "mapImageRotation"; | 
					
						
							|  |  |  |     //roles[mapImageMinZoomRole] = "mapImageMinZoom";
 | 
					
						
							|  |  |  |     roles[bubbleColourRole] = "bubbleColour"; | 
					
						
							|  |  |  |     roles[selectedRole] = "selected"; | 
					
						
							|  |  |  |     roles[targetRole] = "target"; | 
					
						
							| 
									
										
										
										
											2023-12-05 12:30:25 +00:00
										 |  |  |     roles[frequenciesRole] = "frequencies"; | 
					
						
							|  |  |  |     roles[frequencyStringsRole] = "frequencyStrings"; | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     roles[predictedGroundTrack1Role] = "predictedGroundTrack1"; | 
					
						
							|  |  |  |     roles[predictedGroundTrack2Role] = "predictedGroundTrack2"; | 
					
						
							|  |  |  |     roles[groundTrack1Role] = "groundTrack1"; | 
					
						
							|  |  |  |     roles[groundTrack2Role] = "groundTrack2"; | 
					
						
							|  |  |  |     roles[groundTrackColorRole] = "groundTrackColor"; | 
					
						
							|  |  |  |     roles[predictedGroundTrackColorRole] = "predictedGroundTrackColor"; | 
					
						
							|  |  |  |     roles[hasTracksRole] = "hasTracks"; | 
					
						
							|  |  |  |     return roles; | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::updateTarget() | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     // Calculate range, azimuth and elevation to object from station
 | 
					
						
							|  |  |  |     AzEl *azEl = m_gui->getAzEl(); | 
					
						
							|  |  |  |     azEl->setTarget(m_items[m_target]->m_latitude, m_items[m_target]->m_longitude, m_items[m_target]->m_altitude); | 
					
						
							|  |  |  |     azEl->calculate(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Send to Rotator Controllers
 | 
					
						
							| 
									
										
										
										
											2022-03-28 20:12:25 +02:00
										 |  |  |     QList<ObjectPipe*> rotatorPipes; | 
					
						
							|  |  |  |     MainCore::instance()->getMessagePipes().getMessagePipes(this, "target", rotatorPipes); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-28 20:12:25 +02:00
										 |  |  |     for (const auto& pipe : rotatorPipes) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         MessageQueue *messageQueue = qobject_cast<MessageQueue*>(pipe->m_element); | 
					
						
							|  |  |  |         SWGSDRangel::SWGTargetAzimuthElevation *swgTarget = new SWGSDRangel::SWGTargetAzimuthElevation(); | 
					
						
							|  |  |  |         swgTarget->setName(new QString(m_items[m_target]->m_name)); | 
					
						
							|  |  |  |         swgTarget->setAzimuth(azEl->getAzimuth()); | 
					
						
							|  |  |  |         swgTarget->setElevation(azEl->getElevation()); | 
					
						
							|  |  |  |         messageQueue->push(MainCore::MsgTargetAzimuthElevation::create(m_gui->getMap(), swgTarget)); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::setTarget(const QString& name) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     if (name.isEmpty()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QModelIndex idx = index(-1); | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         setData(idx, QVariant(-1), ObjectMapModel::targetRole); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QModelIndex idx = findMapItemIndex(name); | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         setData(idx, QVariant(idx.row()), ObjectMapModel::targetRole); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | bool ObjectMapModel::isTarget(const ObjectMapItem *mapItem) const | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     if (m_target < 0) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         return m_items[m_target] == mapItem; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // FIXME: This should use Z order - rather than adding/removing
 | 
					
						
							|  |  |  | // but I couldn't quite get it to work
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | Q_INVOKABLE void ObjectMapModel::moveToFront(int oldRow) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     // Last item in list is drawn on top, so remove than add to end of list
 | 
					
						
							|  |  |  |     if (oldRow < m_items.size() - 1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         bool wasTarget = m_target == oldRow; | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         ObjectMapItem *item = (ObjectMapItem *)m_items[oldRow]; | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         bool wasSelected = m_selected[oldRow]; | 
					
						
							|  |  |  |         remove(item); | 
					
						
							|  |  |  |         add(item); | 
					
						
							|  |  |  |         int newRow = m_items.size() - 1; | 
					
						
							|  |  |  |         if (wasTarget) { | 
					
						
							|  |  |  |             m_target = newRow; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         m_selected[newRow] = wasSelected; | 
					
						
							|  |  |  |         QModelIndex idx = index(newRow); | 
					
						
							|  |  |  |         emit dataChanged(idx, idx); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | Q_INVOKABLE void ObjectMapModel::moveToBack(int oldRow) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     // First item in list is drawn first, so remove item then add to front of list
 | 
					
						
							|  |  |  |     if ((oldRow < m_items.size()) && (oldRow > 0)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         bool wasTarget = m_target == oldRow; | 
					
						
							|  |  |  |         int newRow = 0; | 
					
						
							|  |  |  |         // See: https://forum.qt.io/topic/122991/changing-the-order-mapquickitems-are-drawn-on-a-map
 | 
					
						
							|  |  |  |         //QModelIndex parent;
 | 
					
						
							|  |  |  |         //beginMoveRows(parent, oldRow, oldRow, parent, newRow);
 | 
					
						
							|  |  |  |         beginResetModel(); | 
					
						
							|  |  |  |         m_items.move(oldRow, newRow); | 
					
						
							|  |  |  |         m_selected.move(oldRow, newRow); | 
					
						
							|  |  |  |         if (wasTarget) { | 
					
						
							|  |  |  |             m_target = newRow; | 
					
						
							|  |  |  |         } else if (m_target >= 0) { | 
					
						
							|  |  |  |             m_target++; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         //endMoveRows();
 | 
					
						
							|  |  |  |         endResetModel(); | 
					
						
							|  |  |  |         //emit dataChanged(index(oldRow), index(newRow));
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 15:40:06 +00:00
										 |  |  | Q_INVOKABLE void ObjectMapModel::link(const QString& url) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     emit linkClicked(url); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | QVariant ObjectMapModel::data(const QModelIndex &index, int role) const | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     int row = index.row(); | 
					
						
							|  |  |  |     if ((row < 0) || (row >= m_items.count())) { | 
					
						
							|  |  |  |         return QVariant(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     ObjectMapItem *mapItem = (ObjectMapItem*)m_items[row]; | 
					
						
							|  |  |  |     switch (role) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case labelRole: // mapTextRole)
 | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         // Create the text to go in the bubble next to the image
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         QString name = mapItem->m_label.isEmpty() ? mapItem->m_name :mapItem->m_label; | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         if (row == m_target) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             AzEl *azEl = m_gui->getAzEl(); | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             QString text = QString("%1<br>Az: %2%5 El: %3%5 Dist: %4 km<br>Coords: %6, %7") | 
					
						
							|  |  |  |                                 .arg(m_selected[row] ? mapItem->m_text : name) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |                                 .arg(std::round(azEl->getAzimuth())) | 
					
						
							|  |  |  |                                 .arg(std::round(azEl->getElevation())) | 
					
						
							|  |  |  |                                 .arg(std::round(azEl->getDistance() / 1000.0)) | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |                                 .arg(QChar(0xb0)) | 
					
						
							|  |  |  |                                 .arg(mapItem->m_latitude) | 
					
						
							|  |  |  |                                 .arg(mapItem->m_longitude); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |             return QVariant::fromValue(text); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (m_selected[row]) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             return QVariant::fromValue(mapItem->m_text); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-03-21 10:14:35 +00:00
										 |  |  |             return QVariant::fromValue(name); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case mapTextVisibleRole: | 
					
						
							|  |  |  |         return QVariant::fromValue((m_selected[row] || m_displayNames) && mapItem->m_itemSettings->m_display2DLabel); | 
					
						
							|  |  |  |     case mapImageVisibleRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(mapItem->m_itemSettings->m_display2DIcon); | 
					
						
							|  |  |  |     case mapImageRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(mapItem->m_image); // Set an image to use
 | 
					
						
							|  |  |  |     case mapImageRotationRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(mapItem->m_imageRotation);  // Angle to rotate image by
 | 
					
						
							|  |  |  |     case bubbleColourRole: | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         // Select a background colour for the text bubble next to the item
 | 
					
						
							|  |  |  |         if (m_selected[row]) { | 
					
						
							|  |  |  |             return QVariant::fromValue(QColor("lightgreen")); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             return QVariant::fromValue(QColor("lightblue")); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case selectedRole: | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         return QVariant::fromValue(m_selected[row]); | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case targetRole: | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         return QVariant::fromValue(m_target == row); | 
					
						
							| 
									
										
										
										
											2023-12-05 12:30:25 +00:00
										 |  |  |     case frequenciesRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(mapItem->m_frequencies); | 
					
						
							|  |  |  |     case frequencyStringsRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(mapItem->m_frequencyStrings); | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case predictedGroundTrack1Role: | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         if (   (m_displayAllGroundTracks || (m_displaySelectedGroundTracks && m_selected[row])) | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             && mapItem->m_itemSettings->m_display2DTrack) { | 
					
						
							|  |  |  |             return mapItem->m_predictedTrack1; | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             return QVariantList(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case predictedGroundTrack2Role: | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         if (   (m_displayAllGroundTracks || (m_displaySelectedGroundTracks && m_selected[row])) | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             && mapItem->m_itemSettings->m_display2DTrack) { | 
					
						
							|  |  |  |             return mapItem->m_predictedTrack2; | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             return QVariantList(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case groundTrack1Role: | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         if (   (m_displayAllGroundTracks || (m_displaySelectedGroundTracks && m_selected[row])) | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             && mapItem->m_itemSettings->m_display2DTrack) { | 
					
						
							|  |  |  |             return mapItem->m_takenTrack1; | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             return QVariantList(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case groundTrack2Role: | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         if (   (m_displayAllGroundTracks || (m_displaySelectedGroundTracks && m_selected[row])) | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             && mapItem->m_itemSettings->m_display2DTrack) { | 
					
						
							|  |  |  |             return mapItem->m_takenTrack2; | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             return QVariantList(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     case groundTrackColorRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(QColor::fromRgb(mapItem->m_itemSettings->m_2DTrackColor)); | 
					
						
							|  |  |  |     case predictedGroundTrackColorRole: | 
					
						
							|  |  |  |         return QVariant::fromValue(QColor::fromRgb(mapItem->m_itemSettings->m_2DTrackColor).lighter()); | 
					
						
							|  |  |  |     case hasTracksRole: | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |         bool hasTracks =    (m_displayAllGroundTracks || (m_displaySelectedGroundTracks && m_selected[row])) | 
					
						
							|  |  |  |                          && (   (mapItem->m_predictedTrack1.size() > 1) | 
					
						
							|  |  |  |                              || (mapItem->m_predictedTrack2.size() > 1) | 
					
						
							|  |  |  |                              || (mapItem->m_takenTrack1.size() > 1) | 
					
						
							|  |  |  |                              || (mapItem->m_takenTrack2.size() > 1) | 
					
						
							|  |  |  |                             ); | 
					
						
							|  |  |  |         return QVariant::fromValue(hasTracks); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     default: | 
					
						
							|  |  |  |         return MapModel::data(index, role); | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | bool ObjectMapModel::setData(const QModelIndex &idx, const QVariant& value, int role) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     int row = idx.row(); | 
					
						
							|  |  |  |     if ((row < 0) || (row >= m_items.count())) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     if (role == selectedRole) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         m_selected[row] = value.toBool(); | 
					
						
							|  |  |  |         emit dataChanged(idx, idx); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     else if (role == targetRole) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         if (m_target >= 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Update text bubble for old target
 | 
					
						
							|  |  |  |             QModelIndex oldIdx = index(m_target); | 
					
						
							|  |  |  |             m_target = -1; | 
					
						
							|  |  |  |             emit dataChanged(oldIdx, oldIdx); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         m_target = row; | 
					
						
							|  |  |  |         updateTarget(); | 
					
						
							|  |  |  |         emit dataChanged(idx, idx); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::setDisplayNames(bool displayNames) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     m_displayNames = displayNames; | 
					
						
							|  |  |  |     allUpdated(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::setDisplaySelectedGroundTracks(bool displayGroundTracks) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     m_displaySelectedGroundTracks = displayGroundTracks; | 
					
						
							|  |  |  |     allUpdated(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::setDisplayAllGroundTracks(bool displayGroundTracks) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     m_displayAllGroundTracks = displayGroundTracks; | 
					
						
							|  |  |  |     allUpdated(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::track3D(int index) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     if (index < m_items.count()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         MapItem *item = m_items[index]; | 
					
						
							|  |  |  |         m_gui->track3D(item->m_name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-05 12:30:25 +00:00
										 |  |  | QStringList ObjectMapModel::getDeviceSets() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return MainCore::instance()->getDeviceSetIds(true, true, false); // FIXME: MIMO currently disabled, as we can't get channel stream indexes
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ObjectMapModel::setFrequency(qint64 frequency, const QString& deviceSetId) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unsigned int deviceSetIndex; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (MainCore::getDeviceSetIndexFromId(deviceSetId, deviceSetIndex)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Set as centre frequency
 | 
					
						
							|  |  |  |         ChannelWebAPIUtils::setCenterFrequency(deviceSetIndex, frequency); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::splitTracks(ObjectMapItem *item) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     if (item->m_takenTrackCoords.size() > 1) | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         splitTrack(item->m_takenTrackCoords, item->m_takenTrack, item->m_takenTrack1, item->m_takenTrack2, | 
					
						
							|  |  |  |                         item->m_takenStart1, item->m_takenStart2, item->m_takenEnd1, item->m_takenEnd2); | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     if (item->m_predictedTrackCoords.size() > 1) | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |         splitTrack(item->m_predictedTrackCoords, item->m_predictedTrack, item->m_predictedTrack1, item->m_predictedTrack2, | 
					
						
							|  |  |  |                         item->m_predictedStart1, item->m_predictedStart2, item->m_predictedEnd1, item->m_predictedEnd2); | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::interpolateEast(QGeoCoordinate *c1, QGeoCoordinate *c2, double x, QGeoCoordinate *ci, bool offScreen) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     double x1 = c1->longitude(); | 
					
						
							|  |  |  |     double y1 = c1->latitude(); | 
					
						
							|  |  |  |     double x2 = c2->longitude(); | 
					
						
							|  |  |  |     double y2 = c2->latitude(); | 
					
						
							|  |  |  |     double y; | 
					
						
							|  |  |  |     if (x2 < x1) | 
					
						
							|  |  |  |         x2 += 360.0; | 
					
						
							|  |  |  |     if (x < x1) | 
					
						
							|  |  |  |         x += 360.0; | 
					
						
							|  |  |  |     y = interpolate(x1, y1, x2, y2, x); | 
					
						
							|  |  |  |     if (x > 180) | 
					
						
							|  |  |  |         x -= 360.0; | 
					
						
							|  |  |  |     if (offScreen) | 
					
						
							|  |  |  |         x -= 0.000000001; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         x += 0.000000001; | 
					
						
							|  |  |  |     ci->setLongitude(x); | 
					
						
							|  |  |  |     ci->setLatitude(y); | 
					
						
							|  |  |  |     ci->setAltitude(c1->altitude()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::interpolateWest(QGeoCoordinate *c1, QGeoCoordinate *c2, double x, QGeoCoordinate *ci, bool offScreen) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     double x1 = c1->longitude(); | 
					
						
							|  |  |  |     double y1 = c1->latitude(); | 
					
						
							|  |  |  |     double x2 = c2->longitude(); | 
					
						
							|  |  |  |     double y2 = c2->latitude(); | 
					
						
							|  |  |  |     double y; | 
					
						
							|  |  |  |     if (x2 > x1) | 
					
						
							|  |  |  |         x2 -= 360.0; | 
					
						
							|  |  |  |     if (x > x1) | 
					
						
							|  |  |  |         x -= 360.0; | 
					
						
							|  |  |  |     y = interpolate(x1, y1, x2, y2, x); | 
					
						
							|  |  |  |     if (x < -180) | 
					
						
							|  |  |  |         x += 360.0; | 
					
						
							|  |  |  |     if (offScreen) | 
					
						
							|  |  |  |         x += 0.000000001; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         x -= 0.000000001; | 
					
						
							|  |  |  |     ci->setLongitude(x); | 
					
						
							|  |  |  |     ci->setLatitude(y); | 
					
						
							|  |  |  |     ci->setAltitude(c1->altitude()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool isOnScreen(double lon, double bottomLeftLongitude, double bottomRightLongitude, double width, bool antimed) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool onScreen = false; | 
					
						
							|  |  |  |     if (width == 360) | 
					
						
							|  |  |  |         onScreen = true; | 
					
						
							|  |  |  |     else if (!antimed) | 
					
						
							|  |  |  |         onScreen = (lon > bottomLeftLongitude) && (lon <= bottomRightLongitude); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         onScreen = (lon > bottomLeftLongitude) || (lon <= bottomRightLongitude); | 
					
						
							|  |  |  |     return onScreen; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool crossesAntimeridian(double prevLon, double lon) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool crosses = false; | 
					
						
							|  |  |  |     if ((prevLon > 90) && (lon < -90)) | 
					
						
							|  |  |  |         crosses = true; // West to East
 | 
					
						
							|  |  |  |     else if ((prevLon < -90) && (lon > 90)) | 
					
						
							|  |  |  |         crosses = true; // East to West
 | 
					
						
							|  |  |  |     return crosses; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool crossesAntimeridianEast(double prevLon, double lon) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool crosses = false; | 
					
						
							|  |  |  |     if ((prevLon > 90) && (lon < -90)) | 
					
						
							|  |  |  |         crosses = true; // West to East
 | 
					
						
							|  |  |  |     return crosses; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool crossesAntimeridianWest(double prevLon, double lon) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool crosses = false; | 
					
						
							|  |  |  |     if ((prevLon < -90) && (lon > 90)) | 
					
						
							|  |  |  |         crosses = true; // East to West
 | 
					
						
							|  |  |  |     return crosses; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool crossesEdge(double lon, double prevLon, double bottomLeftLongitude, double bottomRightLongitude) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // Determine if antimerdian is between the two points
 | 
					
						
							|  |  |  |     if (!crossesAntimeridian(prevLon, lon)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         bool crosses = false; | 
					
						
							|  |  |  |         if ((prevLon <= bottomRightLongitude) && (lon > bottomRightLongitude)) | 
					
						
							|  |  |  |             crosses = true; // Crosses right edge East
 | 
					
						
							|  |  |  |         else if ((prevLon >= bottomRightLongitude) && (lon < bottomRightLongitude)) | 
					
						
							|  |  |  |             crosses = true; // Crosses right edge West
 | 
					
						
							|  |  |  |         else if ((prevLon >= bottomLeftLongitude) && (lon < bottomLeftLongitude)) | 
					
						
							|  |  |  |             crosses = true; // Crosses left edge West
 | 
					
						
							|  |  |  |         else if ((prevLon <= bottomLeftLongitude) && (lon > bottomLeftLongitude)) | 
					
						
							|  |  |  |             crosses = true; // Crosses left edge East
 | 
					
						
							|  |  |  |         return crosses; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Determine which point and the edge the antimerdian is between
 | 
					
						
							|  |  |  |         bool prevLonToRightCrossesAnti = crossesAntimeridianEast(prevLon, bottomRightLongitude); | 
					
						
							|  |  |  |         bool rightToLonCrossesAnti = crossesAntimeridianEast(bottomRightLongitude, lon); | 
					
						
							|  |  |  |         bool prevLonToLeftCrossesAnti = crossesAntimeridianWest(prevLon, bottomLeftLongitude); | 
					
						
							|  |  |  |         bool leftToLonCrossesAnti = crossesAntimeridianWest(bottomLeftLongitude, lon); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         bool crosses = false; | 
					
						
							|  |  |  |         if (   ((prevLon > bottomRightLongitude) && prevLonToRightCrossesAnti && (lon > bottomRightLongitude)) | 
					
						
							|  |  |  |             || ((prevLon <= bottomRightLongitude) && (lon <= bottomRightLongitude) && rightToLonCrossesAnti) | 
					
						
							|  |  |  |            ) | 
					
						
							|  |  |  |             crosses = true; // Crosses right edge East
 | 
					
						
							|  |  |  |         else if (   ((prevLon < bottomRightLongitude) && prevLonToRightCrossesAnti && (lon < bottomRightLongitude)) | 
					
						
							|  |  |  |                  || ((prevLon >= bottomRightLongitude) && (lon >= bottomRightLongitude) && rightToLonCrossesAnti) | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |             crosses = true; // Crosses right edge West
 | 
					
						
							|  |  |  |         else if (   ((prevLon < bottomLeftLongitude) && prevLonToLeftCrossesAnti && (lon < bottomLeftLongitude)) | 
					
						
							|  |  |  |                  || ((prevLon >= bottomLeftLongitude) && (lon >= bottomLeftLongitude) && leftToLonCrossesAnti) | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |             crosses = true; // Crosses left edge West
 | 
					
						
							|  |  |  |         else if (   ((prevLon > bottomLeftLongitude) && prevLonToLeftCrossesAnti && (lon > bottomLeftLongitude)) | 
					
						
							|  |  |  |                  || ((prevLon <= bottomLeftLongitude) && (lon <= bottomLeftLongitude) && leftToLonCrossesAnti) | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |             crosses = true; // Crosses left edge East
 | 
					
						
							|  |  |  |         return crosses; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::interpolate(QGeoCoordinate *c1, QGeoCoordinate *c2, double bottomLeftLongitude, double bottomRightLongitude, QGeoCoordinate* ci, bool offScreen) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     double x1 = c1->longitude(); | 
					
						
							|  |  |  |     double x2 = c2->longitude(); | 
					
						
							|  |  |  |     double crossesAnti = crossesAntimeridian(x1, x2); | 
					
						
							|  |  |  |     double x; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Need to work out which edge we're interpolating too
 | 
					
						
							|  |  |  |     // and whether antimeridian is in the way, as that flips x1<x2 to x1>x2
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (((x1 < x2) && !crossesAnti) || ((x1 > x2) && crossesAnti)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         x = offScreen ? bottomRightLongitude : bottomLeftLongitude; | 
					
						
							|  |  |  |         interpolateEast(c1, c2, x, ci, offScreen); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         x = offScreen ? bottomLeftLongitude : bottomRightLongitude; | 
					
						
							|  |  |  |         interpolateWest(c1, c2, x, ci, offScreen); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::splitTrack(const QList<QGeoCoordinate *>& coords, const QVariantList& track, | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |                         QVariantList& track1, QVariantList& track2, | 
					
						
							|  |  |  |                         QGeoCoordinate& start1, QGeoCoordinate& start2, | 
					
						
							|  |  |  |                         QGeoCoordinate& end1, QGeoCoordinate& end2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /*
 | 
					
						
							|  |  |  |     QStringList l; | 
					
						
							|  |  |  |     for (int i = 0; i < track.size(); i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QGeoCoordinate c = track[i].value<QGeoCoordinate>(); | 
					
						
							|  |  |  |         l.append(QString("%1").arg((int)c.longitude())); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     qDebug() << "Init T: " << l; | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QQuickItem* map = m_gui->getMapItem(); | 
					
						
							|  |  |  |     QVariant rectVariant; | 
					
						
							|  |  |  |     QMetaObject::invokeMethod(map, "mapRect", Q_RETURN_ARG(QVariant, rectVariant)); | 
					
						
							|  |  |  |     QGeoRectangle rect = qvariant_cast<QGeoRectangle>(rectVariant); | 
					
						
							|  |  |  |     double bottomLeftLongitude = rect.bottomLeft().longitude(); | 
					
						
							|  |  |  |     double bottomRightLongitude = rect.bottomRight().longitude(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int width = round(rect.width()); | 
					
						
							|  |  |  |     bool antimed = (width == 360) || (bottomLeftLongitude > bottomRightLongitude); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /*
 | 
					
						
							|  |  |  |     qDebug() << "Anitmed visible: " << antimed; | 
					
						
							|  |  |  |     qDebug() << "bottomLeftLongitude: " << bottomLeftLongitude; | 
					
						
							|  |  |  |     qDebug() << "bottomRightLongitude: " << bottomRightLongitude; | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     track1.clear(); | 
					
						
							|  |  |  |     track2.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     double lon, prevLon; | 
					
						
							|  |  |  |     bool onScreen, prevOnScreen; | 
					
						
							|  |  |  |     QList<QVariantList *> tracks({&track1, &track2}); | 
					
						
							|  |  |  |     QList<QGeoCoordinate *> ends({&end1, &end2}); | 
					
						
							|  |  |  |     QList<QGeoCoordinate *> starts({&start1, &start2}); | 
					
						
							|  |  |  |     int trackIdx = 0; | 
					
						
							|  |  |  |     for (int i = 0; i < coords.size(); i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         lon = coords[i]->longitude(); | 
					
						
							|  |  |  |         if (i == 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             prevLon = lon; | 
					
						
							|  |  |  |             prevOnScreen = true; // To avoid interpolation for first point
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // Can be onscreen after having crossed edge from other side
 | 
					
						
							|  |  |  |         // Or can be onscreen after previously having been off screen
 | 
					
						
							|  |  |  |         onScreen = isOnScreen(lon, bottomLeftLongitude, bottomRightLongitude, width, antimed); | 
					
						
							|  |  |  |         bool crossedEdge = crossesEdge(lon, prevLon, bottomLeftLongitude, bottomRightLongitude); | 
					
						
							|  |  |  |         if ((onScreen && !crossedEdge) || (onScreen && !prevOnScreen)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if ((i > 0) && (tracks[trackIdx]->size() == 0)) // Could also use (onScreen && !prevOnScreen)?
 | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 if (trackIdx >= starts.size()) | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 // Interpolate from edge of screen
 | 
					
						
							|  |  |  |                 interpolate(coords[i-1], coords[i], bottomLeftLongitude, bottomRightLongitude, starts[trackIdx], false); | 
					
						
							|  |  |  |                 tracks[trackIdx]->append(QVariant::fromValue(*starts[trackIdx])); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             tracks[trackIdx]->append(track[i]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (tracks[trackIdx]->size() > 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             // Either we've crossed to the other side, or have gone off screen
 | 
					
						
							|  |  |  |             if (trackIdx >= ends.size()) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             // Interpolate to edge of screen
 | 
					
						
							|  |  |  |             interpolate(coords[i-1], coords[i], bottomLeftLongitude, bottomRightLongitude, ends[trackIdx], true); | 
					
						
							|  |  |  |             tracks[trackIdx]->append(QVariant::fromValue(*ends[trackIdx])); | 
					
						
							|  |  |  |             // Start new track
 | 
					
						
							|  |  |  |             trackIdx++; | 
					
						
							|  |  |  |             if (trackIdx >= tracks.size()) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 // This can happen with highly retrograde orbits, where trace 90% of period
 | 
					
						
							|  |  |  |                 // will cover more than 360 degrees - delete last point as Map
 | 
					
						
							|  |  |  |                 // will not be able to display it properly
 | 
					
						
							|  |  |  |                 tracks[trackIdx-1]->removeLast(); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (onScreen) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 // Interpolate from edge of screen
 | 
					
						
							|  |  |  |                 interpolate(coords[i-1], coords[i], bottomLeftLongitude, bottomRightLongitude, starts[trackIdx], false); | 
					
						
							|  |  |  |                 tracks[trackIdx]->append(QVariant::fromValue(*starts[trackIdx])); | 
					
						
							|  |  |  |                 tracks[trackIdx]->append(track[i]); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         prevLon = lon; | 
					
						
							|  |  |  |         prevOnScreen = onScreen; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /*
 | 
					
						
							|  |  |  |     l.clear(); | 
					
						
							|  |  |  |     for (int i = 0; i < track1.size(); i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QGeoCoordinate c = track1[i].value<QGeoCoordinate>(); | 
					
						
							|  |  |  |         if (!c.isValid()) | 
					
						
							|  |  |  |             l.append("Invalid!"); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             l.append(QString("%1").arg(c.longitude(), 0, 'f', 1)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     qDebug() << "T1: " << l; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     l.clear(); | 
					
						
							|  |  |  |     for (int i = 0; i < track2.size(); i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QGeoCoordinate c = track2[i].value<QGeoCoordinate>(); | 
					
						
							|  |  |  |         if (!c.isValid()) | 
					
						
							|  |  |  |             l.append("Invalid!"); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         l.append(QString("%1").arg(c.longitude(), 0, 'f', 1)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     qDebug() << "T2: " << l; | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | void ObjectMapModel::viewChanged(double bottomLeftLongitude, double bottomRightLongitude) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     (void) bottomRightLongitude; | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |     if (!std::isnan(bottomLeftLongitude)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         for (int row = 0; row < m_items.size(); row++) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |             ObjectMapItem *item = (ObjectMapItem *)m_items[row]; | 
					
						
							|  |  |  |             if (m_items[row]->m_itemSettings->m_enabled) | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  |                 if (item->m_takenTrackCoords.size() > 1) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     splitTrack(item->m_takenTrackCoords, item->m_takenTrack, item->m_takenTrack1, item->m_takenTrack2, | 
					
						
							|  |  |  |                                     item->m_takenStart1, item->m_takenStart2, item->m_takenEnd1, item->m_takenEnd2); | 
					
						
							|  |  |  |                     QModelIndex idx = index(row); | 
					
						
							|  |  |  |                     QVector<int> roles = {groundTrack1Role, groundTrack2Role}; | 
					
						
							|  |  |  |                     emit dataChanged(idx, idx, roles); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (item->m_predictedTrackCoords.size() > 1) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     splitTrack(item->m_predictedTrackCoords, item->m_predictedTrack, item->m_predictedTrack1, item->m_predictedTrack2, | 
					
						
							|  |  |  |                                     item->m_predictedStart1, item->m_predictedStart2, item->m_predictedEnd1, item->m_predictedEnd2); | 
					
						
							|  |  |  |                     QModelIndex idx = index(row); | 
					
						
							|  |  |  |                     QVector<int> roles = {predictedGroundTrack1Role, predictedGroundTrack2Role}; | 
					
						
							|  |  |  |                     emit dataChanged(idx, idx, roles); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2022-02-04 20:40:43 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-14 14:46:08 +00:00
										 |  |  | bool ObjectMapFilter::filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QModelIndex index0 = sourceModel()->index(sourceRow, 0, sourceParent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     MapSettings::MapItemSettings *itemSettings = sourceModel()->data(index0, ObjectMapModel::itemSettingsRole).value<MapSettings::MapItemSettings *>(); | 
					
						
							|  |  |  |     if (!itemSettings->m_enabled) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Don't show tiny items when we're zoomed out
 | 
					
						
							|  |  |  |     int minZoom = sourceModel()->data(index0, ObjectMapModel::mapImageMinZoomRole).toInt(); | 
					
						
							|  |  |  |     if (minZoom - 3 >= m_zoomLevel) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Don't show off-screen items, unless they have tracks, as these may be on screen
 | 
					
						
							|  |  |  |     // In the future, we could calculate a bounding box for tracks, if helpful for performance
 | 
					
						
							|  |  |  |     QGeoCoordinate coord = sourceModel()->data(index0, ObjectMapModel::positionRole).value<QGeoCoordinate>(); | 
					
						
							|  |  |  |     float lat = coord.latitude(); | 
					
						
							|  |  |  |     float lon = coord.longitude(); | 
					
						
							|  |  |  |     bool onScreen = (lat >= m_bottomRightLatitude) && (lat <= m_topLeftLatitude) && (lon >= m_topLeftLongitude) && (lon <= m_bottomRightLongitude); | 
					
						
							|  |  |  |     if (!onScreen) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         bool hasTracks = sourceModel()->data(index0, ObjectMapModel::hasTracksRole).toBool(); | 
					
						
							|  |  |  |         if (!hasTracks) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Apply user filter
 | 
					
						
							|  |  |  |     if (!itemSettings->m_filterName.isEmpty()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QString name = sourceModel()->data(index0, ObjectMapModel::nameRole).toString(); | 
					
						
							|  |  |  |         QRegularExpressionMatch match = itemSettings->m_filterNameRE.match(name); | 
					
						
							|  |  |  |         if (!match.hasMatch()) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (itemSettings->m_filterDistance > 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QGeoCoordinate position = sourceModel()->data(index0, ObjectMapModel::positionRole).value<QGeoCoordinate>(); | 
					
						
							|  |  |  |         if (m_position.distanceTo(position) > itemSettings->m_filterDistance) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ObjectMapFilter::viewChanged(double topLeftLongitude, double topLeftLatitude, double bottomRightLongitude, double bottomRightLatitude, double zoomLevel) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_zoomLevel = zoomLevel; | 
					
						
							|  |  |  |     if (!std::isnan(topLeftLongitude)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_topLeftLongitude = topLeftLongitude; | 
					
						
							|  |  |  |         m_topLeftLatitude = topLeftLatitude; | 
					
						
							|  |  |  |         m_bottomRightLongitude = bottomRightLongitude; | 
					
						
							|  |  |  |         m_bottomRightLatitude = bottomRightLatitude; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     invalidateFilter(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Q_INVOKABLE int ObjectMapFilter::mapRowToSource(int row) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QModelIndex sourceIndex = mapToSource(index(row, 0)); | 
					
						
							|  |  |  |     return sourceIndex.row(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool PolygonFilter::filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QModelIndex index0 = sourceModel()->index(sourceRow, 0, sourceParent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     MapSettings::MapItemSettings *itemSettings = sourceModel()->data(index0, PolygonMapModel::itemSettingsRole).value<MapSettings::MapItemSettings *>(); | 
					
						
							|  |  |  |     if (!itemSettings->m_enabled) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Don't show tiny items when we're zoomed out
 | 
					
						
							|  |  |  |     int minZoom = sourceModel()->data(index0, PolygonMapModel::mapImageMinZoomRole).toInt(); | 
					
						
							|  |  |  |     if (minZoom - 3 >= m_zoomLevel) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Filter off-screen items
 | 
					
						
							|  |  |  |     QGeoRectangle bounds = sourceModel()->data(index0, PolygonMapModel::boundsRole).value<QGeoRectangle>(); | 
					
						
							|  |  |  |     if (!m_view.intersects(bounds)) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Apply user filter
 | 
					
						
							|  |  |  |     if (!itemSettings->m_filterName.isEmpty()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QString name = sourceModel()->data(index0, PolygonMapModel::nameRole).toString(); | 
					
						
							|  |  |  |         QRegularExpressionMatch match = itemSettings->m_filterNameRE.match(name); | 
					
						
							|  |  |  |         if (!match.hasMatch()) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (itemSettings->m_filterDistance > 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QGeoCoordinate position = sourceModel()->data(index0, PolygonMapModel::positionRole).value<QGeoCoordinate>(); | 
					
						
							|  |  |  |         if (m_position.distanceTo(position) > itemSettings->m_filterDistance) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void PolygonFilter::viewChanged(double topLeftLongitude, double topLeftLatitude, double bottomRightLongitude, double bottomRightLatitude, double zoomLevel) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_zoomLevel = zoomLevel; | 
					
						
							|  |  |  |     if (!std::isnan(topLeftLongitude)) { | 
					
						
							|  |  |  |         m_view = QGeoRectangle(QGeoCoordinate(topLeftLatitude, topLeftLongitude), QGeoCoordinate(bottomRightLatitude, bottomRightLongitude)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     invalidateFilter(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Q_INVOKABLE int PolygonFilter::mapRowToSource(int row) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QModelIndex sourceIndex = mapToSource(index(row, 0)); | 
					
						
							|  |  |  |     return sourceIndex.row(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool PolylineFilter::filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QModelIndex index0 = sourceModel()->index(sourceRow, 0, sourceParent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     MapSettings::MapItemSettings *itemSettings = sourceModel()->data(index0, PolylineMapModel::itemSettingsRole).value<MapSettings::MapItemSettings *>(); | 
					
						
							|  |  |  |     if (!itemSettings->m_enabled) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Don't show tiny items when we're zoomed out
 | 
					
						
							|  |  |  |     int minZoom = sourceModel()->data(index0, PolylineMapModel::mapImageMinZoomRole).toInt(); | 
					
						
							|  |  |  |     if (minZoom - 3 >= m_zoomLevel) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Filter off-screen items
 | 
					
						
							|  |  |  |     QGeoRectangle bounds = sourceModel()->data(index0, PolylineMapModel::boundsRole).value<QGeoRectangle>(); | 
					
						
							|  |  |  |     if (!m_view.intersects(bounds)) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Apply user filter
 | 
					
						
							|  |  |  |     if (!itemSettings->m_filterName.isEmpty()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QString name = sourceModel()->data(index0, PolylineMapModel::nameRole).toString(); | 
					
						
							|  |  |  |         QRegularExpressionMatch match = itemSettings->m_filterNameRE.match(name); | 
					
						
							|  |  |  |         if (!match.hasMatch()) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (itemSettings->m_filterDistance > 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QGeoCoordinate position = sourceModel()->data(index0, PolylineMapModel::positionRole).value<QGeoCoordinate>(); | 
					
						
							|  |  |  |         if (m_position.distanceTo(position) > itemSettings->m_filterDistance) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void PolylineFilter::viewChanged(double topLeftLongitude, double topLeftLatitude, double bottomRightLongitude, double bottomRightLatitude, double zoomLevel) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_zoomLevel = zoomLevel; | 
					
						
							|  |  |  |     if (!std::isnan(topLeftLongitude)) { | 
					
						
							|  |  |  |         m_view = QGeoRectangle(QGeoCoordinate(topLeftLatitude, topLeftLongitude), QGeoCoordinate(bottomRightLatitude, bottomRightLongitude)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     invalidateFilter(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Q_INVOKABLE int PolylineFilter::mapRowToSource(int row) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QModelIndex sourceIndex = mapToSource(index(row, 0)); | 
					
						
							|  |  |  |     return sourceIndex.row(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ImageFilter::filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QModelIndex index0 = sourceModel()->index(sourceRow, 0, sourceParent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     MapSettings::MapItemSettings *itemSettings = sourceModel()->data(index0, ImageMapModel::itemSettingsRole).value<MapSettings::MapItemSettings *>(); | 
					
						
							|  |  |  |     if (!itemSettings->m_enabled || !itemSettings->m_display2DIcon) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Don't show tiny items when we're zoomed out
 | 
					
						
							|  |  |  |     int minZoom = sourceModel()->data(index0, ImageMapModel::mapImageMinZoomRole).toInt(); | 
					
						
							|  |  |  |     if (minZoom - 3 >= m_zoomLevel) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Filter off-screen items
 | 
					
						
							|  |  |  |     QGeoRectangle bounds = sourceModel()->data(index0, ImageMapModel::boundsRole).value<QGeoRectangle>(); | 
					
						
							|  |  |  |     if (!m_view.intersects(bounds)) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Apply user filter
 | 
					
						
							|  |  |  |     if (!itemSettings->m_filterName.isEmpty()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QString name = sourceModel()->data(index0, ImageMapModel::nameRole).toString(); | 
					
						
							|  |  |  |         QRegularExpressionMatch match = itemSettings->m_filterNameRE.match(name); | 
					
						
							|  |  |  |         if (!match.hasMatch()) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (itemSettings->m_filterDistance > 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QGeoCoordinate position = sourceModel()->data(index0, ImageMapModel::positionRole).value<QGeoCoordinate>(); | 
					
						
							|  |  |  |         if (m_position.distanceTo(position) > itemSettings->m_filterDistance) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ImageFilter::viewChanged(double topLeftLongitude, double topLeftLatitude, double bottomRightLongitude, double bottomRightLatitude, double zoomLevel) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_zoomLevel = zoomLevel; | 
					
						
							|  |  |  |     if (!std::isnan(topLeftLongitude)) { | 
					
						
							|  |  |  |         m_view = QGeoRectangle(QGeoCoordinate(topLeftLatitude, topLeftLongitude), QGeoCoordinate(bottomRightLatitude, bottomRightLongitude)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     invalidateFilter(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Q_INVOKABLE int ImageFilter::mapRowToSource(int row) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QModelIndex sourceIndex = mapToSource(index(row, 0)); | 
					
						
							|  |  |  |     return sourceIndex.row(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void PolygonFilter::setPosition(const QGeoCoordinate& position) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_position = position; | 
					
						
							|  |  |  |     invalidateFilter(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void PolylineFilter::setPosition(const QGeoCoordinate& position) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_position = position; | 
					
						
							|  |  |  |     invalidateFilter(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ObjectMapFilter::setPosition(const QGeoCoordinate& position) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_position = position; | 
					
						
							|  |  |  |     invalidateFilter(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ImageFilter::setPosition(const QGeoCoordinate& position) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_position = position; | 
					
						
							|  |  |  |     invalidateFilter(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 |