scriptable chat

This commit is contained in:
Kae 2024-11-24 12:51:55 +11:00
parent d4c976bcb3
commit 56c99c086f
10 changed files with 158 additions and 71 deletions

View File

@ -684,8 +684,9 @@ void ClientApplication::changeState(MainAppState newState) {
m_mainInterface = make_shared<MainInterface>(m_universeClient, m_worldPainter, m_cinematicOverlay); m_mainInterface = make_shared<MainInterface>(m_universeClient, m_worldPainter, m_cinematicOverlay);
m_universeClient->setLuaCallbacks("interface", LuaBindings::makeInterfaceCallbacks(m_mainInterface.get())); m_universeClient->setLuaCallbacks("interface", LuaBindings::makeInterfaceCallbacks(m_mainInterface.get()));
m_universeClient->setLuaCallbacks("chat", LuaBindings::makeChatCallbacks(m_mainInterface.get(), m_universeClient.get())); m_universeClient->setLuaCallbacks("chat", LuaBindings::makeChatCallbacks(m_mainInterface.get(), m_universeClient.get()));
m_universeClient->startLua(); m_mainInterface->displayDefaultPanes();
m_universeClient->startLua();
m_mainMixer->setWorldPainter(m_worldPainter); m_mainMixer->setWorldPainter(m_worldPainter);
if (auto renderer = Application::renderer()) { if (auto renderer = Application::renderer()) {

View File

@ -14,7 +14,7 @@
namespace Star { namespace Star {
BaseScriptPane::BaseScriptPane(Json config) : Pane(), m_rawConfig(config) { BaseScriptPane::BaseScriptPane(Json config, bool construct) : Pane(), m_rawConfig(config) {
auto& root = Root::singleton(); auto& root = Root::singleton();
auto assets = root.assets(); auto assets = root.assets();
@ -35,15 +35,8 @@ BaseScriptPane::BaseScriptPane(Json config) : Pane(), m_rawConfig(config) {
}); });
} }
m_reader->construct(assets->fetchJson(m_config.get("gui")), this); if (construct)
this->construct(assets->fetchJson(m_config.get("gui")));
for (auto pair : m_config.getObject("canvasClickCallbacks", {}))
m_canvasClickCallbacks.set(findChild<CanvasWidget>(pair.first), pair.second.toString());
for (auto pair : m_config.getObject("canvasKeyCallbacks", {}))
m_canvasKeyCallbacks.set(findChild<CanvasWidget>(pair.first), pair.second.toString());
m_script.setScripts(jsonToStringList(m_config.get("scripts", JsonArray())));
m_script.setUpdateDelta(m_config.getUInt("scriptDelta", 1));
m_callbacksAdded = false; m_callbacksAdded = false;
} }
@ -140,4 +133,16 @@ GuiReaderPtr BaseScriptPane::reader() {
return m_reader; return m_reader;
} }
void BaseScriptPane::construct(Json config) {
m_reader->construct(config, this);
for (auto pair : m_config.getObject("canvasClickCallbacks", {}))
m_canvasClickCallbacks.set(findChild<CanvasWidget>(pair.first), pair.second.toString());
for (auto pair : m_config.getObject("canvasKeyCallbacks", {}))
m_canvasKeyCallbacks.set(findChild<CanvasWidget>(pair.first), pair.second.toString());
m_script.setScripts(jsonToStringList(m_config.get("scripts", JsonArray())));
m_script.setUpdateDelta(m_config.getUInt("scriptDelta", 1));
}
} }

View File

