2023-06-20 04:33:09 +00:00
|
|
|
#include "StarGraphicsMenu.hpp"
|
|
|
|
#include "StarRoot.hpp"
|
|
|
|
#include "StarAssets.hpp"
|
|
|
|
#include "StarConfiguration.hpp"
|
|
|
|
#include "StarGuiReader.hpp"
|
|
|
|
#include "StarListWidget.hpp"
|
|
|
|
#include "StarLabelWidget.hpp"
|
|
|
|
#include "StarSliderBar.hpp"
|
|
|
|
#include "StarButtonWidget.hpp"
|
|
|
|
#include "StarOrderedSet.hpp"
|
|
|
|
#include "StarJsonExtra.hpp"
|
|
|
|
|
|
|
|
namespace Star {
|
|
|
|
|
|
|
|
GraphicsMenu::GraphicsMenu() {
|
|
|
|
GuiReader reader;
|
|
|
|
reader.registerCallback("cancel",
|
|
|
|
[&](Widget*) {
|
|
|
|
dismiss();
|
|
|
|
});
|
|
|
|
reader.registerCallback("accept",
|
|
|
|
[&](Widget*) {
|
|
|
|
apply();
|
|
|
|
applyWindowSettings();
|
|
|
|
});
|
|
|
|
reader.registerCallback("resSlider", [=](Widget*) {
|
|
|
|
Vec2U res = m_resList[fetchChild<SliderBarWidget>("resSlider")->val()];
|
|
|
|
m_localChanges.set("fullscreenResolution", jsonFromVec2U(res));
|
|
|
|
syncGui();
|
|
|
|
});
|
|
|
|
reader.registerCallback("zoomSlider", [=](Widget*) {
|
2023-11-28 17:04:45 +00:00
|
|
|
auto zoomSlider = fetchChild<SliderBarWidget>("zoomSlider");
|
|
|
|
m_localChanges.set("zoomLevel", m_zoomList[zoomSlider->val()]);
|
|
|
|
Root::singleton().configuration()->set("zoomLevel", m_zoomList[zoomSlider->val()]);
|
|
|
|
syncGui();
|
|
|
|
});
|
|
|
|
reader.registerCallback("cameraSpeedSlider", [=](Widget*) {
|
|
|
|
auto cameraSpeedSlider = fetchChild<SliderBarWidget>("cameraSpeedSlider");
|
|
|
|
m_localChanges.set("cameraSpeedFactor", m_cameraSpeedList[cameraSpeedSlider->val()]);
|
|
|
|
Root::singleton().configuration()->set("cameraSpeedFactor", m_cameraSpeedList[cameraSpeedSlider->val()]);
|
2023-06-20 04:33:09 +00:00
|
|
|
syncGui();
|
|
|
|
});
|
|
|
|
reader.registerCallback("speechBubbleCheckbox", [=](Widget*) {
|
|
|
|
auto button = fetchChild<ButtonWidget>("speechBubbleCheckbox");
|
|
|
|
m_localChanges.set("speechBubbles", button->isChecked());
|
|
|
|
Root::singleton().configuration()->set("speechBubbles", button->isChecked());
|
|
|
|
syncGui();
|
|
|
|
});
|
|
|
|
reader.registerCallback("interactiveHighlightCheckbox", [=](Widget*) {
|
|
|
|
auto button = fetchChild<ButtonWidget>("interactiveHighlightCheckbox");
|
|
|
|
m_localChanges.set("interactiveHighlight", button->isChecked());
|
|
|
|
Root::singleton().configuration()->set("interactiveHighlight", button->isChecked());
|
|
|
|
syncGui();
|
|
|
|
});
|
|
|
|
reader.registerCallback("fullscreenCheckbox", [=](Widget*) {
|
|
|
|
bool checked = fetchChild<ButtonWidget>("fullscreenCheckbox")->isChecked();
|
|
|
|
m_localChanges.set("fullscreen", checked);
|
|
|
|
if (checked)
|
|
|
|
m_localChanges.set("borderless", !checked);
|
|
|
|
syncGui();
|
|
|
|
});
|
|
|
|
reader.registerCallback("borderlessCheckbox", [=](Widget*) {
|
|
|
|
bool checked = fetchChild<ButtonWidget>("borderlessCheckbox")->isChecked();
|
|
|
|
m_localChanges.set("borderless", checked);
|
|
|
|
if (checked)
|
|
|
|
m_localChanges.set("fullscreen", !checked);
|
|
|
|
syncGui();
|
|
|
|
});
|
|
|
|
reader.registerCallback("textureLimitCheckbox", [=](Widget*) {
|
|
|
|
m_localChanges.set("limitTextureAtlasSize", fetchChild<ButtonWidget>("textureLimitCheckbox")->isChecked());
|
|
|
|
syncGui();
|
|
|
|
});
|
|
|
|
reader.registerCallback("multiTextureCheckbox", [=](Widget*) {
|
|
|
|
m_localChanges.set("useMultiTexturing", fetchChild<ButtonWidget>("multiTextureCheckbox")->isChecked());
|
|
|
|
syncGui();
|
|
|
|
});
|
2024-04-08 04:22:22 +00:00
|
|
|
reader.registerCallback("antiAliasingCheckbox", [=](Widget*) {
|
|
|
|
bool checked = fetchChild<ButtonWidget>("antiAliasingCheckbox")->isChecked();
|
|
|
|
m_localChanges.set("antiAliasing", checked);
|
|
|
|
Root::singleton().configuration()->set("antiAliasing", checked);
|
|
|
|
syncGui();
|
|
|
|
});
|
2023-06-20 04:33:09 +00:00
|
|
|
reader.registerCallback("monochromeCheckbox", [=](Widget*) {
|
|
|
|
bool checked = fetchChild<ButtonWidget>("monochromeCheckbox")->isChecked();
|
|
|
|
m_localChanges.set("monochromeLighting", checked);
|
|
|
|
Root::singleton().configuration()->set("monochromeLighting", checked);
|
|
|
|
syncGui();
|
|
|
|
});
|
|
|
|
|
|
|
|
auto assets = Root::singleton().assets();
|
|
|
|
|
|
|
|
Json paneLayout = assets->json("/interface/windowconfig/graphicsmenu.config:paneLayout");
|
|
|
|
|
|
|
|
m_resList = jsonToVec2UList(assets->json("/interface/windowconfig/graphicsmenu.config:resolutionList"));
|
|
|
|
m_zoomList = jsonToFloatList(assets->json("/interface/windowconfig/graphicsmenu.config:zoomList"));
|
2023-11-28 17:04:45 +00:00
|
|
|
m_cameraSpeedList = jsonToFloatList(assets->json("/interface/windowconfig/graphicsmenu.config:cameraSpeedList"));
|
2023-06-20 04:33:09 +00:00
|
|
|
|
|
|
|
reader.construct(paneLayout, this);
|
|
|
|
|
|
|
|
fetchChild<SliderBarWidget>("resSlider")->setRange(0, m_resList.size() - 1, 1);
|
|
|
|
fetchChild<SliderBarWidget>("zoomSlider")->setRange(0, m_zoomList.size() - 1, 1);
|
2023-11-28 17:04:45 +00:00
|
|
|
fetchChild<SliderBarWidget>("cameraSpeedSlider")->setRange(0, m_cameraSpeedList.size() - 1, 1);
|
2023-06-20 04:33:09 +00:00
|
|
|
|
|
|
|
initConfig();
|
|
|
|
syncGui();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsMenu::show() {
|
|
|
|
Pane::show();
|
|
|
|
initConfig();
|
|
|
|
syncGui();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsMenu::dismissed() {
|
|
|
|
Pane::dismissed();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsMenu::toggleFullscreen() {
|
|
|
|
bool fullscreen = m_localChanges.get("fullscreen").toBool();
|
|
|
|
bool borderless = m_localChanges.get("borderless").toBool();
|
|
|
|
|
|
|
|
m_localChanges.set("fullscreen", !(fullscreen || borderless));
|
|
|
|
Root::singleton().configuration()->set("fullscreen", !(fullscreen || borderless));
|
|
|
|
|
|
|
|
m_localChanges.set("borderless", false);
|
|
|
|
Root::singleton().configuration()->set("borderless", false);
|
|
|
|
|
|
|
|
applyWindowSettings();
|
|
|
|
syncGui();
|
|
|
|
}
|
|
|
|
|
|
|
|
StringList const GraphicsMenu::ConfigKeys = {
|
|
|
|
"fullscreenResolution",
|
|
|
|
"zoomLevel",
|
2023-11-28 17:04:45 +00:00
|
|
|
"cameraSpeedFactor",
|
2023-06-20 04:33:09 +00:00
|
|
|
"speechBubbles",
|
|
|
|
"interactiveHighlight",
|
|
|
|
"fullscreen",
|
|
|
|
"borderless",
|
|
|
|
"limitTextureAtlasSize",
|
|
|
|
"useMultiTexturing",
|
2024-04-08 04:22:22 +00:00
|
|
|
"antiAliasing",
|
2023-06-20 04:33:09 +00:00
|
|
|
"monochromeLighting"
|
|
|
|
};
|
|
|
|
|
|
|
|
void GraphicsMenu::initConfig() {
|
|
|
|
auto configuration = Root::singleton().configuration();
|
|
|
|
|
|
|
|
for (auto key : ConfigKeys) {
|
|
|
|
m_localChanges.set(key, configuration->get(key));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsMenu::syncGui() {
|
|
|
|
Vec2U res = jsonToVec2U(m_localChanges.get("fullscreenResolution"));
|
|
|
|
auto resSlider = fetchChild<SliderBarWidget>("resSlider");
|
|
|
|
auto resIt = std::lower_bound(m_resList.begin(), m_resList.end(), res, [&](Vec2U const& a, Vec2U const& b) {
|
|
|
|
return a[0] * a[1] < b[0] * b[1]; // sort by number of pixels
|
|
|
|
});
|
|
|
|
if (resIt != m_resList.end()) {
|
|
|
|
size_t resIndex = resIt - m_resList.begin();
|
|
|
|
resIndex = std::min(resIndex, m_resList.size() - 1);
|
|
|
|
resSlider->setVal(resIndex, false);
|
|
|
|
} else {
|
|
|
|
resSlider->setVal(m_resList.size() - 1);
|
|
|
|
}
|
2023-06-27 10:23:44 +00:00
|
|
|
fetchChild<LabelWidget>("resValueLabel")->setText(strf("{}x{}", res[0], res[1]));
|
2023-06-20 04:33:09 +00:00
|
|
|
|
|
|
|
auto zoomSlider = fetchChild<SliderBarWidget>("zoomSlider");
|
|
|
|
auto zoomIt = std::lower_bound(m_zoomList.begin(), m_zoomList.end(), m_localChanges.get("zoomLevel").toFloat());
|
|
|
|
if (zoomIt != m_zoomList.end()) {
|
|
|
|
size_t zoomIndex = zoomIt - m_zoomList.begin();
|
2023-11-28 17:04:45 +00:00
|
|
|
zoomIndex = std::min(zoomIndex, m_zoomList.size() - 1);
|
2023-06-20 04:33:09 +00:00
|
|
|
zoomSlider->setVal(zoomIndex, false);
|
|
|
|
} else {
|
|
|
|
zoomSlider->setVal(m_zoomList.size() - 1);
|
|
|
|
}
|
2023-06-28 21:05:01 +00:00
|
|
|
fetchChild<LabelWidget>("zoomValueLabel")->setText(strf("{}x", m_localChanges.get("zoomLevel").toFloat()));
|
2023-06-20 04:33:09 +00:00
|
|
|
|
2023-11-28 17:04:45 +00:00
|
|
|
auto cameraSpeedSlider = fetchChild<SliderBarWidget>("cameraSpeedSlider");
|
|
|
|
auto speedIt = std::lower_bound(m_cameraSpeedList.begin(), m_cameraSpeedList.end(), m_localChanges.get("cameraSpeedFactor").toFloat());
|
|
|
|
if (speedIt != m_cameraSpeedList.end()) {
|
|
|
|
size_t speedIndex = speedIt - m_cameraSpeedList.begin();
|
|
|
|
speedIndex = std::min(speedIndex, m_cameraSpeedList.size() - 1);
|
|
|
|
cameraSpeedSlider->setVal(speedIndex, false);
|
|
|
|
} else {
|
|
|
|
cameraSpeedSlider->setVal(m_cameraSpeedList.size() - 1);
|
|
|
|
}
|
|
|
|
fetchChild<LabelWidget>("cameraSpeedValueLabel")->setText(strf("{}x", m_localChanges.get("cameraSpeedFactor").toFloat()));
|
2023-06-20 04:33:09 +00:00
|
|
|
|
|
|
|
fetchChild<ButtonWidget>("speechBubbleCheckbox")->setChecked(m_localChanges.get("speechBubbles").toBool());
|
|
|
|
fetchChild<ButtonWidget>("interactiveHighlightCheckbox")->setChecked(m_localChanges.get("interactiveHighlight").toBool());
|
|
|
|
fetchChild<ButtonWidget>("fullscreenCheckbox")->setChecked(m_localChanges.get("fullscreen").toBool());
|
|
|
|
fetchChild<ButtonWidget>("borderlessCheckbox")->setChecked(m_localChanges.get("borderless").toBool());
|
|
|
|
fetchChild<ButtonWidget>("textureLimitCheckbox")->setChecked(m_localChanges.get("limitTextureAtlasSize").toBool());
|
|
|
|
fetchChild<ButtonWidget>("multiTextureCheckbox")->setChecked(m_localChanges.get("useMultiTexturing").optBool().value(true));
|
2024-04-08 04:22:22 +00:00
|
|
|
fetchChild<ButtonWidget>("antiAliasingCheckbox")->setChecked(m_localChanges.get("antiAliasing").toBool());
|
2023-06-20 04:33:09 +00:00
|
|
|
fetchChild<ButtonWidget>("monochromeCheckbox")->setChecked(m_localChanges.get("monochromeLighting").toBool());
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsMenu::apply() {
|
|
|
|
auto configuration = Root::singleton().configuration();
|
|
|
|
for (auto p : m_localChanges) {
|
|
|
|
configuration->set(p.first, p.second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsMenu::applyWindowSettings() {
|
|
|
|
auto configuration = Root::singleton().configuration();
|
|
|
|
auto appController = GuiContext::singleton().applicationController();
|
|
|
|
if (configuration->get("fullscreen").toBool())
|
|
|
|
appController->setFullscreenWindow(jsonToVec2U(configuration->get("fullscreenResolution")));
|
|
|
|
else if (configuration->get("borderless").toBool())
|
|
|
|
appController->setBorderlessWindow();
|
|
|
|
else if (configuration->get("maximized").toBool())
|
|
|
|
appController->setMaximizedWindow();
|
|
|
|
else
|
|
|
|
appController->setNormalWindow(jsonToVec2U(configuration->get("windowedResolution")));
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|