geode/loader/include/Geode/utils/file.hpp

304 lines
9.1 KiB
C++
Raw Normal View History

2022-07-30 12:24:03 -04:00
#pragma once
#include "Result.hpp"
#include "general.hpp"
2023-04-27 03:41:07 -04:00
#include "../loader/Event.hpp"
2022-10-30 14:59:20 -04:00
#include <matjson.hpp>
2022-10-30 14:59:20 -04:00
#include <Geode/DefaultInclude.hpp>
2023-02-08 08:42:34 -05:00
#include <ghc/fs_fwd.hpp>
2022-10-30 14:59:20 -04:00
#include <string>
#include <unordered_set>
2022-07-30 12:24:03 -04:00
template <>
struct matjson::Serialize<ghc::filesystem::path> {
static matjson::Value to_json(ghc::filesystem::path const& path) {
return path.string();
}
static ghc::filesystem::path from_json(matjson::Value const& value) {
return value.as_string();
}
};
namespace geode::utils::file {
2022-10-30 14:59:20 -04:00
GEODE_DLL Result<std::string> readString(ghc::filesystem::path const& path);
GEODE_DLL Result<matjson::Value> readJson(ghc::filesystem::path const& path);
2022-12-14 06:11:19 -05:00
GEODE_DLL Result<ByteVector> readBinary(ghc::filesystem::path const& path);
2022-07-30 12:24:03 -04:00
template <class T>
Result<T> readFromJson(ghc::filesystem::path const& file) {
GEODE_UNWRAP_INTO(auto json, readJson(file));
try {
return json.template as<T>();
}
catch(std::exception& e) {
return Err("Error parsing JSON: {}", e.what());
}
}
2022-07-30 12:24:03 -04:00
GEODE_DLL Result<> writeString(ghc::filesystem::path const& path, std::string const& data);
2022-12-14 06:11:19 -05:00
GEODE_DLL Result<> writeBinary(ghc::filesystem::path const& path, ByteVector const& data);
2022-07-30 12:24:03 -04:00
template <class T>
Result<> writeToJson(ghc::filesystem::path const& path, T const& data) {
try {
GEODE_UNWRAP(writeString(path, matjson::Value(data).dump()));
return Ok();
}
catch(std::exception& e) {
return Err("Error serializing JSON: {}", e.what());
}
}
GEODE_DLL Result<> createDirectory(ghc::filesystem::path const& path);
GEODE_DLL Result<> createDirectoryAll(ghc::filesystem::path const& path);
GEODE_DLL Result<std::vector<ghc::filesystem::path>> readDirectory(
ghc::filesystem::path const& path, bool recursive = false
);
class Unzip;
class GEODE_DLL Zip final {
public:
using Path = ghc::filesystem::path;
private:
class Impl;
std::unique_ptr<Impl> m_impl;
Zip();
Zip(std::unique_ptr<Impl>&& impl);
Result<> addAllFromRecurse(
Path const& dir, Path const& entry
);
// for sharing Impl
friend class Unzip;
public:
Zip(Zip const&) = delete;
Zip(Zip&& other);
~Zip();
/**
* Create zipper for file
*/
static Result<Zip> create(Path const& file);
/**
* Create zipper for in-memory data
*/
static Result<Zip> create();
/**
* Path to the created zip
* @returns The path to the zip that is being created, or an empty path
* if the zip was opened in memory
*/
Path getPath() const;
/**
* Get the zipped data
*/
ByteVector getData() const;
/**
* Add an entry to the zip with data
*/
2022-12-14 06:11:19 -05:00
Result<> add(Path const& entry, ByteVector const& data);
/**
* Add an entry to the zip with string data
*/
Result<> add(Path const& entry, std::string const& data);
/**
* Add an entry to the zip from a file on disk. If you want to add the
* file with a different name, read it into memory first and add it
* with Zip::add
* @param file File on disk
* @param entryDir Folder to place the file in in the zip
*/
Result<> addFrom(Path const& file, Path const& entryDir = Path());
/**
* Add an entry to the zip from a directory on disk
* @param entry Path in the zip
* @param dir Directory on disk
*/
Result<> addAllFrom(Path const& dir);
/**
* Add a folder entry to the zip. If you want to add a folder from disk,
* use Zip::addAllFrom
* @param entry Folder path in zip
*/
Result<> addFolder(Path const& entry);
};
class GEODE_DLL Unzip final {
private:
using Impl = Zip::Impl;
std::unique_ptr<Impl> m_impl;
Unzip();
Unzip(std::unique_ptr<Impl>&& impl);
public:
Unzip(Unzip const&) = delete;
Unzip(Unzip&& other);
~Unzip();
using Path = ghc::filesystem::path;
/**
* Create unzipper for file
*/
static Result<Unzip> create(Path const& file);
/**
* Create unzipper for data in-memory
*/
static Result<Unzip> create(ByteVector const& data);
/**
* Path to the opened zip
* @returns The path to the zip that is being read, or an empty path
* if the zip was opened in memory
*/
Path getPath() const;
/**
* Get all entries in zip
*/
std::vector<Path> getEntries() const;
/**
* Check if zip has entry
* @param name Entry path in zip
*/
bool hasEntry(Path const& name);
/**
* Extract entry to memory
* @param name Entry path in zip
*/
2022-12-14 06:11:19 -05:00
Result<ByteVector> extract(Path const& name);
/**
* Extract entry to file
* @param name Entry path in zip
* @param path Target file path
*/
Result<> extractTo(Path const& name, Path const& path);
/**
* Extract all entries to directory
* @param dir Directory to unzip the contents to
*/
Result<> extractAllTo(Path const& dir);
/**
* Helper method for quickly unzipping a file
* @param from ZIP file to unzip
* @param to Directory to unzip to
* @param deleteZipAfter Whether to delete the zip after unzipping
* @returns Succesful result on success, errorful result on error
*/
static Result<> intoDir(
Path const& from,
Path const& to,
bool deleteZipAfter = false
);
};
/**
* Open a folder / file in the system's file explorer
* @param path Folder / file to open
*/
GEODE_DLL bool openFolder(ghc::filesystem::path const& path);
enum class PickMode {
OpenFile,
SaveFile,
OpenFolder,
};
struct FilePickOptions {
struct Filter {
// Name of the filter
std::string description;
// Extensions (*.txt, *.doc, *.mp3, etc.)
std::unordered_set<std::string> files;
};
/**
* On PickMode::SaveFile and PickMode::OpenFile, last item is assumed
* to be a filename, unless it points to an extant directory.
* On PickMode::OpenFolder, path is treated as leading up to a directory
*/
std::optional<ghc::filesystem::path> defaultPath;
/**
* File extension filters to show on the file picker
*/
std::vector<Filter> filters;
};
/**
* Prompt the user to pick a file using the system's file system picker
2023-10-08 09:38:17 -04:00
* @note Will not work on Android, use the callback version instead
* @param mode Type of file selection prompt to show
* @param options Picker options
*/
GEODE_DLL Result<ghc::filesystem::path> pickFile(PickMode mode, FilePickOptions const& options);
2023-10-08 09:38:17 -04:00
GEODE_DLL void pickFile(
PickMode mode, FilePickOptions const& options,
utils::MiniFunction<void(ghc::filesystem::path)> callback,
utils::MiniFunction<void()> failed = {}
);
/**
* Prompt the user to pick a bunch of files for opening using the system's file system picker
2023-10-08 09:38:17 -04:00
* @note Will not work on Android, use the callback version instead
* @param options Picker options
*/
GEODE_DLL Result<std::vector<ghc::filesystem::path>> pickFiles(FilePickOptions const& options);
2023-04-27 02:22:56 -04:00
2023-10-08 09:38:17 -04:00
GEODE_DLL void pickFiles(
FilePickOptions const& options,
utils::MiniFunction<void(std::vector<ghc::filesystem::path>)> callback,
utils::MiniFunction<void()> failed = {}
);
2023-04-27 02:22:56 -04:00
class GEODE_DLL FileWatchEvent : public Event {
protected:
ghc::filesystem::path m_path;
public:
FileWatchEvent(ghc::filesystem::path const& path);
ghc::filesystem::path getPath() const;
};
class GEODE_DLL FileWatchFilter : public EventFilter<FileWatchEvent> {
protected:
ghc::filesystem::path m_path;
public:
using Callback = void(FileWatchEvent*);
ListenerResult handle(utils::MiniFunction<Callback> callback, FileWatchEvent* event);
FileWatchFilter(ghc::filesystem::path const& path);
};
/**
* Watch a file for changes. Whenever the file is modified on disk, a
* FileWatchEvent is emitted. Add an EventListener with FileWatchFilter
* to catch these events
* @param file The file to watch
* @note Watching uses file system equivalence instead of path equivalence,
* so different paths that point to the same file will be considered the
* same
*/
GEODE_DLL Result<> watchFile(ghc::filesystem::path const& file);
/**
* Stop watching a file for changes
* @param file The file to unwatch
*/
GEODE_DLL void unwatchFile(ghc::filesystem::path const& file);
2022-07-30 12:24:03 -04:00
}