mirror of
https://github.com/subsurface/subsurface.git
synced 2025-02-19 22:16:15 +00:00
core: replace divesite_table_t by a vector of std::unique_ptr<>s
This is a long commit, because it introduces a new abstraction: a general std::vector<> of std::unique_ptrs<>. Moreover, it replaces a number of pointers by C++ references, when the callee does not suppoert null objects. This simplifies memory management and makes ownership more explicit. It is a proof-of-concept and a test-bed for the other core data structrures. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
This commit is contained in:
parent
411188728d
commit
e39dea3d68
41 changed files with 451 additions and 426 deletions
|
@ -134,9 +134,9 @@ void addDiveSite(const QString &name)
|
|||
execute(new AddDiveSite(name));
|
||||
}
|
||||
|
||||
void importDiveSites(struct dive_site_table *sites, const QString &source)
|
||||
void importDiveSites(dive_site_table sites, const QString &source)
|
||||
{
|
||||
execute(new ImportDiveSites(sites, source));
|
||||
execute(new ImportDiveSites(std::move(sites), source));
|
||||
}
|
||||
|
||||
void mergeDiveSites(dive_site *ds, const QVector<dive_site *> &sites)
|
||||
|
|
|
@ -68,7 +68,7 @@ void editDiveSiteCountry(dive_site *ds, const QString &value);
|
|||
void editDiveSiteLocation(dive_site *ds, location_t value);
|
||||
void editDiveSiteTaxonomy(dive_site *ds, taxonomy_data &value); // value is consumed (i.e. will be erased after call)!
|
||||
void addDiveSite(const QString &name);
|
||||
void importDiveSites(struct dive_site_table *sites, const QString &source);
|
||||
void importDiveSites(dive_site_table sites, const QString &source); // takes ownership of dive site table
|
||||
void mergeDiveSites(dive_site *ds, const QVector<dive_site *> &sites);
|
||||
void purgeUnusedDiveSites();
|
||||
|
||||
|
|
|
@ -142,9 +142,9 @@ DivesAndTripsToAdd DiveListBase::removeDives(DivesAndSitesToRemove &divesAndSite
|
|||
divesAndSitesToDelete.dives.clear();
|
||||
|
||||
for (dive_site *ds: divesAndSitesToDelete.sites) {
|
||||
int idx = unregister_dive_site(ds);
|
||||
sitesToAdd.emplace_back(ds);
|
||||
emit diveListNotifier.diveSiteDeleted(ds, idx);
|
||||
auto res = divelog.sites->pull(ds);
|
||||
sitesToAdd.push_back(std::move(res.ptr));
|
||||
emit diveListNotifier.diveSiteDeleted(ds, res.idx);
|
||||
}
|
||||
divesAndSitesToDelete.sites.clear();
|
||||
|
||||
|
@ -217,9 +217,9 @@ DivesAndSitesToRemove DiveListBase::addDives(DivesAndTripsToAdd &toAdd)
|
|||
|
||||
// Finally, add any necessary dive sites
|
||||
for (std::unique_ptr<dive_site> &ds: toAdd.sites) {
|
||||
sites.push_back(ds.get());
|
||||
int idx = register_dive_site(ds.release()); // Return ownership to backend
|
||||
emit diveListNotifier.diveSiteAdded(sites.back(), idx);
|
||||
auto res = divelog.sites->register_site(std::move(ds));
|
||||
sites.push_back(res.ptr);
|
||||
emit diveListNotifier.diveSiteAdded(sites.back(), res.idx);
|
||||
}
|
||||
toAdd.sites.clear();
|
||||
|
||||
|
@ -478,10 +478,10 @@ ImportDives::ImportDives(struct divelog *log, int flags, const QString &source)
|
|||
struct dive_table dives_to_add = empty_dive_table;
|
||||
struct dive_table dives_to_remove = empty_dive_table;
|
||||
struct trip_table trips_to_add = empty_trip_table;
|
||||
struct dive_site_table sites_to_add = empty_dive_site_table;
|
||||
dive_site_table sites_to_add;
|
||||
process_imported_dives(log, flags,
|
||||
&dives_to_add, &dives_to_remove, &trips_to_add,
|
||||
&sites_to_add, &devicesToAddAndRemove);
|
||||
sites_to_add, &devicesToAddAndRemove);
|
||||
|
||||
// Add trips to the divesToAdd.trips structure
|
||||
divesToAdd.trips.reserve(trips_to_add.nr);
|
||||
|
@ -489,9 +489,7 @@ ImportDives::ImportDives(struct divelog *log, int flags, const QString &source)
|
|||
divesToAdd.trips.emplace_back(trips_to_add.trips[i]);
|
||||
|
||||
// Add sites to the divesToAdd.sites structure
|
||||
divesToAdd.sites.reserve(sites_to_add.nr);
|
||||
for (int i = 0; i < sites_to_add.nr; ++i)
|
||||
divesToAdd.sites.emplace_back(sites_to_add.dive_sites[i]);
|
||||
divesToAdd.sites = std::move(sites_to_add);
|
||||
|
||||
// Add dives to the divesToAdd.dives structure
|
||||
divesToAdd.dives.reserve(dives_to_add.nr);
|
||||
|
@ -525,7 +523,6 @@ ImportDives::ImportDives(struct divelog *log, int flags, const QString &source)
|
|||
free(dives_to_add.dives);
|
||||
free(dives_to_remove.dives);
|
||||
free(trips_to_add.trips);
|
||||
free(sites_to_add.dive_sites);
|
||||
}
|
||||
|
||||
bool ImportDives::workToBeDone()
|
||||
|
|
|
@ -30,9 +30,9 @@ static std::vector<dive_site *> addDiveSites(std::vector<std::unique_ptr<dive_si
|
|||
}
|
||||
|
||||
// Add dive site to core, but remember a non-owning pointer first.
|
||||
res.push_back(ds.get());
|
||||
int idx = register_dive_site(ds.release()); // Return ownership to backend.
|
||||
emit diveListNotifier.diveSiteAdded(res.back(), idx); // Inform frontend of new dive site.
|
||||
auto add_res = divelog.sites->put(std::move(ds)); // Return ownership to backend.
|
||||
res.push_back(add_res.ptr);
|
||||
emit diveListNotifier.diveSiteAdded(res.back(), add_res.idx); // Inform frontend of new dive site.
|
||||
}
|
||||
|
||||
emit diveListNotifier.divesChanged(changedDives, DiveField::DIVESITE);
|
||||
|
@ -60,9 +60,9 @@ static std::vector<std::unique_ptr<dive_site>> removeDiveSites(std::vector<dive_
|
|||
}
|
||||
|
||||
// Remove dive site from core and take ownership.
|
||||
int idx = unregister_dive_site(ds);
|
||||
res.emplace_back(ds);
|
||||
emit diveListNotifier.diveSiteDeleted(ds, idx); // Inform frontend of removed dive site.
|
||||
auto pull_res = divelog.sites->pull(ds);
|
||||
res.push_back(std::move(pull_res.ptr));
|
||||
emit diveListNotifier.diveSiteDeleted(ds, pull_res.idx); // Inform frontend of removed dive site.
|
||||
}
|
||||
|
||||
emit diveListNotifier.divesChanged(changedDives, DiveField::DIVESITE);
|
||||
|
@ -94,25 +94,18 @@ void AddDiveSite::undo()
|
|||
sitesToAdd = removeDiveSites(sitesToRemove);
|
||||
}
|
||||
|
||||
ImportDiveSites::ImportDiveSites(struct dive_site_table *sites, const QString &source)
|
||||
ImportDiveSites::ImportDiveSites(dive_site_table sites, const QString &source)
|
||||
{
|
||||
setText(Command::Base::tr("import dive sites from %1").arg(source));
|
||||
|
||||
for (int i = 0; i < sites->nr; ++i) {
|
||||
struct dive_site *new_ds = sites->dive_sites[i];
|
||||
|
||||
for (auto &new_ds: sites) {
|
||||
// Don't import dive sites that already exist. Currently we only check for
|
||||
// the same name. We might want to be smarter here and merge dive site data, etc.
|
||||
struct dive_site *old_ds = get_same_dive_site(new_ds);
|
||||
if (old_ds) {
|
||||
delete new_ds;
|
||||
struct dive_site *old_ds = get_same_dive_site(*new_ds);
|
||||
if (old_ds)
|
||||
continue;
|
||||
}
|
||||
sitesToAdd.emplace_back(new_ds);
|
||||
sitesToAdd.push_back(std::move(new_ds));
|
||||
}
|
||||
|
||||
// All site have been consumed
|
||||
sites->nr = 0;
|
||||
}
|
||||
|
||||
bool ImportDiveSites::workToBeDone()
|
||||
|
@ -153,10 +146,9 @@ void DeleteDiveSites::undo()
|
|||
PurgeUnusedDiveSites::PurgeUnusedDiveSites()
|
||||
{
|
||||
setText(Command::Base::tr("purge unused dive sites"));
|
||||
for (int i = 0; i < divelog.sites->nr; ++i) {
|
||||
dive_site *ds = divelog.sites->dive_sites[i];
|
||||
for (const auto &ds: *divelog.sites) {
|
||||
if (ds->dives.empty())
|
||||
sitesToRemove.push_back(ds);
|
||||
sitesToRemove.push_back(ds.get());
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -391,7 +383,7 @@ ApplyGPSFixes::ApplyGPSFixes(const std::vector<DiveAndLocation> &fixes)
|
|||
siteLocations.push_back({ ds, dl.location });
|
||||
}
|
||||
} else {
|
||||
ds = create_dive_site(dl.name.toStdString(), divelog.sites);
|
||||
ds = create_dive_site(dl.name.toStdString(), *divelog.sites);
|
||||
ds->location = dl.location;
|
||||
add_dive_to_dive_site(dl.d, ds);
|
||||
dl.d->dive_site = nullptr; // This will be set on redo()
|
||||
|
|
|
@ -36,8 +36,8 @@ private:
|
|||
|
||||
class ImportDiveSites : public Base {
|
||||
public:
|
||||
// Note: the dive site table is consumed after the call it will be empty.
|
||||
ImportDiveSites(struct dive_site_table *sites, const QString &source);
|
||||
// Note: Takes ownership of dive site table
|
||||
ImportDiveSites(dive_site_table sites, const QString &source);
|
||||
private:
|
||||
bool workToBeDone() override;
|
||||
void undo() override;
|
||||
|
|
|
@ -401,17 +401,17 @@ EditDiveSiteNew::EditDiveSiteNew(const QString &newName, bool currentDiveOnly) :
|
|||
void EditDiveSiteNew::undo()
|
||||
{
|
||||
EditDiveSite::undo();
|
||||
int idx = unregister_dive_site(diveSiteToRemove);
|
||||
diveSiteToAdd.reset(diveSiteToRemove);
|
||||
emit diveListNotifier.diveSiteDeleted(diveSiteToRemove, idx); // Inform frontend of removed dive site.
|
||||
auto res = divelog.sites->pull(diveSiteToRemove);
|
||||
diveSiteToAdd = std::move(res.ptr);
|
||||
emit diveListNotifier.diveSiteDeleted(diveSiteToRemove, res.idx); // Inform frontend of removed dive site.
|
||||
diveSiteToRemove = nullptr;
|
||||
}
|
||||
|
||||
void EditDiveSiteNew::redo()
|
||||
{
|
||||
diveSiteToRemove = diveSiteToAdd.get();
|
||||
int idx = register_dive_site(diveSiteToAdd.release()); // Return ownership to backend.
|
||||
emit diveListNotifier.diveSiteAdded(diveSiteToRemove, idx); // Inform frontend of new dive site.
|
||||
auto res = divelog.sites->register_site(std::move(diveSiteToAdd)); // Return ownership to backend.
|
||||
diveSiteToRemove = res.ptr;
|
||||
emit diveListNotifier.diveSiteAdded(diveSiteToRemove, res.idx); // Inform frontend of new dive site.
|
||||
EditDiveSite::redo();
|
||||
}
|
||||
|
||||
|
@ -1439,9 +1439,9 @@ EditDive::EditDive(dive *oldDiveIn, dive *newDiveIn, dive_site *createDs, dive_s
|
|||
void EditDive::undo()
|
||||
{
|
||||
if (siteToRemove) {
|
||||
int idx = unregister_dive_site(siteToRemove);
|
||||
siteToAdd.reset(siteToRemove);
|
||||
emit diveListNotifier.diveSiteDeleted(siteToRemove, idx); // Inform frontend of removed dive site.
|
||||
auto res = divelog.sites->pull(siteToRemove);
|
||||
siteToAdd = std::move(res.ptr);
|
||||
emit diveListNotifier.diveSiteDeleted(siteToRemove, res.idx); // Inform frontend of removed dive site.
|
||||
}
|
||||
|
||||
exchangeDives();
|
||||
|
@ -1451,9 +1451,9 @@ void EditDive::undo()
|
|||
void EditDive::redo()
|
||||
{
|
||||
if (siteToAdd) {
|
||||
siteToRemove = siteToAdd.get();
|
||||
int idx = register_dive_site(siteToAdd.release()); // Return ownership to backend.
|
||||
emit diveListNotifier.diveSiteAdded(siteToRemove, idx); // Inform frontend of new dive site.
|
||||
auto res = divelog.sites->register_site(std::move(siteToAdd)); // Return ownership to backend.
|
||||
siteToRemove = res.ptr;
|
||||
emit diveListNotifier.diveSiteAdded(siteToRemove, res.idx); // Inform frontend of new dive site.
|
||||
}
|
||||
|
||||
exchangeDives();
|
||||
|
|
|
@ -217,9 +217,9 @@ void AddPictures::undo()
|
|||
|
||||
// Remove dive sites
|
||||
for (dive_site *siteToRemove: sitesToRemove) {
|
||||
int idx = unregister_dive_site(siteToRemove);
|
||||
sitesToAdd.emplace_back(siteToRemove);
|
||||
emit diveListNotifier.diveSiteDeleted(siteToRemove, idx); // Inform frontend of removed dive site.
|
||||
auto res = divelog.sites->pull(siteToRemove);
|
||||
sitesToAdd.push_back(std::move(res.ptr));
|
||||
emit diveListNotifier.diveSiteDeleted(siteToRemove, res.idx); // Inform frontend of removed dive site.
|
||||
}
|
||||
sitesToRemove.clear();
|
||||
}
|
||||
|
@ -228,9 +228,9 @@ void AddPictures::redo()
|
|||
{
|
||||
// Add dive sites
|
||||
for (std::unique_ptr<dive_site> &siteToAdd: sitesToAdd) {
|
||||
sitesToRemove.push_back(siteToAdd.get());
|
||||
int idx = register_dive_site(siteToAdd.release()); // Return ownership to backend.
|
||||
emit diveListNotifier.diveSiteAdded(sitesToRemove.back(), idx); // Inform frontend of new dive site.
|
||||
auto res = divelog.sites->register_site(std::move(siteToAdd)); // Return ownership to backend.
|
||||
sitesToRemove.push_back(res.ptr);
|
||||
emit diveListNotifier.diveSiteAdded(sitesToRemove.back(), res.idx); // Inform frontend of new dive site.
|
||||
}
|
||||
sitesToAdd.clear();
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue