1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-16 15:29:55 +00:00
openmw/components/esm3/esmwriter.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

223 lines
5.2 KiB
C++
Raw Normal View History

#include "esmwriter.hpp"
#include <cassert>
#include <fstream>
2013-09-15 12:55:40 +00:00
#include <stdexcept>
#include <components/to_utf8/to_utf8.hpp>
namespace ESM
{
2014-09-26 15:12:48 +00:00
ESMWriter::ESMWriter()
: mRecords()
2018-10-09 06:21:12 +00:00
, mStream(nullptr)
, mHeaderPos()
2018-10-09 06:21:12 +00:00
, mEncoder(nullptr)
, mRecordCount(0)
, mCounting(true)
, mHeader()
2014-09-26 15:12:48 +00:00
{
}
2013-09-15 12:55:40 +00:00
unsigned int ESMWriter::getVersion() const
{
return mHeader.mData.version;
}
2012-04-06 20:25:33 +00:00
2013-09-15 12:55:40 +00:00
void ESMWriter::setVersion(unsigned int ver)
{
mHeader.mData.version = ver;
}
void ESMWriter::setType(int type)
{
mHeader.mData.type = type;
}
2013-09-15 12:55:40 +00:00
void ESMWriter::setAuthor(const std::string& auth)
{
mHeader.mData.author.assign(auth);
}
2013-09-15 12:55:40 +00:00
void ESMWriter::setDescription(const std::string& desc)
{
mHeader.mData.desc.assign(desc);
}
2013-09-15 12:55:40 +00:00
void ESMWriter::setRecordCount(int count)
{
mHeader.mData.records = count;
}
2012-04-06 20:25:33 +00:00
void ESMWriter::setFormatVersion(FormatVersion value)
2013-09-15 12:55:40 +00:00
{
mHeader.mFormatVersion = value;
2013-09-15 12:55:40 +00:00
}
void ESMWriter::clearMaster()
{
mHeader.mMaster.clear();
}
2013-09-15 12:55:40 +00:00
void ESMWriter::addMaster(const std::string& name, uint64_t size)
{
Header::MasterData d;
d.name = name;
d.size = size;
mHeader.mMaster.push_back(d);
}
2013-09-15 12:55:40 +00:00
void ESMWriter::save(std::ostream& file)
{
mRecordCount = 0;
mRecords.clear();
2013-09-15 13:30:17 +00:00
mCounting = true;
2013-09-15 12:55:40 +00:00
mStream = &file;
2012-04-06 20:25:33 +00:00
2013-09-15 12:55:40 +00:00
startRecord("TES3", 0);
2013-09-15 12:55:40 +00:00
mHeader.save(*this);
2012-04-06 20:25:33 +00:00
2013-09-15 12:55:40 +00:00
endRecord("TES3");
}
2013-09-15 12:55:40 +00:00
void ESMWriter::close()
{
if (!mRecords.empty())
throw std::runtime_error("Unclosed record remaining");
}
void ESMWriter::startRecord(NAME name, uint32_t flags)
2013-09-15 12:55:40 +00:00
{
mRecordCount++;
writeName(name);
RecordData rec;
rec.name = name;
rec.position = mStream->tellp();
rec.size = 0;
2014-02-22 01:15:20 +00:00
writeT<uint32_t>(0); // Size goes here
writeT<uint32_t>(0); // Unused header?
2013-09-15 12:55:40 +00:00
writeT(flags);
mRecords.push_back(rec);
assert(mRecords.back().size == 0);
}
void ESMWriter::startRecord(uint32_t name, uint32_t flags)
{
startRecord(NAME(name), flags);
}
void ESMWriter::startSubRecord(NAME name)
2013-09-15 12:55:40 +00:00
{
2014-05-12 19:04:02 +00:00
// Sub-record hierarchies are not properly supported in ESMReader. This should be fixed later.
assert(mRecords.size() <= 1);
2013-09-15 12:55:40 +00:00
writeName(name);
RecordData rec;
rec.name = name;
rec.position = mStream->tellp();
rec.size = 0;
2014-02-22 01:15:20 +00:00
writeT<uint32_t>(0); // Size goes here
2013-09-15 12:55:40 +00:00
mRecords.push_back(rec);
assert(mRecords.back().size == 0);
}
void ESMWriter::endRecord(NAME name)
2013-09-15 12:55:40 +00:00
{
RecordData rec = mRecords.back();
assert(rec.name == name);
mRecords.pop_back();
2013-09-15 12:55:40 +00:00
mStream->seekp(rec.position);
2013-09-15 12:55:40 +00:00
mCounting = false;
2014-02-22 01:15:20 +00:00
write(reinterpret_cast<const char*>(&rec.size), sizeof(uint32_t));
2013-09-15 12:55:40 +00:00
mCounting = true;
2013-09-15 12:55:40 +00:00
mStream->seekp(0, std::ios::end);
}
void ESMWriter::endRecord(uint32_t name)
{
endRecord(NAME(name));
}
void ESMWriter::writeHNString(NAME name, const std::string& data)
{
2013-09-15 12:55:40 +00:00
startSubRecord(name);
writeHString(data);
endRecord(name);
}
void ESMWriter::writeHNString(NAME name, const std::string& data, size_t size)
{
2013-09-15 12:55:40 +00:00
assert(data.size() <= size);
startSubRecord(name);
writeHString(data);
if (data.size() < size)
{
for (size_t i = data.size(); i < size; ++i)
write("\0", 1);
}
2013-09-15 12:55:40 +00:00
endRecord(name);
}
void ESMWriter::writeFixedSizeString(const std::string& data, std::size_t size)
{
std::string string;
if (!data.empty())
string = mEncoder ? mEncoder->getLegacyEnc(data) : data;
if (string.size() > size)
throw std::runtime_error("Fixed string data is too long: \"" + string + "\" ("
+ std::to_string(string.size()) + " > " + std::to_string(size) + ")");
string.resize(size);
write(string.c_str(), string.size());
}
2013-09-15 12:55:40 +00:00
void ESMWriter::writeHString(const std::string& data)
{
if (data.size() == 0)
write("\0", 1);
else
{
// Convert to UTF8 and return
const std::string_view string = mEncoder != nullptr ? mEncoder->getLegacyEnc(data) : data;
2013-09-15 12:55:40 +00:00
write(string.data(), string.size());
2013-09-15 12:55:40 +00:00
}
}
2013-09-15 12:55:40 +00:00
void ESMWriter::writeHCString(const std::string& data)
{
writeHString(data);
if (data.size() > 0 && data[data.size() - 1] != '\0')
write("\0", 1);
}
void ESMWriter::writeName(NAME name)
{
write(name.mData, NAME::sCapacity);
}
2013-09-15 12:55:40 +00:00
void ESMWriter::write(const char* data, size_t size)
{
if (mCounting && !mRecords.empty())
{
for (std::list<RecordData>::iterator it = mRecords.begin(); it != mRecords.end(); ++it)
2021-05-02 06:43:44 +00:00
it->size += static_cast<uint32_t>(size);
2013-09-15 12:55:40 +00:00
}
2013-09-15 12:55:40 +00:00
mStream->write(data, size);
}
2013-09-15 12:55:40 +00:00
void ESMWriter::setEncoder(ToUTF8::Utf8Encoder* encoder)
{
mEncoder = encoder;
}
}