SIEdit/lib/object.cpp
2022-07-11 14:20:56 -07:00

241 lines
5.8 KiB
C++

#include "object.h"
#include <iostream>
namespace si {
Object::Object()
{
type_ = MxOb::Null;
id_ = 0;
}
bool Object::Parse(Chunk *chunk)
{
type_ = static_cast<MxOb::Type>(chunk->data("Type").toU16());
presenter_ = chunk->data("Presenter").toString();
unknown1_ = chunk->data("Unknown1");
name_ = chunk->data("Name").toString();
id_ = chunk->data("ID");
flags_ = chunk->data("Flags");
unknown4_ = chunk->data("Unknown4");
duration_ = chunk->data("Duration");
loops_ = chunk->data("Loops");
position_ = chunk->data("Position");
direction_ = chunk->data("Direction");
up_ = chunk->data("Up");
extra_ = chunk->data("ExtraData");
filename_ = chunk->data("FileName").toString();
unknown26_ = chunk->data("Unknown26");
unknown27_ = chunk->data("Unknown27");
unknown28_ = chunk->data("Unknown28");
filetype_ = static_cast<MxOb::FileType>(chunk->data("FileType").toU32());
unknown29_ = chunk->data("Unknown29");
unknown30_ = chunk->data("Unknown30");
unknown31_ = chunk->data("Unknown31");
if (chunk->HasChildren()) {
Chunk *child = static_cast<Chunk*>(chunk->GetChildAt(0));
if (child->id() == Chunk::TYPE_LIST) {
for (Core *entry : child->GetChildren()) {
Object *o = new Object();
if (!o->Parse(static_cast<Chunk*>(entry))) {
return false;
}
AppendChild(o);
}
}
}
return true;
}
Chunk *Object::Export() const
{
Chunk *chunk = new Chunk(Chunk::TYPE_MxOb);
chunk->data("Type") = type_;
chunk->data("Presenter") = presenter_;
chunk->data("Unknown1") = unknown1_;
chunk->data("Name") = name_;
chunk->data("ID") = id_;
chunk->data("Flags") = flags_;
chunk->data("Unknown4") = unknown4_;
chunk->data("Duration") = duration_;
chunk->data("Loops") = loops_;
chunk->data("Position") = position_;
chunk->data("Direction") = direction_;
chunk->data("Up") = up_;
chunk->data("ExtraData") = extra_;
chunk->data("FileName") = filename_;
chunk->data("Unknown26") = unknown26_;
chunk->data("Unknown27") = unknown27_;
chunk->data("Unknown28") = unknown28_;
chunk->data("FileType") = filetype_;
chunk->data("Unknown29") = unknown29_;
chunk->data("Unknown30") = unknown30_;
chunk->data("Unknown31") = unknown31_;
if (HasChildren()) {
Chunk *list = new Chunk(Chunk::TYPE_LIST);
list->data("Format") = Chunk::TYPE_MxCh;
chunk->AppendChild(list);
for (Children::const_iterator it=GetChildren().begin(); it!=GetChildren().end(); it++) {
Object *child = static_cast<Object*>(*it);
list->AppendChild(child->Export());
}
}
return chunk;
}
bytearray Object::GetNormalizedData() const
{
return ToPackedData(filetype(), data_);
}
void Object::SetNormalizedData(const bytearray &d)
{
SetChunkedData(ToChunkedData(filetype(), d));
}
bytearray Object::ToPackedData(MxOb::FileType filetype, const ChunkedData &chunks)
{
bytearray data;
switch (filetype) {
case MxOb::WAV:
{
// Make space for WAVE header
data.resize(0x2C);
// Merge all chunks after the first one
for (size_t i=1; i<chunks.size(); i++) {
data.append(chunks[i]);
}
// Copy boilerplate bytes for header
uint32_t *header = reinterpret_cast<uint32_t *>(data.data());
header[0] = Chunk::TYPE_RIFF; // "RIFF"
header[1] = data.size() - 8; // Size of total file
header[2] = 0x45564157; // "WAVE"
header[3] = 0x20746D66; // "fmt "
header[4] = 16; // Size of fmt chunk
header[9] = 0x61746164; // "data"
header[10] = data.size() - 0x2C; // Size of data chunk
// Copy fmt header from chunk 1
memcpy(&header[5], chunks[0].data(), 16);
break;
}
case MxOb::STL:
{
// Make space for BMP header
data.resize(14);
// Merge all chunks after the first one
for (size_t i=0; i<chunks.size(); i++) {
data.append(chunks[i]);
}
// Set BM identifier
*(uint16_t *)(data.data()) = 0x4D42;
// Set file size
*(uint32_t*)(data.data()+2) = data.size();
// Set reserved bytes
*(uint32_t*)(data.data()+6) = 0;
// Set offset
*(uint32_t*)(data.data()+10) = chunks.at(0).size() + 14;
break;
}
case MxOb::FLC:
{
// First chunk is a complete FLIC header, so add it as-is
data.append(chunks[0]);
// Subsequent chunks are FLIC frames with an additional 20 byte header that needs to be stripped
const int CUSTOM_HEADER_SZ = 20;
for (size_t i=1; i<chunks.size(); i++) {
data.append(chunks.at(i).data() + CUSTOM_HEADER_SZ, chunks.at(i).size() - CUSTOM_HEADER_SZ);
}
break;
}
default:
std::cout << "Didn't know how to extract type '" << std::string((const char *)&filetype, sizeof(filetype)) << "', merging..." << std::endl;
for (size_t i=0; i<chunks.size(); i++) {
data.append(chunks[i]);
}
break;
}
return data;
}
Object::ChunkedData Object::ToChunkedData(MxOb::FileType filetype, const bytearray &chunks)
{
// FIXME: STUB
return ChunkedData();
}
bytearray Object::GetFileHeader() const
{
switch (filetype()) {
case MxOb::WAV:
case MxOb::STL:
return data_.at(0);
}
return bytearray();
}
bytearray Object::GetFileBody() const
{
bytearray b;
switch (filetype()) {
case MxOb::WAV:
case MxOb::STL:
for (size_t i=1; i<data_.size(); i++) {
b.append(data_.at(i));
}
break;
}
return b;
}
size_t Object::GetFileBodySize() const
{
size_t s = 0;
switch (filetype()) {
case MxOb::WAV:
case MxOb::STL:
for (size_t i=1; i<data_.size(); i++) {
s += data_.at(i).size();
}
break;
}
return s;
}
Object *Object::FindSubObjectWithID(uint32_t id)
{
if (this->id() == id) {
return this;
}
for (Core *child : GetChildren()) {
if (Object *o = static_cast<Object*>(child)->FindSubObjectWithID(id)) {
return o;
}
}
return NULL;
}
}