add pack.request_writeable(...) & update gui.confirm screen
This commit is contained in:
parent
4cdb1fbae2
commit
88b45e5098
@ -35,6 +35,7 @@ GUI::GUI()
|
|||||||
|
|
||||||
menu = std::make_shared<Menu>();
|
menu = std::make_shared<Menu>();
|
||||||
menu->setId("menu");
|
menu->setId("menu");
|
||||||
|
menu->setZIndex(10);
|
||||||
container->add(menu);
|
container->add(menu);
|
||||||
container->setScrollable(false);
|
container->setScrollable(false);
|
||||||
|
|
||||||
|
|||||||
@ -91,7 +91,11 @@ void guiutil::confirm(
|
|||||||
if (yestext.empty()) yestext = langs::get(L"Yes");
|
if (yestext.empty()) yestext = langs::get(L"Yes");
|
||||||
if (notext.empty()) notext = langs::get(L"No");
|
if (notext.empty()) notext = langs::get(L"No");
|
||||||
|
|
||||||
|
auto container = std::make_shared<Container>(glm::vec2(5000, 5000));
|
||||||
|
container->setColor(glm::vec4(0.05f, 0.05f, 0.05f, 0.7f));
|
||||||
auto panel = std::make_shared<Panel>(glm::vec2(600, 200), glm::vec4(8.0f), 8.0f);
|
auto panel = std::make_shared<Panel>(glm::vec2(600, 200), glm::vec4(8.0f), 8.0f);
|
||||||
|
panel->setGravity(Gravity::center_center);
|
||||||
|
container->add(panel);
|
||||||
panel->setColor(glm::vec4(0.0f, 0.0f, 0.0f, 0.5f));
|
panel->setColor(glm::vec4(0.0f, 0.0f, 0.0f, 0.5f));
|
||||||
panel->add(std::make_shared<Label>(text));
|
panel->add(std::make_shared<Label>(text));
|
||||||
auto subpanel = std::make_shared<Panel>(glm::vec2(600, 53));
|
auto subpanel = std::make_shared<Panel>(glm::vec2(600, 53));
|
||||||
@ -103,8 +107,8 @@ void guiutil::confirm(
|
|||||||
menu->removePage("<confirm>");
|
menu->removePage("<confirm>");
|
||||||
if (on_confirm) {
|
if (on_confirm) {
|
||||||
on_confirm();
|
on_confirm();
|
||||||
} else {
|
} else if (!menu->back()) {
|
||||||
menu->back();
|
menu->reset();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -112,8 +116,8 @@ void guiutil::confirm(
|
|||||||
menu->removePage("<confirm>");
|
menu->removePage("<confirm>");
|
||||||
if (on_deny) {
|
if (on_deny) {
|
||||||
on_deny();
|
on_deny();
|
||||||
} else {
|
} else if (!menu->back()) {
|
||||||
menu->back();
|
menu->reset();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -136,7 +140,7 @@ void guiutil::confirm(
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
panel->refresh();
|
panel->refresh();
|
||||||
menu->addPage("<confirm>", panel, true);
|
menu->addPage("<confirm>", container, true);
|
||||||
menu->setPage("<confirm>");
|
menu->setPage("<confirm>");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -150,6 +150,37 @@ void EnginePaths::setCurrentWorldFolder(io::path folder) {
|
|||||||
io::create_subdevice("world", "user", currentWorldFolder);
|
io::create_subdevice("world", "user", currentWorldFolder);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
#include "maths/util.hpp"
|
||||||
|
|
||||||
|
std::string EnginePaths::createWriteablePackDevice(const std::string& name) {
|
||||||
|
const auto& found = writeablePacks.find(name);
|
||||||
|
if (found != writeablePacks.end()) {
|
||||||
|
return found->second;
|
||||||
|
}
|
||||||
|
io::path folder;
|
||||||
|
for (const auto& pack : *contentPacks) {
|
||||||
|
if (pack.id == name) {
|
||||||
|
folder = pack.folder;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (folder.emptyOrInvalid()) {
|
||||||
|
throw std::runtime_error("pack not found");
|
||||||
|
}
|
||||||
|
|
||||||
|
auto now = std::chrono::high_resolution_clock::now();
|
||||||
|
auto seed = now.time_since_epoch().count();
|
||||||
|
|
||||||
|
util::PseudoRandom random(seed); // fixme: replace with safe random
|
||||||
|
auto number = random.rand64();
|
||||||
|
auto entryPoint = std::string("W.") + util::base64_urlsafe_encode(reinterpret_cast<ubyte*>(&number), 6);
|
||||||
|
|
||||||
|
io::create_subdevice(entryPoint, folder.entryPoint(), folder.pathPart());
|
||||||
|
writeablePacks[name] = entryPoint;
|
||||||
|
return entryPoint;
|
||||||
|
}
|
||||||
|
|
||||||
void EnginePaths::setContentPacks(std::vector<ContentPack>* contentPacks) {
|
void EnginePaths::setContentPacks(std::vector<ContentPack>* contentPacks) {
|
||||||
// Remove previous content entry-points
|
// Remove previous content entry-points
|
||||||
for (const auto& id : contentEntryPoints) {
|
for (const auto& id : contentEntryPoints) {
|
||||||
|
|||||||
@ -1,5 +1,6 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <unordered_map>
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
#include <string>
|
#include <string>
|
||||||
@ -33,6 +34,8 @@ public:
|
|||||||
io::path getControlsFile() const;
|
io::path getControlsFile() const;
|
||||||
io::path getSettingsFile() const;
|
io::path getSettingsFile() const;
|
||||||
|
|
||||||
|
std::string createWriteablePackDevice(const std::string& name);
|
||||||
|
|
||||||
void setContentPacks(std::vector<ContentPack>* contentPacks);
|
void setContentPacks(std::vector<ContentPack>* contentPacks);
|
||||||
|
|
||||||
std::vector<io::path> scanForWorlds() const;
|
std::vector<io::path> scanForWorlds() const;
|
||||||
@ -47,6 +50,7 @@ private:
|
|||||||
std::optional<std::filesystem::path> scriptFolder;
|
std::optional<std::filesystem::path> scriptFolder;
|
||||||
std::vector<ContentPack>* contentPacks = nullptr;
|
std::vector<ContentPack>* contentPacks = nullptr;
|
||||||
std::vector<std::string> contentEntryPoints;
|
std::vector<std::string> contentEntryPoints;
|
||||||
|
std::unordered_map<std::string, std::string> writeablePacks;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct PathsRoot {
|
struct PathsRoot {
|
||||||
|
|||||||
@ -41,10 +41,23 @@ static std::set<std::string> writeable_entry_points {
|
|||||||
"world", "export", "config"
|
"world", "export", "config"
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static bool is_writeable(const std::string& entryPoint) {
|
||||||
|
if (entryPoint.length() < 2) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (entryPoint.substr(0, 2) == "W.") {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (writeable_entry_points.find(entryPoint) != writeable_entry_points.end()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
static io::path get_writeable_path(lua::State* L) {
|
static io::path get_writeable_path(lua::State* L) {
|
||||||
io::path path = lua::require_string(L, 1);
|
io::path path = lua::require_string(L, 1);
|
||||||
auto entryPoint = path.entryPoint();
|
auto entryPoint = path.entryPoint();
|
||||||
if (writeable_entry_points.find(entryPoint) == writeable_entry_points.end()) {
|
if (!is_writeable(entryPoint)) {
|
||||||
throw std::runtime_error("access denied");
|
throw std::runtime_error("access denied");
|
||||||
}
|
}
|
||||||
return path;
|
return path;
|
||||||
@ -60,7 +73,7 @@ static int l_write(lua::State* L) {
|
|||||||
static int l_remove(lua::State* L) {
|
static int l_remove(lua::State* L) {
|
||||||
io::path path = lua::require_string(L, 1);
|
io::path path = lua::require_string(L, 1);
|
||||||
auto entryPoint = path.entryPoint();
|
auto entryPoint = path.entryPoint();
|
||||||
if (writeable_entry_points.find(entryPoint) == writeable_entry_points.end()) {
|
if (!is_writeable(entryPoint)) {
|
||||||
throw std::runtime_error("access denied");
|
throw std::runtime_error("access denied");
|
||||||
}
|
}
|
||||||
return lua::pushboolean(L, io::remove(path));
|
return lua::pushboolean(L, io::remove(path));
|
||||||
@ -69,7 +82,7 @@ static int l_remove(lua::State* L) {
|
|||||||
static int l_remove_tree(lua::State* L) {
|
static int l_remove_tree(lua::State* L) {
|
||||||
io::path path = lua::require_string(L, 1);
|
io::path path = lua::require_string(L, 1);
|
||||||
auto entryPoint = path.entryPoint();
|
auto entryPoint = path.entryPoint();
|
||||||
if (writeable_entry_points.find(entryPoint) == writeable_entry_points.end()) {
|
if (!is_writeable(entryPoint)) {
|
||||||
throw std::runtime_error("access denied");
|
throw std::runtime_error("access denied");
|
||||||
}
|
}
|
||||||
return lua::pushinteger(L, io::remove_all(path));
|
return lua::pushinteger(L, io::remove_all(path));
|
||||||
@ -222,10 +235,7 @@ static int l_read_combined_object(lua::State* L) {
|
|||||||
static int l_is_writeable(lua::State* L) {
|
static int l_is_writeable(lua::State* L) {
|
||||||
io::path path = lua::require_string(L, 1);
|
io::path path = lua::require_string(L, 1);
|
||||||
auto entryPoint = path.entryPoint();
|
auto entryPoint = path.entryPoint();
|
||||||
if (writeable_entry_points.find(entryPoint) == writeable_entry_points.end()) {
|
return lua::pushboolean(L, is_writeable(entryPoint));
|
||||||
return lua::pushboolean(L, false);
|
|
||||||
}
|
|
||||||
return lua::pushboolean(L, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const luaL_Reg filelib[] = {
|
const luaL_Reg filelib[] = {
|
||||||
@ -249,4 +259,5 @@ const luaL_Reg filelib[] = {
|
|||||||
{"read_combined_list", lua::wrap<l_read_combined_list>},
|
{"read_combined_list", lua::wrap<l_read_combined_list>},
|
||||||
{"read_combined_object", lua::wrap<l_read_combined_object>},
|
{"read_combined_object", lua::wrap<l_read_combined_object>},
|
||||||
{"is_writeable", lua::wrap<l_is_writeable>},
|
{"is_writeable", lua::wrap<l_is_writeable>},
|
||||||
{NULL, NULL}};
|
{NULL, NULL}
|
||||||
|
};
|
||||||
|
|||||||
@ -7,6 +7,9 @@
|
|||||||
#include "assets/AssetsLoader.hpp"
|
#include "assets/AssetsLoader.hpp"
|
||||||
#include "content/Content.hpp"
|
#include "content/Content.hpp"
|
||||||
#include "engine/Engine.hpp"
|
#include "engine/Engine.hpp"
|
||||||
|
#include "graphics/ui/gui_util.hpp"
|
||||||
|
#include "graphics/ui/elements/Menu.hpp"
|
||||||
|
#include "frontend/locale.hpp"
|
||||||
#include "world/files/WorldFiles.hpp"
|
#include "world/files/WorldFiles.hpp"
|
||||||
#include "io/engine_paths.hpp"
|
#include "io/engine_paths.hpp"
|
||||||
#include "world/Level.hpp"
|
#include "world/Level.hpp"
|
||||||
@ -247,6 +250,20 @@ static int l_pack_assemble(lua::State* L) {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int l_pack_request_writeable(lua::State* L) {
|
||||||
|
auto packid = lua::require_string(L, 1);
|
||||||
|
lua::pushvalue(L, 2);
|
||||||
|
auto handler = lua::create_lambda_nothrow(L);
|
||||||
|
|
||||||
|
auto str = langs::get(L"Grant %{0} pack modification permission?");
|
||||||
|
util::replaceAll(str, L"%{0}", util::str2wstr_utf8(packid));
|
||||||
|
guiutil::confirm(*engine, str, [packid, handler]() {
|
||||||
|
handler({engine->getPaths().createWriteablePackDevice(packid)});
|
||||||
|
engine->getGUI()->getMenu()->reset();
|
||||||
|
});
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
const luaL_Reg packlib[] = {
|
const luaL_Reg packlib[] = {
|
||||||
{"get_folder", lua::wrap<l_pack_get_folder>},
|
{"get_folder", lua::wrap<l_pack_get_folder>},
|
||||||
{"get_installed", lua::wrap<l_pack_get_installed>},
|
{"get_installed", lua::wrap<l_pack_get_installed>},
|
||||||
@ -254,4 +271,6 @@ const luaL_Reg packlib[] = {
|
|||||||
{"get_info", lua::wrap<l_pack_get_info>},
|
{"get_info", lua::wrap<l_pack_get_info>},
|
||||||
{"get_base_packs", lua::wrap<l_pack_get_base_packs>},
|
{"get_base_packs", lua::wrap<l_pack_get_base_packs>},
|
||||||
{"assemble", lua::wrap<l_pack_assemble>},
|
{"assemble", lua::wrap<l_pack_assemble>},
|
||||||
{NULL, NULL}};
|
{"request_writeable", lua::wrap<l_pack_request_writeable>},
|
||||||
|
{NULL, NULL}
|
||||||
|
};
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user