try to add a red black tree. I haven't done algorithms in a long time

This commit is contained in:
Thomas Fussell 2017-04-24 22:16:03 -04:00
parent be59d522a6
commit 562e163bf6

View File

@ -29,6 +29,7 @@
#include <vector>
#include <detail/binary.hpp>
#include <detail/unicode.hpp>
#include <detail/cryptography/compound_document.hpp>
#include <xlnt/utils/exceptions.hpp>
@ -78,11 +79,72 @@ bool header_is_valid(const compound_document_header &h)
const directory_id End = -1;
void recolor_entries(std::vector<compound_document_entry> &entries)
void visit(compound_document_entry *entry)
{
}
void traverse_storages(std::vector<compound_document_entry> &entries, compound_document_entry *storage);
void traverse_storage(std::vector<compound_document_entry> &entries, compound_document_entry *entry)
{
if (entry->prev >= 0)
{
traverse_storage(entries, &entries[entry->prev]);
}
visit(entry);
if (entry->type == compound_document_entry::entry_type::UserStorage
|| entry->type == compound_document_entry::entry_type::RootStorage)
{
traverse_storages(entries, entry);
}
if (entry->next >= 0)
{
traverse_storage(entries, &entries[entry->next]);
}
}
void traverse_storages(std::vector<compound_document_entry> &entries, compound_document_entry *storage)
{
visit(storage);
if (storage->child >= 0)
{
traverse_storage(entries, &entries[storage->child]);
}
}
void update_red_black_tree(std::vector<compound_document_entry> &entries)
{
using entry_type = compound_document_entry::entry_type;
if (entries.empty()) return;
traverse_storages(entries, &entries[0]);
return;
}
std::vector<std::u16string> split_path(const std::u16string &path_string)
{
auto sep = path_string.find(u'/');
auto prev = std::size_t(0);
auto split = std::vector<std::u16string>();
while (sep != std::u16string::npos)
{
split.push_back(path_string.substr(prev, sep - prev));
prev = sep;
sep = path_string.find(u'/');
}
return split;
}
} // namespace
namespace xlnt {
@ -360,10 +422,52 @@ compound_document_entry &compound_document::insert_entry(
const std::u16string &name,
compound_document_entry::entry_type type)
{
auto entry_id = directory_id(entries_.size());
entries_.push_back(compound_document_entry());
auto &entry = entries_.back();
entry.name(name);
entry.name(name);
entry.type = type;
if (entry_id != 0)
{
auto &root_entry = entries_.at(0);
if (root_entry.child == -1)
{
root_entry.child = entry_id;
}
else
{
auto parent = 0;
auto current = root_entry.child;
while (current >= 0)
{
if (name > entries_[current].name())
{
parent = current;
current = entries_[current].next;
}
else
{
parent = current;
current = entries_[current].prev;
}
}
if (name > entries_[parent].name())
{
entries_[parent].next = entry_id;
}
else
{
entries_[parent].prev = entry_id;
}
}
}
update_red_black_tree(entries_);
write_directory_tree();
return entry;
@ -378,8 +482,6 @@ void compound_document::write_directory_tree()
auto current_sector_id = header_.directory_start;
auto entry_index = directory_id(0);
recolor_entries(entries_);
for (auto &e : entries_)
{
writer_->offset(sector_data_start() + current_sector_id * sector_size());