mirror of
				https://github.com/ciphervance/supercell-wx.git
				synced 2025-10-31 06:20:06 +00:00 
			
		
		
		
	Move MarkerManager to using an ID system for markers, instead of index
This commit is contained in:
		
							parent
							
								
									236d7c1e35
								
							
						
					
					
						commit
						7a070b3e7f
					
				
					 8 changed files with 233 additions and 74 deletions
				
			
		|  | @ -38,6 +38,8 @@ public: | ||||||
| 
 | 
 | ||||||
|    std::string                                markerSettingsPath_ {""}; |    std::string                                markerSettingsPath_ {""}; | ||||||
|    std::vector<std::shared_ptr<MarkerRecord>> markerRecords_ {}; |    std::vector<std::shared_ptr<MarkerRecord>> markerRecords_ {}; | ||||||
|  |    std::unordered_map<types::MarkerId, size_t> idToIndex_ {}; | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
|    MarkerManager* self_; |    MarkerManager* self_; | ||||||
| 
 | 
 | ||||||
|  | @ -48,6 +50,10 @@ public: | ||||||
|    void                          ReadMarkerSettings(); |    void                          ReadMarkerSettings(); | ||||||
|    void                          WriteMarkerSettings(); |    void                          WriteMarkerSettings(); | ||||||
|    std::shared_ptr<MarkerRecord> GetMarkerByName(const std::string& name); |    std::shared_ptr<MarkerRecord> GetMarkerByName(const std::string& name); | ||||||
|  | 
 | ||||||
