2023-11-10 20:47:08 +01:00
|
|
|
#pragma once
|
|
|
|
|
2023-04-08 00:58:53 +02:00
|
|
|
#include <hex/ui/popup.hpp>
|
|
|
|
|
2023-11-21 14:38:01 +01:00
|
|
|
#include <hex/api/localization_manager.hpp>
|
2023-04-08 00:58:53 +02:00
|
|
|
|
2023-07-09 12:53:31 +02:00
|
|
|
#include <wolv/utils/string.hpp>
|
|
|
|
|
2023-04-08 00:58:53 +02:00
|
|
|
#include <functional>
|
|
|
|
#include <string>
|
|
|
|
|
2023-12-23 21:09:41 +01:00
|
|
|
namespace hex::ui {
|
2023-04-08 00:58:53 +02:00
|
|
|
|
|
|
|
class PopupFileChooser : public Popup<PopupFileChooser> {
|
|
|
|
public:
|
2023-11-14 00:37:07 +01:00
|
|
|
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)
|
2023-12-23 21:09:41 +01:00
|
|
|
: hex::Popup<PopupFileChooser>("hex.ui.common.choose_file"),
|
2023-11-14 00:37:07 +01:00
|
|
|
m_indices({ }),
|
2023-04-08 00:58:53 +02:00
|
|
|
m_openCallback(callback),
|
2023-05-21 18:41:43 +02:00
|
|
|
m_validExtensions(validExtensions), m_multiple(multiple) {
|
|
|
|
|
2023-11-14 00:37:07 +01:00
|
|
|
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-11-14 00:37:07 +01:00
|
|
|
}
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
std::sort(m_files.begin(), m_files.end(), [](const auto &a, const auto &b) {
|
2023-11-14 00:37:07 +01:00
|
|
|
return a.first < b.first;
|
2023-05-21 18:41:43 +02:00
|
|
|
});
|
|
|
|
}
|
2023-04-08 00:58:53 +02:00
|
|
|
|
|
|
|
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) {
|
2023-04-08 00:58:53 +02:00
|
|
|
ImGui::PushID(index);
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
bool selected = m_indices.contains(index);
|
2023-11-14 00:37:07 +01:00
|
|
|
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);
|
2023-04-08 00:58:53 +02:00
|
|
|
} else {
|
|
|
|
if (selected) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_indices.erase(index);
|
2023-04-08 00:58:53 +02:00
|
|
|
} else {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_indices.insert(index);
|
2023-04-08 00:58:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ImGui::IsItemHovered() && ImGui::IsMouseDoubleClicked(0))
|
|
|
|
doubleClicked = true;
|
|
|
|
|
2023-11-16 22:24:06 +01:00
|
|
|
ImGuiExt::InfoTooltip(wolv::util::toUTF8String(path).c_str());
|
2023-04-08 00:58:53 +02:00
|
|
|
|
|
|
|
ImGui::PopID();
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::EndListBox();
|
|
|
|
}
|
|
|
|
|
2023-12-23 21:09:41 +01:00
|
|
|
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();
|
2023-04-08 00:58:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::SameLine();
|
|
|
|
|
2023-12-23 21:09:41 +01:00
|
|
|
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);
|
2023-04-08 00:58:53 +02:00
|
|
|
}
|
2023-12-13 15:08:27 +01:00
|
|
|
|
|
|
|
if (ImGui::IsKeyPressed(ImGui::GetKeyIndex(ImGuiKey_Escape)))
|
|
|
|
this->close();
|
2023-04-08 00:58:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] ImGuiWindowFlags getFlags() const override {
|
|
|
|
return ImGuiWindowFlags_AlwaysAutoResize;
|
|
|
|
}
|
|
|
|
|
2023-11-14 00:37:07 +01:00
|
|
|
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] != '.';
|
|
|
|
}
|
|
|
|
|
2023-04-08 00:58:53 +02:00
|
|
|
private:
|
|
|
|
std::set<u32> m_indices;
|
2023-11-14 00:37:07 +01:00
|
|
|
std::vector<std::pair<std::fs::path, std::fs::path>> m_files;
|
2023-04-08 00:58:53 +02:00
|
|
|
std::function<void(std::fs::path)> m_openCallback;
|
2023-10-04 12:00:32 +02:00
|
|
|
std::vector<hex::fs::ItemFilter> m_validExtensions;
|
2023-04-08 00:58:53 +02:00
|
|
|
bool m_multiple = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|