Merge branch 'deserializedeez' into 'master'

Make RefId::deserializeText return an empty RefId if no pre-existing StringRefId can be found

See merge request OpenMW/openmw!3226
revert-6246b479
psi29a 1 year ago
commit 98bb9fc125

@ -41,13 +41,14 @@ namespace MWLua
} }
else else
{ {
ESM::RefId recordId = ESM::RefId::deserializeText(std::get<std::string>(item)); const auto& stringId = std::get<std::string>(item);
ESM::RefId recordId = ESM::RefId::deserializeText(stringId);
if (old_it != store.end() && old_it->getCellRef().getRefId() == recordId) if (old_it != store.end() && old_it->getCellRef().getRefId() == recordId)
return { old_it, true }; // already equipped return { old_it, true }; // already equipped
itemPtr = store.search(recordId); itemPtr = store.search(recordId);
if (itemPtr.isEmpty() || itemPtr.getRefData().getCount() == 0) if (itemPtr.isEmpty() || itemPtr.getRefData().getCount() == 0)
{ {
Log(Debug::Warning) << "There is no object with recordId='" << recordId << "' in inventory"; Log(Debug::Warning) << "There is no object with recordId='" << stringId << "' in inventory";
return { store.end(), false }; return { store.end(), false };
} }
} }

@ -124,6 +124,12 @@ namespace ESM
EXPECT_LT(a, b); EXPECT_LT(a, b);
} }
TEST(ESMRefIdTest, stringRefIdDeserializationReturnsEmptyRefIdForNonExistentValues)
{
RefId id = RefId::deserializeText("this stringrefid should not exist");
EXPECT_TRUE(id.empty());
}
TEST(ESMRefIdTest, lessThanIsDefinedForStringRefIdAndRefId) TEST(ESMRefIdTest, lessThanIsDefinedForStringRefIdAndRefId)
{ {
const StringRefId stringRefId("a"); const StringRefId stringRefId("a");

@ -254,6 +254,8 @@ namespace ESM
return ESM::ESM3ExteriorCellRefId(x, y); return ESM::ESM3ExteriorCellRefId(x, y);
} }
return ESM::RefId::stringRefId(value); if (auto id = ESM::StringRefId::deserializeExisting(value))
return *id;
return {};
} }
} }

@ -21,10 +21,15 @@ namespace ESM
const std::string emptyString; const std::string emptyString;
Misc::NotNullPtr<const std::string> getOrInsertString(std::string_view id) Misc::ScopeGuarded<StringsSet>& getRefIds()
{ {
static Misc::ScopeGuarded<StringsSet> refIds; static Misc::ScopeGuarded<StringsSet> refIds;
const auto locked = refIds.lock(); return refIds;
}
Misc::NotNullPtr<const std::string> getOrInsertString(std::string_view id)
{
const auto locked = getRefIds().lock();
auto it = locked->find(id); auto it = locked->find(id);
if (it == locked->end()) if (it == locked->end())
it = locked->emplace(id).first; it = locked->emplace(id).first;
@ -129,4 +134,15 @@ namespace ESM
{ {
return Misc::StringUtils::ciFind(*mValue, subString) != std::string_view::npos; return Misc::StringUtils::ciFind(*mValue, subString) != std::string_view::npos;
} }
std::optional<StringRefId> StringRefId::deserializeExisting(std::string_view value)
{
const auto locked = getRefIds().lock();
auto it = locked->find(value);
if (it == locked->end())
return {};
StringRefId id;
id.mValue = &*it;
return id;
}
} }

@ -3,6 +3,7 @@
#include <functional> #include <functional>
#include <iosfwd> #include <iosfwd>
#include <optional>
#include <string> #include <string>
#include <string_view> #include <string_view>
#include <variant> #include <variant>
@ -46,6 +47,9 @@ namespace ESM
friend struct std::hash<StringRefId>; friend struct std::hash<StringRefId>;
// Similar to the constructor but only returns preexisting ids
static std::optional<StringRefId> deserializeExisting(std::string_view value);
private: private:
Misc::NotNullPtr<const std::string> mValue; Misc::NotNullPtr<const std::string> mValue;
}; };

Loading…
Cancel
Save