mirror of
https://github.com/luanti-org/luanti.git
synced 2025-06-27 16:36:03 +00:00
415 lines
10 KiB
C++
415 lines
10 KiB
C++
// Luanti
|
|
// SPDX-License-Identifier: LGPL-2.1-or-later
|
|
// Copyright (C) 2024 grorp, Gregor Parzefall <grorp@posteo.de>
|
|
|
|
#include "touchscreenlayout.h"
|
|
#include "client/renderingengine.h"
|
|
#include "client/texturesource.h"
|
|
#include "convert_json.h"
|
|
#include "gettext.h"
|
|
#include "settings.h"
|
|
#include <json/json.h>
|
|
|
|
#include "IGUIFont.h"
|
|
#include "IGUIStaticText.h"
|
|
#include "util/enum_string.h"
|
|
|
|
const struct EnumString es_TouchInteractionStyle[] =
|
|
{
|
|
{TAP, "tap"},
|
|
{TAP_CROSSHAIR, "tap_crosshair"},
|
|
{BUTTONS_CROSSHAIR, "buttons_crosshair"},
|
|
{0, NULL},
|
|
};
|
|
|
|
const char *button_names[] = {
|
|
"dig",
|
|
"place",
|
|
|
|
"jump",
|
|
"sneak",
|
|
"zoom",
|
|
"aux1",
|
|
"overflow",
|
|
|
|
"chat",
|
|
"inventory",
|
|
"drop",
|
|
"exit",
|
|
|
|
"fly",
|
|
"fast",
|
|
"noclip",
|
|
"debug",
|
|
"camera",
|
|
"range",
|
|
"minimap",
|
|
"toggle_chat",
|
|
|
|
"joystick_off",
|
|
"joystick_bg",
|
|
"joystick_center",
|
|
};
|
|
|
|
// compare with GUIKeyChangeMenu::init_keys
|
|
const char *button_titles[] = {
|
|
N_("Dig/punch/use"),
|
|
N_("Place/use"),
|
|
|
|
N_("Jump"),
|
|
N_("Sneak"),
|
|
N_("Zoom"),
|
|
N_("Aux1"),
|
|
N_("Overflow menu"),
|
|
|
|
N_("Chat"),
|
|
N_("Inventory"),
|
|
N_("Drop"),
|
|
N_("Exit"),
|
|
|
|
N_("Toggle fly"),
|
|
N_("Toggle fast"),
|
|
N_("Toggle noclip"),
|
|
N_("Toggle debug"),
|
|
N_("Change camera"),
|
|
N_("Range select"),
|
|
N_("Toggle minimap"),
|
|
N_("Toggle chat log"),
|
|
|
|
N_("Joystick"),
|
|
N_("Joystick"),
|
|
N_("Joystick"),
|
|
};
|
|
|
|
const char *button_image_names[] = {
|
|
"dig_btn.png",
|
|
"place_btn.png",
|
|
|
|
"jump_btn.png",
|
|
"down.png",
|
|
"zoom.png",
|
|
"aux1_btn.png",
|
|
"overflow_btn.png",
|
|
|
|
"chat_btn.png",
|
|
"inventory_btn.png",
|
|
"drop_btn.png",
|
|
"exit_btn.png",
|
|
|
|
"fly_btn.png",
|
|
"fast_btn.png",
|
|
"noclip_btn.png",
|
|
"debug_btn.png",
|
|
"camera_btn.png",
|
|
"rangeview_btn.png",
|
|
"minimap_btn.png",
|
|
// toggle button: switches between "chat_hide_btn.png" and "chat_show_btn.png"
|
|
"chat_hide_btn.png",
|
|
|
|
"joystick_off.png",
|
|
"joystick_bg.png",
|
|
"joystick_center.png",
|
|
};
|
|
|
|
v2s32 ButtonMeta::getPos(v2u32 screensize, s32 button_size) const
|
|
{
|
|
return v2s32((position.X * screensize.X) + (offset.X * button_size),
|
|
(position.Y * screensize.Y) + (offset.Y * button_size));
|
|
}
|
|
|
|
void ButtonMeta::setPos(v2s32 pos, v2u32 screensize, s32 button_size)
|
|
{
|
|
v2s32 third(screensize.X / 3, screensize.Y / 3);
|
|
|
|
if (pos.X < third.X)
|
|
position.X = 0.0f;
|
|
else if (pos.X < 2 * third.X)
|
|
position.X = 0.5f;
|
|
else
|
|
position.X = 1.0f;
|
|
|
|
if (pos.Y < third.Y)
|
|
position.Y = 0.0f;
|
|
else if (pos.Y < 2 * third.Y)
|
|
position.Y = 0.5f;
|
|
else
|
|
position.Y = 1.0f;
|
|
|
|
offset.X = (pos.X - (position.X * screensize.X)) / button_size;
|
|
offset.Y = (pos.Y - (position.Y * screensize.Y)) / button_size;
|
|
}
|
|
|
|
bool ButtonLayout::isButtonValid(touch_gui_button_id id)
|
|
{
|
|
return id != joystick_off_id && id != joystick_bg_id && id != joystick_center_id &&
|
|
id < touch_gui_button_id_END;
|
|
}
|
|
|
|
static const char *buttons_crosshair = enum_to_string(es_TouchInteractionStyle, BUTTONS_CROSSHAIR);
|
|
|
|
bool ButtonLayout::isButtonAllowed(touch_gui_button_id id)
|
|
{
|
|
if (id == dig_id || id == place_id)
|
|
return g_settings->get("touch_interaction_style") == buttons_crosshair;
|
|
if (id == aux1_id)
|
|
return !g_settings->getBool("virtual_joystick_triggers_aux1");
|
|
return true;
|
|
}
|
|
|
|
bool ButtonLayout::isButtonRequired(touch_gui_button_id id)
|
|
{
|
|
return id == overflow_id;
|
|
}
|
|
|
|
s32 ButtonLayout::getButtonSize(v2u32 screensize)
|
|
{
|
|
return std::min(screensize.Y / 4.5f,
|
|
RenderingEngine::getDisplayDensity() * 65.0f *
|
|
g_settings->getFloat("hud_scaling"));
|
|
}
|
|
|
|
const ButtonLayout::ButtonMap ButtonLayout::default_data {
|
|
{dig_id, {
|
|
v2f(1.0f, 1.0f),
|
|
v2f(-2.0f, -2.75f),
|
|
}},
|
|
{place_id, {
|
|
v2f(1.0f, 1.0f),
|
|
v2f(-2.0f, -4.25f),
|
|
}},
|
|
{jump_id, {
|
|
v2f(1.0f, 1.0f),
|
|
v2f(-1.0f, -0.5f),
|
|
}},
|
|
{sneak_id, {
|
|
v2f(1.0f, 1.0f),
|
|
v2f(-2.5f, -0.5f),
|
|
}},
|
|
{zoom_id, {
|
|
v2f(1.0f, 1.0f),
|
|
v2f(-0.75f, -3.5f),
|
|
}},
|
|
{aux1_id, {
|
|
v2f(1.0f, 1.0f),
|
|
v2f(-0.75f, -2.0f),
|
|
}},
|
|
{overflow_id, {
|
|
v2f(1.0f, 1.0f),
|
|
v2f(-0.75f, -5.0f),
|
|
}},
|
|
};
|
|
|
|
ButtonLayout ButtonLayout::postProcessLoaded(const ButtonMap &data)
|
|
{
|
|
ButtonLayout layout;
|
|
for (const auto &[id, meta] : data) {
|
|
assert(isButtonValid(id));
|
|
if (isButtonAllowed(id))
|
|
layout.layout.emplace(id, meta);
|
|
else
|
|
layout.preserved_disallowed.emplace(id, meta);
|
|
}
|
|
return layout;
|
|
}
|
|
|
|
ButtonLayout ButtonLayout::loadDefault()
|
|
{
|
|
return postProcessLoaded(default_data);
|
|
}
|
|
|
|
ButtonLayout ButtonLayout::loadFromSettings()
|
|
{
|
|
std::string str = g_settings->get("touch_layout");
|
|
if (!str.empty()) {
|
|
std::istringstream iss(str);
|
|
try {
|
|
ButtonMap data = deserializeJson(iss);
|
|
return postProcessLoaded(data);
|
|
} catch (const Json::Exception &e) {
|
|
warningstream << "Could not parse touchscreen layout: " << e.what() << std::endl;
|
|
}
|
|
}
|
|
|
|
return loadDefault();
|
|
}
|
|
|
|
std::unordered_map<touch_gui_button_id, irr_ptr<video::ITexture>> ButtonLayout::texture_cache;
|
|
|
|
video::ITexture *ButtonLayout::getTexture(touch_gui_button_id btn, ISimpleTextureSource *tsrc)
|
|
{
|
|
if (texture_cache.count(btn) > 0)
|
|
return texture_cache.at(btn).get();
|
|
|
|
video::ITexture *tex = tsrc->getTexture(button_image_names[btn]);
|
|
if (!tex)
|
|
// necessary in the mainmenu
|
|
tex = tsrc->getTexture(porting::path_share + "/textures/base/pack/" +
|
|
button_image_names[btn]);
|
|
irr_ptr<video::ITexture> ptr;
|
|
ptr.grab(tex);
|
|
texture_cache[btn] = ptr;
|
|
return tex;
|
|
}
|
|
|
|
void ButtonLayout::clearTextureCache()
|
|
{
|
|
texture_cache.clear();
|
|
}
|
|
|
|
core::recti ButtonLayout::getRect(touch_gui_button_id btn,
|
|
v2u32 screensize, s32 button_size, ISimpleTextureSource *tsrc)
|
|
{
|
|
const ButtonMeta &meta = layout.at(btn);
|
|
v2s32 pos = meta.getPos(screensize, button_size);
|
|
|
|
v2u32 orig_size = getTexture(btn, tsrc)->getOriginalSize();
|
|
v2s32 size((button_size * orig_size.X) / orig_size.Y, button_size);
|
|
|
|
return core::recti(pos - size / 2, core::dimension2di(size));
|
|
}
|
|
|
|
std::vector<touch_gui_button_id> ButtonLayout::getMissingButtons()
|
|
{
|
|
std::vector<touch_gui_button_id> missing_buttons;
|
|
for (u8 i = 0; i < touch_gui_button_id_END; i++) {
|
|
touch_gui_button_id btn = (touch_gui_button_id)i;
|
|
if (isButtonValid(btn) && isButtonAllowed(btn) && layout.count(btn) == 0)
|
|
missing_buttons.push_back(btn);
|
|
}
|
|
return missing_buttons;
|
|
}
|
|
|
|
void ButtonLayout::serializeJson(std::ostream &os) const
|
|
{
|
|
Json::Value root = Json::objectValue;
|
|
root["version"] = 1;
|
|
root["layout"] = Json::objectValue;
|
|
|
|
for (const auto &list : {layout, preserved_disallowed}) {
|
|
for (const auto &[id, meta] : list) {
|
|
Json::Value button = Json::objectValue;
|
|
button["position_x"] = meta.position.X;
|
|
button["position_y"] = meta.position.Y;
|
|
button["offset_x"] = meta.offset.X;
|
|
button["offset_y"] = meta.offset.Y;
|
|
|
|
root["layout"][button_names[id]] = button;
|
|
}
|
|
}
|
|
|
|
fastWriteJson(root, os);
|
|
}
|
|
|
|
static touch_gui_button_id button_name_to_id(const std::string &name)
|
|
{
|
|
for (u8 i = 0; i < touch_gui_button_id_END; i++) {
|
|
if (name == button_names[i])
|
|
return (touch_gui_button_id)i;
|
|
}
|
|
return touch_gui_button_id_END;
|
|
}
|
|
|
|
ButtonLayout::ButtonMap ButtonLayout::deserializeJson(std::istream &is)
|
|
{
|
|
ButtonMap data;
|
|
|
|
Json::Value root;
|
|
is >> root;
|
|
|
|
u8 version;
|
|
if (root["version"].isUInt())
|
|
version = root["version"].asUInt();
|
|
else
|
|
version = 0;
|
|
|
|
if (!root["layout"].isObject())
|
|
throw Json::RuntimeError("invalid type for layout");
|
|
|
|
Json::Value &obj = root["layout"];
|
|
Json::ValueIterator iter;
|
|
for (iter = obj.begin(); iter != obj.end(); iter++) {
|
|
touch_gui_button_id id = button_name_to_id(iter.name());
|
|
if (!isButtonValid(id))
|
|
throw Json::RuntimeError("invalid button name");
|
|
|
|
Json::Value &value = *iter;
|
|
if (!value.isObject())
|
|
throw Json::RuntimeError("invalid type for button metadata");
|
|
|
|
ButtonMeta meta;
|
|
|
|
if (!value["position_x"].isNumeric() || !value["position_y"].isNumeric())
|
|
throw Json::RuntimeError("invalid type for position_x or position_y in button metadata");
|
|
meta.position.X = value["position_x"].asFloat();
|
|
meta.position.Y = value["position_y"].asFloat();
|
|
|
|
if (!value["offset_x"].isNumeric() || !value["offset_y"].isNumeric())
|
|
throw Json::RuntimeError("invalid type for offset_x or offset_y in button metadata");
|
|
meta.offset.X = value["offset_x"].asFloat();
|
|
meta.offset.Y = value["offset_y"].asFloat();
|
|
|
|
data.emplace(id, meta);
|
|
}
|
|
|
|
if (version < 1) {
|
|
// Version 0 did not have dig/place buttons, so add them in.
|
|
// Otherwise, the missing buttons would cause confusion if the user
|
|
// switches to "touch_interaction_style = buttons_crosshair".
|
|
// This may result in overlapping buttons (could be fixed by resolving
|
|
// collisions in postProcessLoaded).
|
|
data.emplace(dig_id, default_data.at(dig_id));
|
|
data.emplace(place_id, default_data.at(place_id));
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
void layout_button_grid(v2u32 screensize, ISimpleTextureSource *tsrc,
|
|
const std::vector<touch_gui_button_id> &buttons,
|
|
// pos refers to the center of the button
|
|
const std::function<void(touch_gui_button_id btn, v2s32 pos, core::recti rect)> &callback)
|
|
{
|
|
s32 cols = 4;
|
|
s32 rows = 3;
|
|
f32 screen_aspect = (f32)screensize.X / (f32)screensize.Y;
|
|
while ((s32)buttons.size() > cols * rows) {
|
|
f32 aspect = (f32)cols / (f32)rows;
|
|
if (aspect > screen_aspect)
|
|
rows++;
|
|
else
|
|
cols++;
|
|
}
|
|
|
|
s32 button_size = ButtonLayout::getButtonSize(screensize);
|
|
v2s32 spacing(screensize.X / (cols + 1), screensize.Y / (rows + 1));
|
|
v2s32 pos(spacing);
|
|
|
|
for (touch_gui_button_id btn : buttons) {
|
|
v2u32 orig_size = ButtonLayout::getTexture(btn, tsrc)->getOriginalSize();
|
|
v2s32 size((button_size * orig_size.X) / orig_size.Y, button_size);
|
|
|
|
core::recti rect(pos - size / 2, core::dimension2di(size));
|
|
|
|
if (rect.LowerRightCorner.X > (s32)screensize.X) {
|
|
pos.X = spacing.X;
|
|
pos.Y += spacing.Y;
|
|
rect = core::recti(pos - size / 2, core::dimension2di(size));
|
|
}
|
|
|
|
callback(btn, pos, rect);
|
|
|
|
pos.X += spacing.X;
|
|
}
|
|
}
|
|
|
|
void make_button_grid_title(gui::IGUIStaticText *text, touch_gui_button_id btn, v2s32 pos, core::recti rect)
|
|
{
|
|
std::wstring str = wstrgettext(button_titles[btn]);
|
|
text->setText(str.c_str());
|
|
gui::IGUIFont *font = text->getActiveFont();
|
|
core::dimension2du dim = font->getDimension(str.c_str());
|
|
dim = core::dimension2du(dim.Width * 1.25f, dim.Height * 1.25f); // avoid clipping
|
|
text->setRelativePosition(core::recti(pos.X - dim.Width / 2, rect.LowerRightCorner.Y,
|
|
pos.X + dim.Width / 2, rect.LowerRightCorner.Y + dim.Height));
|
|
text->setTextAlignment(gui::EGUIA_CENTER, gui::EGUIA_UPPERLEFT);
|
|
}
|