|  |    void InitalizeIds(); | ||||||
|  |    types::MarkerId NewId(); | ||||||
|  |    types::MarkerId lastId_; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| class MarkerManager::Impl::MarkerRecord | class MarkerManager::Impl::MarkerRecord | ||||||
|  | @ -88,6 +94,16 @@ public: | ||||||
|    } |    } | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | void MarkerManager::Impl::InitalizeIds() | ||||||
|  | { | ||||||
|  |    lastId_ = 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | types::MarkerId MarkerManager::Impl::NewId() | ||||||
|  | { | ||||||
|  |    return ++lastId_; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void MarkerManager::Impl::InitializeMarkerSettings() | void MarkerManager::Impl::InitializeMarkerSettings() | ||||||
| { | { | ||||||
|    std::string appDataPath { |    std::string appDataPath { | ||||||
|  | @ -109,6 +125,7 @@ void MarkerManager::Impl::InitializeMarkerSettings() | ||||||
| void MarkerManager::Impl::ReadMarkerSettings() | void MarkerManager::Impl::ReadMarkerSettings() | ||||||
| { | { | ||||||
|    logger_->info("Reading location marker settings"); |    logger_->info("Reading location marker settings"); | ||||||
|  |    InitalizeIds(); | ||||||
| 
 | 
 | ||||||
|    boost::json::value markerJson = nullptr; |    boost::json::value markerJson = nullptr; | ||||||
|    { |    { | ||||||
|  | @ -125,6 +142,7 @@ void MarkerManager::Impl::ReadMarkerSettings() | ||||||
|          // For each marker entry
 |          // For each marker entry
 | ||||||
|          auto& markerArray = markerJson.as_array(); |          auto& markerArray = markerJson.as_array(); | ||||||
|          markerRecords_.reserve(markerArray.size()); |          markerRecords_.reserve(markerArray.size()); | ||||||
|  |          idToIndex_.reserve(markerArray.size()); | ||||||
|          for (auto& markerEntry : markerArray) |          for (auto& markerEntry : markerArray) | ||||||
|          { |          { | ||||||
|             try |             try | ||||||
|  | @ -134,8 +152,12 @@ void MarkerManager::Impl::ReadMarkerSettings() | ||||||
| 
 | 
 | ||||||
|                if (!record.markerInfo_.name.empty()) |                if (!record.markerInfo_.name.empty()) | ||||||
|                { |                { | ||||||
|  |                   types::MarkerId id    = NewId(); | ||||||
|  |                   size_t          index = markerRecords_.size(); | ||||||
|  |                   record.markerInfo_.id = id; | ||||||
|                   markerRecords_.emplace_back( |                   markerRecords_.emplace_back( | ||||||
|                      std::make_shared<MarkerRecord>(record.markerInfo_)); |                      std::make_shared<MarkerRecord>(record.markerInfo_)); | ||||||
|  |                   idToIndex_.emplace(id, index); | ||||||
|                } |                } | ||||||
|             } |             } | ||||||
|             catch (const std::exception& ex) |             catch (const std::exception& ex) | ||||||
|  | @ -206,11 +228,17 @@ size_t MarkerManager::marker_count() | ||||||
|    return p->markerRecords_.size(); |    return p->markerRecords_.size(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::optional<types::MarkerInfo> MarkerManager::get_marker(size_t index) | std::optional<types::MarkerInfo> MarkerManager::get_marker(types::MarkerId id) | ||||||
| { | { | ||||||
|    std::shared_lock lock(p->markerRecordLock_); |    std::shared_lock lock(p->markerRecordLock_); | ||||||
|  |    if (!p->idToIndex_.contains(id)) | ||||||
|  |    { | ||||||
|  |       return {}; | ||||||
|  |    } | ||||||
|  |    size_t index = p->idToIndex_[id]; | ||||||
|    if (index >= p->markerRecords_.size()) |    if (index >= p->markerRecords_.size()) | ||||||
|    { |    { | ||||||
|  |       logger_->warn("id in idToIndex_ but out of range!"); | ||||||
|       return {}; |       return {}; | ||||||
|    } |    } | ||||||
|    std::shared_ptr<MarkerManager::Impl::MarkerRecord>& markerRecord = |    std::shared_ptr<MarkerManager::Impl::MarkerRecord>& markerRecord = | ||||||
|  | @ -218,45 +246,81 @@ std::optional<types::MarkerInfo> MarkerManager::get_marker(size_t index) | ||||||
|    return markerRecord->toMarkerInfo(); |    return markerRecord->toMarkerInfo(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MarkerManager::set_marker(size_t index, const types::MarkerInfo& marker) | std::optional<size_t> MarkerManager::get_index(types::MarkerId id) | ||||||
|  | { | ||||||
|  |    std::shared_lock lock(p->markerRecordLock_); | ||||||
|  |    if (!p->idToIndex_.contains(id)) | ||||||
|  |    { | ||||||
|  |       return {}; | ||||||
|  |    } | ||||||
|  |    return p->idToIndex_[id]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void MarkerManager::set_marker(types::MarkerId id, const types::MarkerInfo& marker) | ||||||
| { | { | ||||||
|    { |    { | ||||||
|       std::unique_lock lock(p->markerRecordLock_); |       std::unique_lock lock(p->markerRecordLock_); | ||||||
|  |       if (!p->idToIndex_.contains(id)) | ||||||
|  |       { | ||||||
|  |          return; | ||||||
|  |       } | ||||||
|  |       size_t index = p->idToIndex_[id]; | ||||||
|       if (index >= p->markerRecords_.size()) |       if (index >= p->markerRecords_.size()) | ||||||
|       { |       { | ||||||
|  |          logger_->warn("id in idToIndex_ but out of range!"); | ||||||
|          return; |          return; | ||||||
|       } |       } | ||||||
|       std::shared_ptr<MarkerManager::Impl::MarkerRecord>& markerRecord = |       std::shared_ptr<MarkerManager::Impl::MarkerRecord>& markerRecord = | ||||||
|          p->markerRecords_[index]; |          p->markerRecords_[index]; | ||||||
|       markerRecord->markerInfo_ = marker; |       markerRecord->markerInfo_ = marker; | ||||||
|    } |    } | ||||||
|    Q_EMIT MarkerChanged(index); |    Q_EMIT MarkerChanged(id); | ||||||
|    Q_EMIT MarkersUpdated(); |    Q_EMIT MarkersUpdated(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MarkerManager::add_marker(const types::MarkerInfo& marker) | void MarkerManager::add_marker(const types::MarkerInfo& marker) | ||||||
| { | { | ||||||
|  |    types::MarkerId id; | ||||||
|    { |    { | ||||||
|       std::unique_lock lock(p->markerRecordLock_); |       std::unique_lock lock(p->markerRecordLock_); | ||||||
|  |       id = p->NewId(); | ||||||
|  |       size_t index = p->markerRecords_.size(); | ||||||
|  |       p->idToIndex_.emplace(id, index); | ||||||
|       p->markerRecords_.emplace_back(std::make_shared<Impl::MarkerRecord>(marker)); |       p->markerRecords_.emplace_back(std::make_shared<Impl::MarkerRecord>(marker)); | ||||||
|  |       p->markerRecords_[index]->markerInfo_.id = id; | ||||||
|    } |    } | ||||||
|    Q_EMIT MarkerAdded(); |    Q_EMIT MarkerAdded(id); | ||||||
|    Q_EMIT MarkersUpdated(); |    Q_EMIT MarkersUpdated(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MarkerManager::remove_marker(size_t index) | void MarkerManager::remove_marker(types::MarkerId id) | ||||||
| { | { | ||||||
|    { |    { | ||||||
|       std::unique_lock lock(p->markerRecordLock_); |       std::unique_lock lock(p->markerRecordLock_); | ||||||
|  |       if (!p->idToIndex_.contains(id)) | ||||||
|  |       { | ||||||
|  |          return; | ||||||
|  |       } | ||||||
|  |       size_t index = p->idToIndex_[id]; | ||||||
|       if (index >= p->markerRecords_.size()) |       if (index >= p->markerRecords_.size()) | ||||||
|       { |       { | ||||||
|  |          logger_->warn("id in idToIndex_ but out of range!"); | ||||||
|          return; |          return; | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       p->markerRecords_.erase(std::next(p->markerRecords_.begin(), index)); |       p->markerRecords_.erase(std::next(p->markerRecords_.begin(), index)); | ||||||
|  |       p->idToIndex_.erase(id); | ||||||
|  | 
 | ||||||
|  |       for (auto& pair : p->idToIndex_) | ||||||
|  |       { | ||||||
|  |          if (pair.second > index) | ||||||
|  |          { | ||||||
|  |             p->idToIndex_[pair.first] = pair.second - 1; | ||||||
|  |          } | ||||||
|  |       } | ||||||
|    } |    } | ||||||
| 
 | 
 | ||||||
|    Q_EMIT MarkerRemoved(index); |    Q_EMIT MarkerRemoved(id); | ||||||
|    Q_EMIT MarkersUpdated(); |    Q_EMIT MarkersUpdated(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -292,6 +356,15 @@ void MarkerManager::move_marker(size_t from, size_t to) | ||||||
|    Q_EMIT MarkersUpdated(); |    Q_EMIT MarkersUpdated(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void MarkerManager::for_each(std::function<MarkerForEachFunc> func) | ||||||
|  | { | ||||||
|  |    std::shared_lock lock(p->markerRecordLock_); | ||||||
|  |    for (auto marker : p->markerRecords_) | ||||||
|  |    { | ||||||
|  |       func(marker->markerInfo_); | ||||||
|  |    } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // Only use for testing
 | // Only use for testing
 | ||||||
| void MarkerManager::set_marker_settings_path(const std::string& path) | void MarkerManager::set_marker_settings_path(const std::string& path) | ||||||
| { | { | ||||||
|  |  | ||||||
|  | @ -12,6 +12,7 @@ namespace qt | ||||||
| namespace manager | namespace manager | ||||||
| { | { | ||||||
| 
 | 
 | ||||||
|  | typedef void MarkerForEachFunc(const types::MarkerInfo&); | ||||||
| class MarkerManager : public QObject | class MarkerManager : public QObject | ||||||
| { | { | ||||||
|    Q_OBJECT |    Q_OBJECT | ||||||
|  | @ -21,12 +22,15 @@ public: | ||||||
|    ~MarkerManager(); |    ~MarkerManager(); | ||||||
| 
 | 
 | ||||||
|    size_t                           marker_count(); |    size_t                           marker_count(); | ||||||
|    std::optional<types::MarkerInfo> get_marker(size_t index); |    std::optional<types::MarkerInfo> get_marker(types::MarkerId id); | ||||||
|    void set_marker(size_t index, const types::MarkerInfo& marker); |    std::optional<size_t> get_index(types::MarkerId id); | ||||||
|  |    void set_marker(types::MarkerId id, const types::MarkerInfo& marker); | ||||||
|    void add_marker(const types::MarkerInfo& marker); |    void add_marker(const types::MarkerInfo& marker); | ||||||
|    void remove_marker(size_t index); |    void remove_marker(types::MarkerId id); | ||||||
|    void move_marker(size_t from, size_t to); |    void move_marker(size_t from, size_t to); | ||||||
| 
 | 
 | ||||||
|  |    void for_each(std::function<MarkerForEachFunc> func); | ||||||
|  | 
 | ||||||
|    // Only use for testing
 |    // Only use for testing
 | ||||||
|    void set_marker_settings_path(const std::string& path); |    void set_marker_settings_path(const std::string& path); | ||||||
| 
 | 
 | ||||||
|  | @ -35,9 +39,9 @@ public: | ||||||
| signals: | signals: | ||||||
|    void MarkersInitialized(size_t count); |    void MarkersInitialized(size_t count); | ||||||
|    void MarkersUpdated(); |    void MarkersUpdated(); | ||||||
|    void MarkerChanged(size_t index); |    void MarkerChanged(types::MarkerId id); | ||||||
|    void MarkerAdded(); |    void MarkerAdded(types::MarkerId id); | ||||||
|    void MarkerRemoved(size_t index); |    void MarkerRemoved(types::MarkerId id); | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|    class Impl; |    class Impl; | ||||||
|  |  | ||||||
|  | @ -55,17 +55,13 @@ void MarkerLayer::Impl::ReloadMarkers() | ||||||
| 
 | 
 | ||||||
|    geoIcons_->StartIcons(); |    geoIcons_->StartIcons(); | ||||||
| 
 | 
 | ||||||
|    for (size_t i = 0; i < markerManager->marker_count(); i++) |    markerManager->for_each( | ||||||
|  |       [this](const types::MarkerInfo& marker) | ||||||
|       { |       { | ||||||
|       std::optional<types::MarkerInfo> marker = markerManager->get_marker(i); |  | ||||||
|       if (!marker) |  | ||||||
|       { |  | ||||||
|          break; |  | ||||||
|       } |  | ||||||
|          std::shared_ptr<gl::draw::GeoIconDrawItem> icon = geoIcons_->AddIcon(); |          std::shared_ptr<gl::draw::GeoIconDrawItem> icon = geoIcons_->AddIcon(); | ||||||
|          geoIcons_->SetIconTexture(icon, markerIconName_, 0); |          geoIcons_->SetIconTexture(icon, markerIconName_, 0); | ||||||
|       geoIcons_->SetIconLocation(icon, marker->latitude, marker->longitude); |          geoIcons_->SetIconLocation(icon, marker.latitude, marker.longitude); | ||||||
|    } |       }); | ||||||
| 
 | 
 | ||||||
|    geoIcons_->FinishIcons(); |    geoIcons_->FinishIcons(); | ||||||
|    Q_EMIT self_->NeedsRendering(); |    Q_EMIT self_->NeedsRendering(); | ||||||
|  |  | ||||||
|  | @ -5,6 +5,8 @@ | ||||||
| #include <scwx/qt/types/qt_types.hpp> | #include <scwx/qt/types/qt_types.hpp> | ||||||
| #include <scwx/util/logger.hpp> | #include <scwx/util/logger.hpp> | ||||||
| 
 | 
 | ||||||
|  | #include <vector> | ||||||
|  | 
 | ||||||
| #include <QApplication> | #include <QApplication> | ||||||
| 
 | 
 | ||||||
| namespace scwx | namespace scwx | ||||||
|  | @ -30,6 +32,7 @@ public: | ||||||
|    ~Impl() = default; |    ~Impl() = default; | ||||||
|    std::shared_ptr<manager::MarkerManager> markerManager_ { |    std::shared_ptr<manager::MarkerManager> markerManager_ { | ||||||
|       manager::MarkerManager::Instance()}; |       manager::MarkerManager::Instance()}; | ||||||
|  |    std::vector<types::MarkerId> markerIds_; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| MarkerModel::MarkerModel(QObject* parent) : | MarkerModel::MarkerModel(QObject* parent) : | ||||||
|  | @ -63,7 +66,7 @@ int MarkerModel::rowCount(const QModelIndex& parent) const | ||||||
| { | { | ||||||
|    return parent.isValid() ? |    return parent.isValid() ? | ||||||
|              0 : |              0 : | ||||||
|              static_cast<int>(p->markerManager_->marker_count()); |              static_cast<int>(p->markerIds_.size()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int MarkerModel::columnCount(const QModelIndex& parent) const | int MarkerModel::columnCount(const QModelIndex& parent) const | ||||||
|  | @ -95,15 +98,19 @@ QVariant MarkerModel::data(const QModelIndex& index, int role) const | ||||||
|    static const char COORDINATE_FORMAT    = 'g'; |    static const char COORDINATE_FORMAT    = 'g'; | ||||||
|    static const int  COORDINATE_PRECISION = 10; |    static const int  COORDINATE_PRECISION = 10; | ||||||
| 
 | 
 | ||||||
|    if (!index.isValid() || index.row() < 0) |    if (!index.isValid() || index.row() < 0 || | ||||||
|  |        static_cast<size_t>(index.row()) >= p->markerIds_.size()) | ||||||
|    { |    { | ||||||
|  |       logger_->debug("Failed to get data index {}", index.row()); | ||||||
|       return QVariant(); |       return QVariant(); | ||||||
|    } |    } | ||||||
| 
 | 
 | ||||||
|  |    types::MarkerId id = p->markerIds_[index.row()]; | ||||||
|    std::optional<types::MarkerInfo> markerInfo = |    std::optional<types::MarkerInfo> markerInfo = | ||||||
|       p->markerManager_->get_marker(index.row()); |       p->markerManager_->get_marker(id); | ||||||
|    if (!markerInfo) |    if (!markerInfo) | ||||||
|    { |    { | ||||||
|  |       logger_->debug("Failed to get data index {} id {}", index.row(), id); | ||||||
|       return QVariant(); |       return QVariant(); | ||||||
|    } |    } | ||||||
| 
 | 
 | ||||||
|  | @ -154,6 +161,16 @@ QVariant MarkerModel::data(const QModelIndex& index, int role) const | ||||||
|    return QVariant(); |    return QVariant(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | std::optional<types::MarkerId> MarkerModel::getId(int index) | ||||||
|  | { | ||||||
|  |    if (index < 0 || static_cast<size_t>(index) >= p->markerIds_.size()) | ||||||
|  |    { | ||||||
|  |       return {}; | ||||||
|  |    } | ||||||
|  | 
 | ||||||
|  |    return p->markerIds_[index]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| QVariant MarkerModel::headerData(int             section, | QVariant MarkerModel::headerData(int             section, | ||||||
|                                  Qt::Orientation orientation, |                                  Qt::Orientation orientation, | ||||||
|                                  int             role) const |                                  int             role) const | ||||||
|  | @ -186,12 +203,16 @@ bool MarkerModel::setData(const QModelIndex& index, | ||||||
|                           const QVariant&    value, |                           const QVariant&    value, | ||||||
|                           int                role) |                           int                role) | ||||||
| { | { | ||||||
|    if (!index.isValid() || index.row() < 0) | 
 | ||||||
|  |    if (!index.isValid() || index.row() < 0 || | ||||||
|  |        static_cast<size_t>(index.row()) >= p->markerIds_.size()) | ||||||
|    { |    { | ||||||
|       return false; |       return false; | ||||||
|    } |    } | ||||||
|  | 
 | ||||||
|  |    types::MarkerId id = p->markerIds_[index.row()]; | ||||||
|    std::optional<types::MarkerInfo> markerInfo = |    std::optional<types::MarkerInfo> markerInfo = | ||||||
|       p->markerManager_->get_marker(index.row()); |       p->markerManager_->get_marker(id); | ||||||
|    if (!markerInfo) |    if (!markerInfo) | ||||||
|    { |    { | ||||||
|       return false; |       return false; | ||||||
|  | @ -205,7 +226,7 @@ bool MarkerModel::setData(const QModelIndex& index, | ||||||
|       { |       { | ||||||
|          QString str = value.toString(); |          QString str = value.toString(); | ||||||
|          markerInfo->name = str.toStdString(); |          markerInfo->name = str.toStdString(); | ||||||
|          p->markerManager_->set_marker(index.row(), *markerInfo); |          p->markerManager_->set_marker(id, *markerInfo); | ||||||
|          result = true; |          result = true; | ||||||
|       } |       } | ||||||
|       break; |       break; | ||||||
|  | @ -219,7 +240,7 @@ bool MarkerModel::setData(const QModelIndex& index, | ||||||
|          if (!str.isEmpty() && ok && -90 <= latitude && latitude <= 90) |          if (!str.isEmpty() && ok && -90 <= latitude && latitude <= 90) | ||||||
|          { |          { | ||||||
|             markerInfo->latitude = latitude; |             markerInfo->latitude = latitude; | ||||||
|             p->markerManager_->set_marker(index.row(), *markerInfo); |             p->markerManager_->set_marker(id, *markerInfo); | ||||||
|             result = true; |             result = true; | ||||||
|          } |          } | ||||||
|       } |       } | ||||||
|  | @ -234,7 +255,7 @@ bool MarkerModel::setData(const QModelIndex& index, | ||||||
|          if (!str.isEmpty() && ok && -180 <= longitude && longitude <= 180) |          if (!str.isEmpty() && ok && -180 <= longitude && longitude <= 180) | ||||||
|          { |          { | ||||||
|             markerInfo->longitude = longitude; |             markerInfo->longitude = longitude; | ||||||
|             p->markerManager_->set_marker(index.row(), *markerInfo); |             p->markerManager_->set_marker(id, *markerInfo); | ||||||
|             result = true; |             result = true; | ||||||
|          } |          } | ||||||
|       } |       } | ||||||
|  | @ -260,32 +281,49 @@ void MarkerModel::HandleMarkersInitialized(size_t count) | ||||||
|    } |    } | ||||||
|    const int index = static_cast<int>(count - 1); |    const int index = static_cast<int>(count - 1); | ||||||
| 
 | 
 | ||||||
|  |    p->markerIds_.reserve(count); | ||||||
|    beginInsertRows(QModelIndex(), 0, index); |    beginInsertRows(QModelIndex(), 0, index); | ||||||
|  |    p->markerManager_->for_each( | ||||||
|  |       [this](const types::MarkerInfo& info) | ||||||
|  |       { | ||||||
|  |          p->markerIds_.push_back(info.id); | ||||||
|  |       }); | ||||||
|    endInsertRows(); |    endInsertRows(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MarkerModel::HandleMarkerAdded() | void MarkerModel::HandleMarkerAdded(types::MarkerId id) | ||||||
| { | { | ||||||
|    const int newIndex = static_cast<int>(p->markerManager_->marker_count() - 1); |    std::optional<size_t> index = p->markerManager_->get_index(id); | ||||||
|  |    const int newIndex = static_cast<int>(*index); | ||||||
| 
 | 
 | ||||||
|    beginInsertRows(QModelIndex(), newIndex, newIndex); |    beginInsertRows(QModelIndex(), newIndex, newIndex); | ||||||
|  |    p->markerIds_.emplace_back(id); | ||||||
|    endInsertRows(); |    endInsertRows(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MarkerModel::HandleMarkerChanged(size_t index) | void MarkerModel::HandleMarkerChanged(types::MarkerId id) | ||||||
| { | { | ||||||
|    const int changedIndex = static_cast<int>(index); |    std::optional<size_t> index = p->markerManager_->get_index(id); | ||||||
|  |    const int changedIndex = static_cast<int>(*index); | ||||||
|  | 
 | ||||||
|    QModelIndex topLeft = createIndex(changedIndex, kFirstColumn); |    QModelIndex topLeft = createIndex(changedIndex, kFirstColumn); | ||||||
|    QModelIndex bottomRight = createIndex(changedIndex, kLastColumn); |    QModelIndex bottomRight = createIndex(changedIndex, kLastColumn); | ||||||
| 
 | 
 | ||||||
|    Q_EMIT dataChanged(topLeft, bottomRight); |    Q_EMIT dataChanged(topLeft, bottomRight); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MarkerModel::HandleMarkerRemoved(size_t index) | void MarkerModel::HandleMarkerRemoved(types::MarkerId id) | ||||||
| { | { | ||||||
|    const int removedIndex = static_cast<int>(index); |    auto it = std::find(p->markerIds_.begin(), p->markerIds_.end(), id); | ||||||
|  |    if (it == p->markerIds_.end()) | ||||||
|  |    { | ||||||
|  |       return; | ||||||
|  |    } | ||||||
|  | 
 | ||||||
|  |    const int removedIndex = std::distance(p->markerIds_.begin(), it); | ||||||
| 
 | 
 | ||||||
|    beginRemoveRows(QModelIndex(), removedIndex, removedIndex); |    beginRemoveRows(QModelIndex(), removedIndex, removedIndex); | ||||||
|  |    p->markerIds_.erase(it); | ||||||
|    endRemoveRows(); |    endRemoveRows(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,6 +1,7 @@ | ||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
| #include <QAbstractTableModel> | #include <QAbstractTableModel> | ||||||
|  | #include <scwx/qt/types/marker_types.hpp> | ||||||
| 
 | 
 | ||||||
| namespace scwx | namespace scwx | ||||||
| { | { | ||||||
|  | @ -37,12 +38,13 @@ public: | ||||||
|                 const QVariant&    value, |                 const QVariant&    value, | ||||||
|                 int                role = Qt::EditRole) override; |                 int                role = Qt::EditRole) override; | ||||||
| 
 | 
 | ||||||
|  |    std::optional<types::MarkerId> getId(int index); | ||||||
| 
 | 
 | ||||||
| public slots: | public slots: | ||||||
|    void HandleMarkersInitialized(size_t count); |    void HandleMarkersInitialized(size_t count); | ||||||
|    void HandleMarkerAdded(); |    void HandleMarkerAdded(types::MarkerId id); | ||||||
|    void HandleMarkerChanged(size_t index); |    void HandleMarkerChanged(types::MarkerId id); | ||||||
|    void HandleMarkerRemoved(size_t index); |    void HandleMarkerRemoved(types::MarkerId id); | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|    class Impl; |    class Impl; | ||||||
|  |  | ||||||
|  | @ -1,6 +1,7 @@ | ||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
| #include <string> | #include <string> | ||||||
|  | #include <cstdint> | ||||||
| 
 | 
 | ||||||
| namespace scwx | namespace scwx | ||||||
| { | { | ||||||
|  | @ -8,6 +9,7 @@ namespace qt | ||||||
| { | { | ||||||
| namespace types | namespace types | ||||||
| { | { | ||||||
|  | typedef std::uint64_t MarkerId; | ||||||
| 
 | 
 | ||||||
| struct MarkerInfo | struct MarkerInfo | ||||||
| { | { | ||||||
|  | @ -16,6 +18,7 @@ struct MarkerInfo | ||||||
|    { |    { | ||||||
|    } |    } | ||||||
| 
 | 
 | ||||||
|  |    MarkerId    id; | ||||||
|    std::string name; |    std::string name; | ||||||
|    double      latitude; |    double      latitude; | ||||||
|    double      longitude; |    double      longitude; | ||||||
|  |  | ||||||
|  | @ -65,20 +65,24 @@ void MarkerSettingsWidgetImpl::ConnectSignals() | ||||||
|                     { |                     { | ||||||
|                        markerManager_->add_marker(types::MarkerInfo("", 0, 0)); |                        markerManager_->add_marker(types::MarkerInfo("", 0, 0)); | ||||||
|                     }); |                     }); | ||||||
|    QObject::connect(self_->ui->removeButton, |    QObject::connect( | ||||||
|  |       self_->ui->removeButton, | ||||||
|       &QPushButton::clicked, |       &QPushButton::clicked, | ||||||
|       self_, |       self_, | ||||||
|       [this]() |       [this]() | ||||||
|       { |       { | ||||||
|                        auto selectionModel = |          auto        selectionModel = self_->ui->markerView->selectionModel(); | ||||||
|                           self_->ui->markerView->selectionModel(); |          QModelIndex selected       = selectionModel | ||||||
|                        QModelIndex selected = |  | ||||||
|                           selectionModel |  | ||||||
|                                    ->selectedRows(static_cast<int>( |                                    ->selectedRows(static_cast<int>( | ||||||
|                                       model::MarkerModel::Column::Name)) |                                       model::MarkerModel::Column::Name)) | ||||||
|                                    .first(); |                                    .first(); | ||||||
|  |          std::optional<types::MarkerId> id = markerModel_->getId(selected.row()); | ||||||
|  |          if (!id) | ||||||
|  |          { | ||||||
|  |             return; | ||||||
|  |          } | ||||||
| 
 | 
 | ||||||
|                        markerManager_->remove_marker(selected.row()); |          markerManager_->remove_marker(*id); | ||||||
|       }); |       }); | ||||||
|    QObject::connect( |    QObject::connect( | ||||||
|       self_->ui->markerView->selectionModel(), |       self_->ui->markerView->selectionModel(), | ||||||
|  |  | ||||||
|  | @ -65,7 +65,7 @@ void RunTest(const std::string& filename, TestFunction testFunction) | ||||||
|       initialized = false; |       initialized = false; | ||||||
|       QObject::connect(manager.get(), |       QObject::connect(manager.get(), | ||||||
|                        &manager::MarkerManager::MarkersInitialized, |                        &manager::MarkerManager::MarkersInitialized, | ||||||
|                        []() |                        [](size_t count) | ||||||
|                        { |                        { | ||||||
|                           std::unique_lock lock(initializedMutex); |                           std::unique_lock lock(initializedMutex); | ||||||
|                           initialized = true; |                           initialized = true; | ||||||
|  | @ -79,6 +79,8 @@ void RunTest(const std::string& filename, TestFunction testFunction) | ||||||
|       { |       { | ||||||
|          initializedCond.wait(lock); |          initializedCond.wait(lock); | ||||||
|       } |       } | ||||||
|  |       // This is not jank
 | ||||||
|  |       model.HandleMarkersInitialized(manager->marker_count()); | ||||||
| 
 | 
 | ||||||
|       testFunction(manager, model); |       testFunction(manager, model); | ||||||
|    } |    } | ||||||
|  | @ -118,9 +120,17 @@ TEST(MarkerModelTest, AddRemove) | ||||||
|    RunTest(ONE_MARKERS_FILE, |    RunTest(ONE_MARKERS_FILE, | ||||||
|            [](std::shared_ptr<manager::MarkerManager> manager, MarkerModel&) |            [](std::shared_ptr<manager::MarkerManager> manager, MarkerModel&) | ||||||
|            { manager->add_marker(types::MarkerInfo("Null", 0, 0)); }); |            { manager->add_marker(types::MarkerInfo("Null", 0, 0)); }); | ||||||
|    RunTest(EMPTY_MARKERS_FILE, |    RunTest( | ||||||
|            [](std::shared_ptr<manager::MarkerManager> manager, MarkerModel&) |       EMPTY_MARKERS_FILE, | ||||||
|            { manager->remove_marker(0); }); |       [](std::shared_ptr<manager::MarkerManager> manager, MarkerModel& model) | ||||||
|  |       { | ||||||
|  |          std::optional<types::MarkerId> id = model.getId(0); | ||||||
|  |          EXPECT_TRUE(id); | ||||||
|  |          if (id) | ||||||
|  |          { | ||||||
|  |             manager->remove_marker(*id); | ||||||
|  |          } | ||||||
|  |       }); | ||||||
| 
 | 
 | ||||||
|    std::filesystem::remove(TEMP_MARKERS_FILE); |    std::filesystem::remove(TEMP_MARKERS_FILE); | ||||||
|    EXPECT_EQ(std::filesystem::exists(TEMP_MARKERS_FILE), false); |    EXPECT_EQ(std::filesystem::exists(TEMP_MARKERS_FILE), false); | ||||||
|  | @ -165,14 +175,30 @@ TEST(MarkerModelTest, RemoveFive) | ||||||
| { | { | ||||||
|    CopyFile(FIVE_MARKERS_FILE, TEMP_MARKERS_FILE); |    CopyFile(FIVE_MARKERS_FILE, TEMP_MARKERS_FILE); | ||||||
| 
 | 
 | ||||||
|    RunTest(EMPTY_MARKERS_FILE, |    RunTest( | ||||||
|            [](std::shared_ptr<manager::MarkerManager> manager, MarkerModel&) |       EMPTY_MARKERS_FILE, | ||||||
|  |       [](std::shared_ptr<manager::MarkerManager> manager, MarkerModel& model) | ||||||
|       { |       { | ||||||
|               manager->remove_marker(4); |          std::optional<types::MarkerId> id; | ||||||
|               manager->remove_marker(3); |          id = model.getId(4); | ||||||
|               manager->remove_marker(2); |          EXPECT_TRUE(id); | ||||||
|               manager->remove_marker(1); |          manager->remove_marker(*id); | ||||||
|               manager->remove_marker(0); | 
 | ||||||
|  |          id = model.getId(3); | ||||||
|  |          EXPECT_TRUE(id); | ||||||
|  |          manager->remove_marker(*id); | ||||||
|  | 
 | ||||||
|  |          id = model.getId(2); | ||||||
|  |          EXPECT_TRUE(id); | ||||||
|  |          manager->remove_marker(*id); | ||||||
|  | 
 | ||||||
|  |          id = model.getId(1); | ||||||
|  |          EXPECT_TRUE(id); | ||||||
|  |          manager->remove_marker(*id); | ||||||
|  | 
 | ||||||
|  |          id = model.getId(0); | ||||||
|  |          EXPECT_TRUE(id); | ||||||
|  |          manager->remove_marker(*id); | ||||||
|       }); |       }); | ||||||
| 
 | 
 | ||||||
|    std::filesystem::remove(TEMP_MARKERS_FILE); |    std::filesystem::remove(TEMP_MARKERS_FILE); | ||||||
|  | @ -183,13 +209,26 @@ TEST(MarkerModelTest, RemoveFour) | ||||||
| { | { | ||||||
|    CopyFile(FIVE_MARKERS_FILE, TEMP_MARKERS_FILE); |    CopyFile(FIVE_MARKERS_FILE, TEMP_MARKERS_FILE); | ||||||
| 
 | 
 | ||||||
|    RunTest(ONE_MARKERS_FILE, |    RunTest( | ||||||
|            [](std::shared_ptr<manager::MarkerManager> manager, MarkerModel&) |       ONE_MARKERS_FILE, | ||||||
|  |       [](std::shared_ptr<manager::MarkerManager> manager, MarkerModel& model) | ||||||
|       { |       { | ||||||
|               manager->remove_marker(4); |          std::optional<types::MarkerId> id; | ||||||
|               manager->remove_marker(3); |          id = model.getId(4); | ||||||
|               manager->remove_marker(2); |          EXPECT_TRUE(id); | ||||||
|               manager->remove_marker(1); |          manager->remove_marker(*id); | ||||||
|  | 
 | ||||||
|  |          id = model.getId(3); | ||||||
|  |          EXPECT_TRUE(id); | ||||||
|  |          manager->remove_marker(*id); | ||||||
|  | 
 | ||||||
|  |          id = model.getId(2); | ||||||
|  |          EXPECT_TRUE(id); | ||||||
|  |          manager->remove_marker(*id); | ||||||
|  | 
 | ||||||
|  |          id = model.getId(1); | ||||||
|  |          EXPECT_TRUE(id); | ||||||
|  |          manager->remove_marker(*id); | ||||||
|       }); |       }); | ||||||
| 
 | 
 | ||||||
|    std::filesystem::remove(TEMP_MARKERS_FILE); |    std::filesystem::remove(TEMP_MARKERS_FILE); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 AdenKoperczak
						AdenKoperczak