1
0
mirror of synced 2025-01-09 13:11:38 +01:00
ImHex/plugins/ui/include/popups/popup_file_chooser.hpp

114 lines
3.9 KiB
C++
Raw Normal View History

2023-11-10 20:47:08 +01:00
#pragma once
#include <hex/ui/popup.hpp>
#include <hex/api/localization_manager.hpp>
2023-07-09 12:53:31 +02:00
#include <wolv/utils/string.hpp>
#include <functional>
#include <string>
namespace hex::ui {
class PopupFileChooser : public Popup<PopupFileChooser> {
public:
PopupFileChooser(const std::vector<std::fs::path> &basePaths, const std::vector<std::fs::path> &files, const std::vector<hex::fs::ItemFilter> &validExtensions, bool multiple, const std::function<void(std::fs::path)> &callback)
: hex::Popup<PopupFileChooser>("hex.ui.common.choose_file"),
m_indices({ }),
m_openCallback(callback),
m_validExtensions(validExtensions), m_multiple(multiple) {
for (const auto &path : files) {
std::fs::path adjustedPath;
for (const auto &basePath : basePaths) {
if (isSubpath(basePath, path)) {
adjustedPath = std::fs::relative(path, basePath);
break;
}
}
if (adjustedPath.empty())
adjustedPath = path.filename();
2023-12-19 13:10:25 +01:00
m_files.push_back({ path, adjustedPath });
}
2023-12-19 13:10:25 +01:00
std::sort(m_files.begin(), m_files.end(), [](const auto &a, const auto &b) {
return a.first < b.first;
});
}
void drawContent() override {
bool doubleClicked = false;
if (ImGui::BeginListBox("##files", scaled(ImVec2(500, 400)))) {
u32 index = 0;
2023-12-19 13:10:25 +01:00
for (auto &[path, pathName] : m_files) {
ImGui::PushID(index);
2023-12-19 13:10:25 +01:00
bool selected = m_indices.contains(index);
if (ImGui::Selectable(wolv::util::toUTF8String(pathName).c_str(), selected, ImGuiSelectableFlags_DontClosePopups)) {
2023-12-19 13:10:25 +01:00
if (!m_multiple) {
m_indices.clear();
m_indices.insert(index);
} else {
if (selected) {
2023-12-19 13:10:25 +01:00
m_indices.erase(index);
} else {
2023-12-19 13:10:25 +01:00
m_indices.insert(index);
}
}
}
if (ImGui::IsItemHovered() && ImGui::IsMouseDoubleClicked(0))
doubleClicked = true;
ImGuiExt::InfoTooltip(wolv::util::toUTF8String(path).c_str());
ImGui::PopID();
index++;
}
ImGui::EndListBox();
}
if (ImGui::Button("hex.ui.common.open"_lang) || doubleClicked) {
2023-12-19 13:10:25 +01:00
for (const auto &index : m_indices)
m_openCallback(m_files[index].first);
2023-04-08 23:34:46 +02:00
Popup::close();
}
ImGui::SameLine();
if (ImGui::Button("hex.ui.common.browse"_lang)) {
2023-12-19 13:10:25 +01:00
fs::openFileBrowser(fs::DialogMode::Open, m_validExtensions, [this](const auto &path) {
m_openCallback(path);
2023-04-08 23:34:46 +02:00
Popup::close();
2023-12-19 13:10:25 +01:00
}, {}, m_multiple);
}
if (ImGui::IsKeyPressed(ImGui::GetKeyIndex(ImGuiKey_Escape)))
this->close();
}
[[nodiscard]] ImGuiWindowFlags getFlags() const override {
return ImGuiWindowFlags_AlwaysAutoResize;
}
private:
static bool isSubpath(const std::fs::path &basePath, const std::fs::path &path) {
auto relativePath = std::fs::relative(path, basePath);
return !relativePath.empty() && relativePath.native()[0] != '.';
}
private:
std::set<u32> m_indices;
std::vector<std::pair<std::fs::path, std::fs::path>> m_files;
std::function<void(std::fs::path)> m_openCallback;
std::vector<hex::fs::ItemFilter> m_validExtensions;
bool m_multiple = false;
};
}