2021-12-07 22:47:41 +01:00
|
|
|
#include "content/providers/file_provider.hpp"
|
2023-08-09 20:04:12 +02:00
|
|
|
#include "content/providers/memory_file_provider.hpp"
|
2020-11-11 09:18:35 +01:00
|
|
|
|
2024-05-19 15:10:22 +02:00
|
|
|
#include <hex/api/content_registry.hpp>
|
2023-11-30 11:23:12 +01:00
|
|
|
#include <hex/api/imhex_api.hpp>
|
2023-11-21 14:38:01 +01:00
|
|
|
#include <hex/api/localization_manager.hpp>
|
2023-01-11 23:31:25 +01:00
|
|
|
#include <hex/api/project_file_manager.hpp>
|
2023-11-18 14:50:43 +01:00
|
|
|
#include <hex/api/task_manager.hpp>
|
2023-08-09 20:04:12 +02:00
|
|
|
|
2024-05-19 15:10:22 +02:00
|
|
|
#include <popups/popup_question.hpp>
|
2024-02-27 20:20:14 +01:00
|
|
|
#include <toasts/toast_notification.hpp>
|
|
|
|
|
2021-08-29 22:15:18 +02:00
|
|
|
#include <hex/helpers/utils.hpp>
|
2022-02-01 18:09:40 +01:00
|
|
|
#include <hex/helpers/fmt.hpp>
|
2023-11-30 14:40:07 +01:00
|
|
|
#include <fmt/chrono.h>
|
2022-08-08 21:23:52 +02:00
|
|
|
|
2023-03-12 18:43:05 +01:00
|
|
|
#include <wolv/utils/string.hpp>
|
2024-05-19 15:10:22 +02:00
|
|
|
#include <wolv/literals.hpp>
|
2023-03-12 18:43:05 +01:00
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
#include <nlohmann/json.hpp>
|
2024-02-27 20:20:14 +01:00
|
|
|
#include <cstring>
|
2020-11-15 16:06:10 +01:00
|
|
|
|
2023-08-16 23:18:16 +02:00
|
|
|
#if defined(OS_WINDOWS)
|
|
|
|
#include <windows.h>
|
|
|
|
#endif
|
|
|
|
|
2022-11-07 00:04:47 +01:00
|
|
|
namespace hex::plugin::builtin {
|
2021-12-07 22:47:41 +01:00
|
|
|
|
2024-05-19 15:10:22 +02:00
|
|
|
using namespace wolv::literals;
|
|
|
|
|
2024-02-24 12:06:06 +01:00
|
|
|
std::set<FileProvider*> FileProvider::s_openedFiles;
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool FileProvider::isAvailable() const {
|
2023-03-17 11:31:50 +01:00
|
|
|
return true;
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool FileProvider::isReadable() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return isAvailable() && m_readable;
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool FileProvider::isWritable() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return isAvailable() && m_writable;
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool FileProvider::isResizable() const {
|
2022-09-03 23:08:40 +02:00
|
|
|
return isAvailable() && isWritable();
|
2021-07-27 21:07:36 +02:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
bool FileProvider::isSavable() const {
|
2024-03-10 14:31:39 +01:00
|
|
|
return m_loadedIntoMemory;
|
2020-11-27 09:09:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void FileProvider::readRaw(u64 offset, void *buffer, size_t size) {
|
2023-12-26 23:42:22 +01:00
|
|
|
if (m_fileSize == 0 || (offset + size) > m_fileSize || buffer == nullptr || size == 0)
|
2020-11-27 09:09:48 +01:00
|
|
|
return;
|
|
|
|
|
2024-03-10 14:31:39 +01:00
|
|
|
if (m_loadedIntoMemory)
|
|
|
|
std::memcpy(buffer, m_data.data() + offset, size);
|
|
|
|
else
|
|
|
|
m_file.readBufferAtomic(offset, static_cast<u8*>(buffer), size);
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2020-11-27 13:44:52 +01:00
|
|
|
void FileProvider::writeRaw(u64 offset, const void *buffer, size_t size) {
|
2021-12-09 21:10:24 +01:00
|
|
|
if ((offset + size) > this->getActualSize() || buffer == nullptr || size == 0)
|
2020-11-27 09:09:48 +01:00
|
|
|
return;
|
|
|
|
|
2024-03-10 14:31:39 +01:00
|
|
|
if (m_loadedIntoMemory)
|
|
|
|
std::memcpy(m_data.data() + offset, buffer, size);
|
|
|
|
else
|
|
|
|
m_file.writeBufferAtomic(offset, static_cast<const u8*>(buffer), size);
|
2020-11-27 09:09:48 +01:00
|
|
|
}
|
2021-07-27 21:07:36 +02:00
|
|
|
|
2021-08-21 13:53:50 +02:00
|
|
|
void FileProvider::save() {
|
2024-03-10 14:31:39 +01:00
|
|
|
if (m_loadedIntoMemory) {
|
2024-03-15 17:57:06 +01:00
|
|
|
m_ignoreNextChangeEvent = true;
|
2024-03-10 14:31:39 +01:00
|
|
|
m_file.open();
|
|
|
|
m_file.writeVectorAtomic(0x00, m_data);
|
|
|
|
m_file.setSize(m_data.size());
|
|
|
|
} else {
|
|
|
|
m_file.flush();
|
|
|
|
}
|
2024-02-15 21:54:41 +01:00
|
|
|
|
2023-08-16 23:18:16 +02:00
|
|
|
#if defined(OS_WINDOWS)
|
|
|
|
FILETIME ft;
|
|
|
|
SYSTEMTIME st;
|
|
|
|
|
2024-02-15 21:54:41 +01:00
|
|
|
if (m_file.isValid()) {
|
2023-08-16 23:18:16 +02:00
|
|
|
GetSystemTime(&st);
|
|
|
|
if (SystemTimeToFileTime(&st, &ft)) {
|
2024-02-15 21:54:41 +01:00
|
|
|
auto fileHandle = HANDLE(_get_osfhandle(_fileno(m_file.getHandle())));
|
2023-11-10 20:47:08 +01:00
|
|
|
SetFileTime(fileHandle, nullptr, nullptr, &ft);
|
2023-08-16 23:18:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2024-03-10 14:31:39 +01:00
|
|
|
if (m_loadedIntoMemory)
|
|
|
|
m_file.close();
|
|
|
|
|
2023-04-06 17:36:28 +02:00
|
|
|
Provider::save();
|
2021-08-21 13:53:50 +02:00
|
|
|
}
|
|
|
|
|
2023-03-21 15:42:10 +01:00
|
|
|
void FileProvider::saveAs(const std::fs::path &path) {
|
2023-12-19 13:10:25 +01:00
|
|
|
if (path == m_path)
|
2023-03-21 15:42:10 +01:00
|
|
|
this->save();
|
|
|
|
else
|
|
|
|
Provider::saveAs(path);
|
|
|
|
}
|
|
|
|
|
2023-12-08 16:22:36 +01:00
|
|
|
void FileProvider::resizeRaw(u64 newSize) {
|
2024-03-10 14:31:39 +01:00
|
|
|
if (m_loadedIntoMemory)
|
|
|
|
m_data.resize(newSize);
|
|
|
|
else
|
|
|
|
m_file.setSize(newSize);
|
2022-06-17 15:21:56 +03:00
|
|
|
|
2024-03-10 14:31:39 +01:00
|
|
|
m_fileSize = newSize;
|
2022-06-17 15:21:56 +03:00
|
|
|
}
|
|
|
|
|
2023-12-07 12:06:26 +01:00
|
|
|
u64 FileProvider::getActualSize() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_fileSize;
|
2020-11-11 09:18:35 +01:00
|
|
|
}
|
|
|
|
|
2021-09-21 02:29:54 +02:00
|
|
|
std::string FileProvider::getName() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return wolv::util::toUTF8String(m_path.filename());
|
2021-09-21 02:29:54 +02:00
|
|
|
}
|
|
|
|
|
2023-06-04 10:42:11 +02:00
|
|
|
std::vector<FileProvider::Description> FileProvider::getDataDescription() const {
|
|
|
|
std::vector<Description> result;
|
2020-11-15 16:06:10 +01:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
result.emplace_back("hex.builtin.provider.file.path"_lang, wolv::util::toUTF8String(m_path));
|
2021-02-22 13:07:25 +01:00
|
|
|
result.emplace_back("hex.builtin.provider.file.size"_lang, hex::toByteString(this->getActualSize()));
|
2020-11-15 16:06:10 +01:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
if (m_fileStats.has_value()) {
|
2023-08-02 12:52:10 +02:00
|
|
|
std::string creationTime, accessTime, modificationTime;
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
try { creationTime = hex::format("{:%Y-%m-%d %H:%M:%S}", fmt::localtime(m_fileStats->st_ctime)); }
|
2023-08-02 12:52:10 +02:00
|
|
|
catch (const std::exception&) { creationTime = "???"; }
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
try { accessTime = hex::format("{:%Y-%m-%d %H:%M:%S}", fmt::localtime(m_fileStats->st_atime)); }
|
2023-08-02 12:52:10 +02:00
|
|
|
catch (const std::exception&) { accessTime = "???"; }
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
try { modificationTime = hex::format("{:%Y-%m-%d %H:%M:%S}", fmt::localtime(m_fileStats->st_mtime)); }
|
2023-08-02 12:52:10 +02:00
|
|
|
catch (const std::exception&) { modificationTime = "???"; }
|
|
|
|
|
|
|
|
result.emplace_back("hex.builtin.provider.file.creation"_lang, creationTime);
|
|
|
|
result.emplace_back("hex.builtin.provider.file.access"_lang, accessTime);
|
|
|
|
result.emplace_back("hex.builtin.provider.file.modification"_lang, modificationTime);
|
2020-11-15 16:06:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-01-24 09:07:11 +01:00
|
|
|
std::variant<std::string, i128> FileProvider::queryInformation(const std::string &category, const std::string &argument) {
|
|
|
|
if (category == "file_path")
|
2024-05-08 21:30:20 +02:00
|
|
|
return wolv::io::fs::toNormalizedPathString(m_path);
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "file_name")
|
2024-05-08 21:30:20 +02:00
|
|
|
return wolv::io::fs::toNormalizedPathString(m_path.filename());
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "file_extension")
|
2024-05-08 21:30:20 +02:00
|
|
|
return wolv::io::fs::toNormalizedPathString(m_path.extension());
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "creation_time")
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_fileStats->st_ctime;
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "access_time")
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_fileStats->st_atime;
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "modification_time")
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_fileStats->st_mtime;
|
2023-01-24 09:07:11 +01:00
|
|
|
else if (category == "permissions")
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_fileStats->st_mode & 0777;
|
2023-01-24 09:07:11 +01:00
|
|
|
else
|
|
|
|
return Provider::queryInformation(category, argument);
|
|
|
|
}
|
|
|
|
|
2022-08-12 15:11:27 +02:00
|
|
|
bool FileProvider::handleFilePicker() {
|
|
|
|
return fs::openFileBrowser(fs::DialogMode::Open, {}, [this](const auto &path) {
|
|
|
|
this->setPath(path);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-06-09 20:59:06 +02:00
|
|
|
std::vector<FileProvider::MenuEntry> FileProvider::getMenuEntries() {
|
|
|
|
FileProvider::MenuEntry loadMenuItem;
|
|
|
|
|
|
|
|
if (m_loadedIntoMemory)
|
|
|
|
loadMenuItem = { "hex.builtin.provider.file.menu.direct_access"_lang, [this] { this->convertToDirectAccess(); } };
|
|
|
|
else
|
|
|
|
loadMenuItem = { "hex.builtin.provider.file.menu.into_memory"_lang, [this] { this->convertToMemoryFile(); } };
|
|
|
|
|
2023-05-21 13:21:53 +02:00
|
|
|
return {
|
2023-12-19 13:10:25 +01:00
|
|
|
{ "hex.builtin.provider.file.menu.open_folder"_lang, [this] { fs::openFolderWithSelectionExternal(m_path); } },
|
|
|
|
{ "hex.builtin.provider.file.menu.open_file"_lang, [this] { fs::openFileExternal(m_path); } },
|
2024-06-09 20:59:06 +02:00
|
|
|
loadMenuItem
|
2023-05-21 13:21:53 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-03-04 11:36:37 +01:00
|
|
|
void FileProvider::setPath(const std::fs::path &path) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_path = path;
|
2021-12-12 00:41:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool FileProvider::open() {
|
2024-06-09 20:59:06 +02:00
|
|
|
const size_t maxMemoryFileSize = ContentRegistry::Settings::read<u64>("hex.builtin.setting.general", "hex.builtin.setting.general.max_mem_file_size", 128_MiB);
|
|
|
|
|
|
|
|
size_t fileSize = 0x00;
|
|
|
|
{
|
|
|
|
wolv::io::File file(m_path, wolv::io::File::Mode::Read);
|
|
|
|
if (!file.isValid())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
fileSize = file.getSize();
|
|
|
|
}
|
|
|
|
|
|
|
|
const bool directAccess = fileSize >= maxMemoryFileSize;
|
|
|
|
const bool result = open(directAccess);
|
|
|
|
|
|
|
|
if (result && directAccess) {
|
|
|
|
m_writable = false;
|
|
|
|
|
|
|
|
ui::PopupQuestion::open("hex.builtin.provider.file.too_large"_lang,
|
|
|
|
[this] {
|
|
|
|
m_writable = false;
|
|
|
|
},
|
|
|
|
[this] {
|
|
|
|
m_writable = true;
|
|
|
|
RequestUpdateWindowTitle::post();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool FileProvider::open(bool directAccess) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_readable = true;
|
|
|
|
m_writable = true;
|
2021-07-27 21:07:36 +02:00
|
|
|
|
2024-03-17 13:20:02 +01:00
|
|
|
if (!wolv::io::fs::exists(m_path)) {
|
2023-12-19 13:10:25 +01:00
|
|
|
this->setErrorMessage(hex::format("hex.builtin.provider.file.error.open"_lang, m_path.string(), ::strerror(ENOENT)));
|
2023-06-21 20:07:36 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
wolv::io::File file(m_path, wolv::io::File::Mode::Write);
|
2023-02-17 10:26:09 +01:00
|
|
|
if (!file.isValid()) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_writable = false;
|
2023-05-19 20:01:42 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
file = wolv::io::File(m_path, wolv::io::File::Mode::Read);
|
2023-05-19 20:01:42 +02:00
|
|
|
if (!file.isValid()) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_readable = false;
|
|
|
|
this->setErrorMessage(hex::format("hex.builtin.provider.file.error.open"_lang, m_path.string(), ::strerror(errno)));
|
2023-05-19 20:01:42 +02:00
|
|
|
return false;
|
|
|
|
}
|
2024-02-27 20:20:14 +01:00
|
|
|
|
|
|
|
ui::ToastInfo::open("hex.builtin.popup.error.read_only"_lang);
|
2023-02-17 10:26:09 +01:00
|
|
|
}
|
2022-08-08 21:23:52 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
m_file = std::move(file);
|
2024-01-06 16:09:28 +01:00
|
|
|
m_fileStats = m_file.getFileInfo();
|
|
|
|
m_fileSize = m_file.getSize();
|
2023-05-20 18:07:15 +02:00
|
|
|
|
2024-02-24 12:06:06 +01:00
|
|
|
// Make sure the current file is not already opened
|
|
|
|
{
|
|
|
|
auto alreadyOpenedFileProvider = std::ranges::find_if(s_openedFiles, [this](const FileProvider *provider) {
|
|
|
|
return provider->m_path == m_path;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (alreadyOpenedFileProvider != s_openedFiles.end()) {
|
|
|
|
ImHexApi::Provider::setCurrentProvider(*alreadyOpenedFileProvider);
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
s_openedFiles.insert(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-10 14:31:39 +01:00
|
|
|
if (m_writable) {
|
2024-06-09 20:59:06 +02:00
|
|
|
if (directAccess) {
|
|
|
|
m_loadedIntoMemory = false;
|
|
|
|
} else {
|
2024-03-10 14:31:39 +01:00
|
|
|
m_data = m_file.readVectorAtomic(0x00, m_fileSize);
|
|
|
|
if (!m_data.empty()) {
|
|
|
|
m_changeTracker = wolv::io::ChangeTracker(m_file);
|
2024-03-10 15:40:32 +01:00
|
|
|
m_changeTracker.startTracking([this]{ this->handleFileChange(); });
|
2024-03-10 14:31:39 +01:00
|
|
|
m_file.close();
|
|
|
|
m_loadedIntoMemory = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-04 11:16:20 +02:00
|
|
|
return true;
|
2022-08-08 21:23:52 +02:00
|
|
|
}
|
|
|
|
|
2024-06-09 20:59:06 +02:00
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
void FileProvider::close() {
|
2024-01-06 16:09:28 +01:00
|
|
|
m_file.close();
|
2024-03-10 14:31:39 +01:00
|
|
|
m_data.clear();
|
2024-02-24 12:06:06 +01:00
|
|
|
s_openedFiles.erase(this);
|
2024-03-10 14:31:39 +01:00
|
|
|
m_changeTracker.stopTracking();
|
2022-08-08 21:23:52 +02:00
|
|
|
}
|
2021-12-12 00:41:44 +01:00
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
void FileProvider::loadSettings(const nlohmann::json &settings) {
|
|
|
|
Provider::loadSettings(settings);
|
2022-04-17 16:57:30 +02:00
|
|
|
|
2023-05-16 11:33:00 +02:00
|
|
|
auto pathString = settings.at("path").get<std::string>();
|
2023-01-11 23:31:25 +01:00
|
|
|
std::fs::path path = std::u8string(pathString.begin(), pathString.end());
|
|
|
|
|
2023-07-19 22:35:55 +02:00
|
|
|
if (auto projectPath = ProjectFile::getPath(); !projectPath.empty()) {
|
2024-06-20 11:11:01 +02:00
|
|
|
std::fs::path fullPath;
|
2023-07-19 22:35:55 +02:00
|
|
|
try {
|
2024-06-20 11:11:01 +02:00
|
|
|
fullPath = std::fs::weakly_canonical(projectPath.parent_path() / path);
|
2023-07-19 22:35:55 +02:00
|
|
|
} catch (const std::fs::filesystem_error &) {
|
2024-06-20 11:11:01 +02:00
|
|
|
fullPath = projectPath.parent_path() / path;
|
2023-07-19 22:35:55 +02:00
|
|
|
}
|
2024-06-20 11:11:01 +02:00
|
|
|
|
|
|
|
if (!wolv::io::fs::exists(fullPath))
|
|
|
|
fullPath = path;
|
|
|
|
|
|
|
|
if (!wolv::io::fs::exists(fullPath)) {
|
|
|
|
this->setErrorMessage(hex::format("hex.builtin.provider.file.error.open"_lang, m_path.string(), ::strerror(ENOENT)));
|
|
|
|
}
|
|
|
|
|
|
|
|
path = std::move(fullPath);
|
2023-12-27 16:33:49 +01:00
|
|
|
}
|
2024-06-20 11:11:01 +02:00
|
|
|
|
|
|
|
this->setPath(path);
|
2021-07-27 21:07:36 +02:00
|
|
|
}
|
|
|
|
|
2022-08-08 21:23:52 +02:00
|
|
|
nlohmann::json FileProvider::storeSettings(nlohmann::json settings) const {
|
2024-06-20 11:11:01 +02:00
|
|
|
std::fs::path path;
|
|
|
|
|
|
|
|
if (m_path.u8string().starts_with(u8"//")) {
|
|
|
|
path = m_path;
|
|
|
|
} else {
|
|
|
|
if (auto projectPath = ProjectFile::getPath(); !projectPath.empty())
|
|
|
|
path = std::fs::proximate(m_path, projectPath.parent_path());
|
|
|
|
if (path.empty())
|
|
|
|
path = m_path;
|
|
|
|
}
|
2023-01-19 11:09:24 +01:00
|
|
|
|
2024-06-20 11:11:01 +02:00
|
|
|
settings["path"] = wolv::io::fs::toNormalizedPathString(path);
|
2022-08-08 21:23:52 +02:00
|
|
|
|
|
|
|
return Provider::storeSettings(settings);
|
2021-07-27 21:07:36 +02:00
|
|
|
}
|
|
|
|
|
2022-08-14 09:38:38 +02:00
|
|
|
std::pair<Region, bool> FileProvider::getRegionValidity(u64 address) const {
|
2022-08-17 16:28:44 +02:00
|
|
|
address -= this->getBaseAddress();
|
|
|
|
|
2022-08-14 09:38:38 +02:00
|
|
|
if (address < this->getActualSize())
|
2022-08-17 16:28:44 +02:00
|
|
|
return { Region { this->getBaseAddress() + address, this->getActualSize() - address }, true };
|
2022-08-14 09:38:38 +02:00
|
|
|
else
|
|
|
|
return { Region::Invalid(), false };
|
|
|
|
}
|
|
|
|
|
2023-08-09 20:04:12 +02:00
|
|
|
void FileProvider::convertToMemoryFile() {
|
2024-06-09 20:59:06 +02:00
|
|
|
this->close();
|
|
|
|
this->open(false);
|
|
|
|
}
|
2023-08-09 20:04:12 +02:00
|
|
|
|
2024-06-09 20:59:06 +02:00
|
|
|
void FileProvider::convertToDirectAccess() {
|
|
|
|
this->close();
|
|
|
|
this->open(true);
|
2023-08-09 20:04:12 +02:00
|
|
|
}
|
|
|
|
|
2024-03-10 14:31:39 +01:00
|
|
|
void FileProvider::handleFileChange() {
|
2024-03-15 17:57:06 +01:00
|
|
|
if (m_ignoreNextChangeEvent) {
|
|
|
|
m_ignoreNextChangeEvent = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-06-09 20:59:06 +02:00
|
|
|
if (m_changeEventAcknowledgementPending) {
|
2024-05-07 14:43:20 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_changeEventAcknowledgementPending = true;
|
|
|
|
|
2024-03-10 14:31:39 +01:00
|
|
|
ui::PopupQuestion::open("hex.builtin.provider.file.reload_changes"_lang, [this] {
|
|
|
|
this->close();
|
2024-06-09 20:59:06 +02:00
|
|
|
(void)this->open(!m_loadedIntoMemory);
|
2024-03-10 14:31:39 +01:00
|
|
|
getUndoStack().reapply();
|
2024-05-07 14:43:20 -07:00
|
|
|
m_changeEventAcknowledgementPending = false;
|
2024-03-10 14:31:39 +01:00
|
|
|
},
|
2024-05-07 14:43:20 -07:00
|
|
|
[this]{
|
|
|
|
m_changeEventAcknowledgementPending = false;
|
|
|
|
});
|
2024-03-10 14:31:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-12-22 07:36:26 -05:00
|
|
|
}
|