123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240 |
- /*
- * Copyright (c) 2021, Arne Elster <arne@elster.li>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include "HexDocument.h"
- #include <LibCore/Stream.h>
- void HexDocument::set(size_t position, u8 value)
- {
- auto unchanged_value = get_unchanged(position);
- if (value == unchanged_value) {
- m_changes.remove(position);
- } else {
- m_changes.set(position, value);
- }
- }
- bool HexDocument::is_dirty() const
- {
- return m_changes.size() > 0;
- }
- HexDocumentMemory::HexDocumentMemory(ByteBuffer&& buffer)
- : m_buffer(move(buffer))
- {
- }
- HexDocument::Cell HexDocumentMemory::get(size_t position)
- {
- auto tracked_change = m_changes.get(position);
- if (tracked_change.has_value()) {
- return Cell { tracked_change.value(), true };
- } else {
- return Cell { m_buffer[position], false };
- }
- }
- u8 HexDocumentMemory::get_unchanged(size_t position)
- {
- return m_buffer[position];
- }
- size_t HexDocumentMemory::size() const
- {
- return m_buffer.size();
- }
- HexDocument::Type HexDocumentMemory::type() const
- {
- return Type::Memory;
- }
- void HexDocumentMemory::clear_changes()
- {
- m_changes.clear();
- }
- ErrorOr<void> HexDocumentMemory::write_to_file(Core::Stream::File& file)
- {
- TRY(file.seek(0, SeekMode::SetPosition));
- TRY(file.write(m_buffer));
- for (auto& change : m_changes) {
- TRY(file.seek(change.key, SeekMode::SetPosition));
- TRY(file.write({ &change.value, 1 }));
- }
- return {};
- }
- ErrorOr<NonnullOwnPtr<HexDocumentFile>> HexDocumentFile::create(NonnullOwnPtr<Core::Stream::File> file)
- {
- auto document = TRY(adopt_nonnull_own_or_enomem(new HexDocumentFile(move(file))));
- // FIXME: Remove this hackery
- document->set_file(move(document->m_file));
- return document;
- }
- HexDocumentFile::HexDocumentFile(NonnullOwnPtr<Core::Stream::File> file)
- : m_file(move(file))
- {
- }
- void HexDocumentFile::write_to_file()
- {
- for (auto& change : m_changes) {
- m_file->seek(change.key, SeekMode::SetPosition).release_value_but_fixme_should_propagate_errors();
- m_file->write({ &change.value, 1 }).release_value_but_fixme_should_propagate_errors();
- }
- clear_changes();
- // make sure the next get operation triggers a read
- m_buffer_file_pos = m_file_size + 1;
- }
- ErrorOr<void> HexDocumentFile::write_to_file(Core::Stream::File& file)
- {
- TRY(file.truncate(size()));
- TRY(file.seek(0, SeekMode::SetPosition));
- TRY(m_file->seek(0, SeekMode::SetPosition));
- while (true) {
- Array<u8, 64 * KiB> buffer;
- auto copy_buffer = TRY(m_file->read(buffer));
- if (copy_buffer.size() == 0)
- break;
- TRY(file.write(copy_buffer));
- }
- for (auto& change : m_changes) {
- TRY(file.seek(change.key, SeekMode::SetPosition));
- TRY(file.write({ &change.value, 1 }));
- }
- return {};
- }
- HexDocument::Cell HexDocumentFile::get(size_t position)
- {
- auto tracked_change = m_changes.get(position);
- if (tracked_change.has_value()) {
- return Cell { tracked_change.value(), true };
- }
- ensure_position_in_buffer(position);
- return { m_buffer[position - m_buffer_file_pos], false };
- }
- u8 HexDocumentFile::get_unchanged(size_t position)
- {
- ensure_position_in_buffer(position);
- return m_buffer[position - m_buffer_file_pos];
- }
- size_t HexDocumentFile::size() const
- {
- return m_file_size;
- }
- HexDocument::Type HexDocumentFile::type() const
- {
- return Type::File;
- }
- void HexDocumentFile::clear_changes()
- {
- m_changes.clear();
- }
- void HexDocumentFile::set_file(NonnullOwnPtr<Core::Stream::File> file)
- {
- m_file = move(file);
- if (auto result = m_file->seek(0, SeekMode::FromEndPosition); result.is_error())
- m_file_size = 0;
- else
- m_file_size = result.value();
- m_file->seek(0, SeekMode::SetPosition).release_value_but_fixme_should_propagate_errors();
- clear_changes();
- // make sure the next get operation triggers a read
- m_buffer_file_pos = m_file_size + 1;
- }
- NonnullOwnPtr<Core::Stream::File> const& HexDocumentFile::file() const
- {
- return m_file;
- }
- void HexDocumentFile::ensure_position_in_buffer(size_t position)
- {
- if (position < m_buffer_file_pos || position >= m_buffer_file_pos + m_buffer.size()) {
- m_file->seek(position, SeekMode::SetPosition).release_value_but_fixme_should_propagate_errors();
- m_file->read(m_buffer).release_value_but_fixme_should_propagate_errors();
- m_buffer_file_pos = position;
- }
- }
- HexDocumentUndoCommand::HexDocumentUndoCommand(WeakPtr<HexDocument> document, size_t position)
- : m_document(move(document))
- , m_position(position)
- {
- }
- void HexDocumentUndoCommand::undo()
- {
- for (size_t i = 0; i < m_old.size(); i++)
- m_document->set(m_position + i, m_old[i]);
- }
- void HexDocumentUndoCommand::redo()
- {
- for (size_t i = 0; i < m_new.size(); i++)
- m_document->set(m_position + i, m_new[i]);
- }
- bool HexDocumentUndoCommand::merge_with(GUI::Command const& other)
- {
- if (!is<HexDocumentUndoCommand>(other) || commit_time_expired())
- return false;
- auto const& typed_other = static_cast<HexDocumentUndoCommand const&>(other);
- size_t relative_start = typed_other.m_position - m_position;
- size_t other_length = typed_other.m_old.size();
- size_t length = m_old.size();
- if (typed_other.m_position < m_position || m_position + length < typed_other.m_position)
- return false;
- m_old.resize(relative_start + other_length);
- m_new.resize(relative_start + other_length);
- for (size_t i = 0; i < other_length; i++) {
- m_new[relative_start + i] = typed_other.m_new[i];
- if (relative_start + i >= length)
- m_old[relative_start + i] = typed_other.m_old[i];
- }
- m_timestamp = Time::now_monotonic();
- return true;
- }
- ErrorOr<void> HexDocumentUndoCommand::try_add_changed_byte(u8 old_value, u8 new_value)
- {
- TRY(m_old.try_append(old_value));
- TRY(m_new.try_append(new_value));
- return {};
- }
- ErrorOr<void> HexDocumentUndoCommand::try_add_changed_bytes(ByteBuffer old_values, ByteBuffer new_values)
- {
- TRY(m_old.try_append(move(old_values)));
- TRY(m_new.try_append(move(new_values)));
- return {};
- }
|