@ -15,7 +15,7 @@ STAR_CLASS(BaseScriptPane);
class BaseScriptPane : public Pane { class BaseScriptPane : public Pane {
public: public:
BaseScriptPane(Json config); BaseScriptPane(Json config, bool construct = true);
virtual void show() override; virtual void show() override;
void displayed() override; void displayed() override;
@ -34,6 +34,8 @@ public:
Maybe<String> cursorOverride(Vec2I const& screenPosition) override; Maybe<String> cursorOverride(Vec2I const& screenPosition) override;
protected: protected:
virtual GuiReaderPtr reader() override; virtual GuiReaderPtr reader() override;
void construct(Json config);
Json m_config; Json m_config;
Json m_rawConfig; Json m_rawConfig;
@ -45,7 +47,7 @@ protected:
bool m_interactive; bool m_interactive;
bool m_callbacksAdded; bool m_callbacksAdded;
LuaUpdatableComponent<LuaBaseComponent> m_script; mutable LuaUpdatableComponent<LuaBaseComponent> m_script;
}; };
} }

View File

@ -13,14 +13,25 @@
#include "StarPlayerStorage.hpp" #include "StarPlayerStorage.hpp"
#include "StarTeamClient.hpp" #include "StarTeamClient.hpp"
#include "StarPlayer.hpp"
#include "StarConfigLuaBindings.hpp"
#include "StarPlayerLuaBindings.hpp"
#include "StarStatusControllerLuaBindings.hpp"
#include "StarCelestialLuaBindings.hpp"
#include "StarLuaGameConverters.hpp"
namespace Star { namespace Star {
Chat::Chat(UniverseClientPtr client) : m_client(client) { Chat::Chat(UniverseClientPtr client, Json const& baseConfig) : BaseScriptPane(baseConfig, false) {
m_client = client;
m_scripted = baseConfig.get("scripts", Json()).isType(Json::Type::Array);
m_script.setLuaRoot(m_client->luaRoot());
m_script.addCallbacks("world", LuaBindings::makeWorldCallbacks((World*)m_client->worldClient().get()));
m_chatPrevIndex = 0; m_chatPrevIndex = 0;
m_historyOffset = 0; m_historyOffset = 0;
auto assets = Root::singleton().assets(); auto assets = Root::singleton().assets();
auto config = assets->json("/interface/chat/chat.config:config"); auto config = baseConfig.get("config");
m_timeChatLastActive = Time::monotonicMilliseconds(); m_timeChatLastActive = Time::monotonicMilliseconds();
m_chatTextStyle = config.get("textStyle"); m_chatTextStyle = config.get("textStyle");
m_chatTextStyle.lineSpacing = config.get("lineHeight").toFloat(); m_chatTextStyle.lineSpacing = config.get("lineHeight").toFloat();
@ -45,15 +56,13 @@ Chat::Chat(UniverseClientPtr client) : m_client(client) {
m_colorCodes[MessageContext::CommandResult] = config.query("colors.commandResult").toString(); m_colorCodes[MessageContext::CommandResult] = config.query("colors.commandResult").toString();
m_colorCodes[MessageContext::RadioMessage] = config.query("colors.radioMessage").toString(); m_colorCodes[MessageContext::RadioMessage] = config.query("colors.radioMessage").toString();
m_colorCodes[MessageContext::World] = config.query("colors.world").toString(); m_colorCodes[MessageContext::World] = config.query("colors.world").toString();
if (!m_scripted) {
m_reader->registerCallback("textBox", [=](Widget*) { startChat(); });
m_reader->registerCallback("upButton", [=](Widget*) { scrollUp(); });
m_reader->registerCallback("downButton", [=](Widget*) { scrollDown(); });
m_reader->registerCallback("bottomButton", [=](Widget*) { scrollBottom(); });
GuiReader reader; m_reader->registerCallback("filterGroup", [=](Widget* widget) {
reader.registerCallback("textBox", [=](Widget*) { startChat(); });
reader.registerCallback("upButton", [=](Widget*) { scrollUp(); });
reader.registerCallback("downButton", [=](Widget*) { scrollDown(); });
reader.registerCallback("bottomButton", [=](Widget*) { scrollBottom(); });
reader.registerCallback("filterGroup", [=](Widget* widget) {
Json data = as<ButtonWidget>(widget)->data(); Json data = as<ButtonWidget>(widget)->data();
auto filter = data.getArray("filter", {}); auto filter = data.getArray("filter", {});
m_modeFilter.clear(); m_modeFilter.clear();
@ -62,41 +71,49 @@ Chat::Chat(UniverseClientPtr client) : m_client(client) {
m_sendMode = ChatSendModeNames.getLeft(data.getString("sendMode", "Broadcast")); m_sendMode = ChatSendModeNames.getLeft(data.getString("sendMode", "Broadcast"));
m_historyOffset = 0; m_historyOffset = 0;
}); });
}
construct(baseConfig.get("gui"));
m_sendMode = ChatSendMode::Broadcast; m_sendMode = ChatSendMode::Broadcast;
reader.construct(assets->json("/interface/chat/chat.config:gui"), this);
m_textBox = fetchChild<TextBoxWidget>("textBox");
m_say = fetchChild<LabelWidget>("say");
m_chatLog = fetchChild<CanvasWidget>("chatLog"); m_chatLog = fetchChild<CanvasWidget>("chatLog");
if (auto logPadding = config.optQuery("padding")) {
m_chatLogPadding = jsonToVec2I(logPadding.get());
m_chatLog->setSize(m_chatLog->size() + m_chatLogPadding * 2);
m_chatLog->setPosition(m_chatLog->position() - m_chatLogPadding);
}
else
m_chatLogPadding = Vec2I();
m_bottomButton = fetchChild<ButtonWidget>("bottomButton"); m_bottomButton = fetchChild<ButtonWidget>("bottomButton");
m_upButton = fetchChild<ButtonWidget>("upButton"); m_upButton = fetchChild<ButtonWidget>("upButton");
m_textBox = fetchChild<TextBoxWidget>("textBox");
m_say = fetchChild<LabelWidget>("say");
if (!m_scripted) {
if (auto logPadding = config.optQuery("padding")) {
m_chatLogPadding = jsonToVec2I(logPadding.get());
m_chatLog->setSize(m_chatLog->size() + m_chatLogPadding * 2);
m_chatLog->setPosition(m_chatLog->position() - m_chatLogPadding);
} else
m_chatLogPadding = Vec2I();
m_chatHistory.appendAll(m_client->playerStorage()->getMetadata("chatHistory").opt().apply(jsonToStringList).value()); m_chatHistory.appendAll(m_client->playerStorage()->getMetadata("chatHistory").opt().apply(jsonToStringList).value());
} else {
m_script.addCallbacks("player", LuaBindings::makePlayerCallbacks(m_client->mainPlayer().get()));
m_script.addCallbacks("status", LuaBindings::makeStatusControllerCallbacks(m_client->mainPlayer()->statusController()));
m_script.addCallbacks("celestial", LuaBindings::makeCelestialCallbacks(m_client.get()));
}
show(); show();
updateBottomButton(); //if (!m_scripted) {
//updateBottomButton();
m_background = fetchChild<ImageStretchWidget>("background"); m_background = fetchChild<ImageStretchWidget>("background");
m_defaultHeight = m_background->size()[1]; m_defaultHeight = m_background->size()[1];
m_expanded = false; m_expanded = false;
updateSize(); updateSize();
//}
} }
void Chat::update(float dt) { void Chat::update(float dt) {
Pane::update(dt); Pane::update(dt);
if (m_scripted)
return;
auto team = m_client->teamClient()->currentTeam(); auto team = m_client->teamClient()->currentTeam();
for (auto button : fetchChild<ButtonGroup>("filterGroup")->buttons()) { for (auto button : fetchChild<ButtonGroup>("filterGroup")->buttons()) {
auto mode = ChatSendModeNames.getLeft(button->data().getString("sendMode", "Broadcast")); auto mode = ChatSendModeNames.getLeft(button->data().getString("sendMode", "Broadcast"));
@ -108,40 +125,64 @@ void Chat::update(float dt) {
} }
void Chat::startChat() { void Chat::startChat() {
show(); if (m_scripted)
m_textBox->focus(); m_script.invoke("startChat");
else {
show();
m_textBox->focus();
}
} }
void Chat::startCommand() { void Chat::startCommand() {
show(); if (m_scripted)
m_textBox->setText("/"); m_script.invoke("startCommand");
m_textBox->focus(); else {
show();
m_textBox->setText("/");
m_textBox->focus();
}
} }
bool Chat::hasFocus() const { bool Chat::hasFocus() const {
if (m_scripted)
return m_script.invoke<bool>("hasFocus").value();
return m_textBox->hasFocus(); return m_textBox->hasFocus();
} }
void Chat::stopChat() { void Chat::stopChat() {
m_textBox->setText(""); if (m_scripted)
m_textBox->blur(); m_script.invoke("stopChat");
m_timeChatLastActive = Time::monotonicMilliseconds(); else {
m_textBox->setText("");
m_textBox->blur();
m_timeChatLastActive = Time::monotonicMilliseconds();
}
} }
String Chat::currentChat() const { String Chat::currentChat() const {
if (m_scripted)
return m_script.invoke<String>("currentChat").value();
return m_textBox->getText(); return m_textBox->getText();
} }
bool Chat::setCurrentChat(String const& chat, bool moveCursor) { bool Chat::setCurrentChat(String const& chat, bool moveCursor) {
if (m_scripted)
return m_script.invoke<bool>("setCurrentChat").value();
return m_textBox->setText(chat, true, moveCursor); return m_textBox->setText(chat, true, moveCursor);
} }
void Chat::clearCurrentChat() { void Chat::clearCurrentChat() {
m_textBox->setText(""); if (m_scripted)
m_chatPrevIndex = 0; m_script.invoke("clearCurrentChat");
else {
m_textBox->setText("");
m_chatPrevIndex = 0;
}
} }
ChatSendMode Chat::sendMode() const { ChatSendMode Chat::sendMode() const {
if (m_scripted)
return ChatSendModeNames.getLeft(m_script.invoke<String>("sendMode").value());
return m_sendMode; return m_sendMode;
} }
@ -171,6 +212,13 @@ void Chat::addMessages(List<ChatReceivedMessage> const& messages, bool showPane)
if (messages.empty()) if (messages.empty())
return; return;
if (m_scripted) {
m_script.invoke("addMessages", messages.transformed([](ChatReceivedMessage const& message) {
return message.toJson();
}), showPane);
return;
}
GuiContext& guiContext = GuiContext::singleton(); GuiContext& guiContext = GuiContext::singleton();
for (auto const& message : messages) { for (auto const& message : messages) {
@ -209,17 +257,22 @@ void Chat::addMessages(List<ChatReceivedMessage> const& messages, bool showPane)
} }
void Chat::addHistory(String const& chat) { void Chat::addHistory(String const& chat) {
if (m_chatHistory.size() > 0 && m_chatHistory.get(0).equals(chat)) if (m_scripted)
m_script.invoke("addHistory", chat);
else if (m_chatHistory.size() > 0 && m_chatHistory.get(0).equals(chat))
return; return;
else {
m_chatHistory.prepend(chat); m_chatHistory.prepend(chat);
m_chatHistory.resize(std::min((unsigned)m_chatHistory.size(), m_chatHistoryLimit)); m_chatHistory.resize(std::min((unsigned)m_chatHistory.size(), m_chatHistoryLimit));
m_timeChatLastActive = Time::monotonicMilliseconds(); m_timeChatLastActive = Time::monotonicMilliseconds();
m_client->playerStorage()->setMetadata("chatHistory", JsonArray::from(m_chatHistory)); m_client->playerStorage()->setMetadata("chatHistory", JsonArray::from(m_chatHistory));
}
} }
void Chat::clear(size_t count) { void Chat::clear(size_t count) {
if (count > m_receivedMessages.size()) if (m_scripted)
m_script.invoke("clear", count);
else if (count > m_receivedMessages.size())
m_receivedMessages.clear(); m_receivedMessages.clear();
else else
m_receivedMessages.erase(m_receivedMessages.begin(), m_receivedMessages.begin() + count); m_receivedMessages.erase(m_receivedMessages.begin(), m_receivedMessages.begin() + count);
@ -227,6 +280,8 @@ void Chat::clear(size_t count) {
void Chat::renderImpl() { void Chat::renderImpl() {
Pane::renderImpl(); Pane::renderImpl();
if (m_scripted)
return;
if (m_textBox->hasFocus()) if (m_textBox->hasFocus())
m_timeChatLastActive = Time::monotonicMilliseconds(); m_timeChatLastActive = Time::monotonicMilliseconds();
Vec4B fade = {255, 255, 255, 255}; Vec4B fade = {255, 255, 255, 255};
@ -303,6 +358,9 @@ void Chat::hide() {
} }
float Chat::visible() const { float Chat::visible() const {
if (m_scripted)
return m_script.invoke<float>("visible").value(1.0f);
double difference = (Time::monotonicMilliseconds() - m_timeChatLastActive) / 1000.0; double difference = (Time::monotonicMilliseconds() - m_timeChatLastActive) / 1000.0;
if (difference < m_chatVisTime) if (difference < m_chatVisTime)
return 1; return 1;
@ -310,7 +368,7 @@ float Chat::visible() const {
} }
bool Chat::sendEvent(InputEvent const& event) { bool Chat::sendEvent(InputEvent const& event) {
if (active()) { if (!m_scripted && active()) {
if (hasFocus()) { if (hasFocus()) {
if (event.is<KeyDownEvent>()) { if (event.is<KeyDownEvent>()) {
auto actions = context()->actions(event); auto actions = context()->actions(event);

View File

@ -1,6 +1,6 @@
#pragma once #pragma once
#include "StarPane.hpp" #include "StarBaseScriptPane.hpp"
#include "StarChatTypes.hpp" #include "StarChatTypes.hpp"
namespace Star { namespace Star {
@ -13,9 +13,9 @@ STAR_CLASS(ImageStretchWidget);
STAR_CLASS(CanvasWidget); STAR_CLASS(CanvasWidget);
STAR_CLASS(Chat); STAR_CLASS(Chat);
class Chat : public Pane { class Chat : public BaseScriptPane {
public: public:
Chat(UniverseClientPtr client); Chat(UniverseClientPtr client, Json const&);
void startChat(); void startChat();
void startCommand(); void startCommand();
@ -57,6 +57,7 @@ private:
void updateBottomButton(); void updateBottomButton();
UniverseClientPtr m_client; UniverseClientPtr m_client;
bool m_scripted;
TextBoxWidgetPtr m_textBox; TextBoxWidgetPtr m_textBox;
LabelWidgetPtr m_say; LabelWidgetPtr m_say;

View File

@ -141,7 +141,7 @@ MainInterface::MainInterface(UniverseClientPtr client, WorldPainterPtr painter,
m_collections = make_shared<ScriptPane>(m_client, "/interface/scripted/collections/collectionsgui.config"); m_collections = make_shared<ScriptPane>(m_client, "/interface/scripted/collections/collectionsgui.config");
m_paneManager.registerPane(MainInterfacePanes::Collections, PaneLayer::Window, m_collections); m_paneManager.registerPane(MainInterfacePanes::Collections, PaneLayer::Window, m_collections);
m_chat = make_shared<Chat>(m_client); m_chat = make_shared<Chat>(m_client, Root::singleton().assets()->json("/interface/chat/chat.config"));
m_paneManager.registerPane(MainInterfacePanes::Chat, PaneLayer::Hud, m_chat); m_paneManager.registerPane(MainInterfacePanes::Chat, PaneLayer::Hud, m_chat);
m_clientCommandProcessor = make_shared<ClientCommandProcessor>(m_client, m_cinematicOverlay, &m_paneManager, m_config->macroCommands); m_clientCommandProcessor = make_shared<ClientCommandProcessor>(m_client, m_cinematicOverlay, &m_paneManager, m_config->macroCommands);
@ -171,11 +171,6 @@ MainInterface::MainInterface(UniverseClientPtr client, WorldPainterPtr painter,
m_nameplatePainter = make_shared<NameplatePainter>(); m_nameplatePainter = make_shared<NameplatePainter>();
m_questIndicatorPainter = make_shared<QuestIndicatorPainter>(m_client); m_questIndicatorPainter = make_shared<QuestIndicatorPainter>(m_client);
m_chatBubbleManager = make_shared<ChatBubbleManager>(); m_chatBubbleManager = make_shared<ChatBubbleManager>();
m_paneManager.displayRegisteredPane(MainInterfacePanes::ActionBar);
m_paneManager.displayRegisteredPane(MainInterfacePanes::Chat);
m_paneManager.displayRegisteredPane(MainInterfacePanes::TeamBar);
m_paneManager.displayRegisteredPane(MainInterfacePanes::StatusPane);
} }
MainInterface::~MainInterface() { MainInterface::~MainInterface() {
@ -997,6 +992,13 @@ void MainInterface::reviveScriptPanes(List<ScriptPaneInfo>& panes) {
} }
} }
void MainInterface::displayDefaultPanes() {
m_paneManager.displayRegisteredPane(MainInterfacePanes::ActionBar);
m_paneManager.displayRegisteredPane(MainInterfacePanes::Chat);
m_paneManager.displayRegisteredPane(MainInterfacePanes::TeamBar);
m_paneManager.displayRegisteredPane(MainInterfacePanes::StatusPane);
}
PanePtr MainInterface::createEscapeDialog() { PanePtr MainInterface::createEscapeDialog() {
auto assets = Root::singleton().assets(); auto assets = Root::singleton().assets();

View File

@ -132,6 +132,7 @@ public:
void takeScriptPanes(List<ScriptPaneInfo>& out); void takeScriptPanes(List<ScriptPaneInfo>& out);
void reviveScriptPanes(List<ScriptPaneInfo>& panes); void reviveScriptPanes(List<ScriptPaneInfo>& panes);
void displayDefaultPanes();
private: private:
PanePtr createEscapeDialog(); PanePtr createEscapeDialog();

View File

@ -519,6 +519,10 @@ void UniverseClient::stopLua() {
m_scriptContexts.clear(); m_scriptContexts.clear();
} }
LuaRootPtr UniverseClient::luaRoot() {
return m_luaRoot;
}
bool UniverseClient::reloadPlayer(Json const& data, Uuid const&, bool resetInterfaces, bool showIndicator) { bool UniverseClient::reloadPlayer(Json const& data, Uuid const&, bool resetInterfaces, bool showIndicator) {
auto player = mainPlayer(); auto player = mainPlayer();
bool playerInWorld = player->inWorld(); bool playerInWorld = player->inWorld();

View File

@ -89,6 +89,7 @@ public:
void setLuaCallbacks(String const& groupName, LuaCallbacks const& callbacks); void setLuaCallbacks(String const& groupName, LuaCallbacks const& callbacks);
void startLua(); void startLua();
void stopLua(); void stopLua();
LuaRootPtr luaRoot();
bool reloadPlayer(Json const& data, Uuid const& uuid, bool resetInterfaces = false, bool showIndicator = false); bool reloadPlayer(Json const& data, Uuid const& uuid, bool resetInterfaces = false, bool showIndicator = false);
bool switchPlayer(Uuid const& uuid); bool switchPlayer(Uuid const& uuid);

View File

@ -411,6 +411,18 @@ LuaCallbacks Pane::makePaneCallbacks() {
return isDisplayed(); return isDisplayed();
}); });
callbacks.registerCallback("hasFocus", [this]() {
hasFocus();
});
callbacks.registerCallback("show", [this]() {
show();
});
callbacks.registerCallback("hide", [this]() {
hide();
});
return callbacks; return callbacks;
} }