Client: add form interception

register_on_receiving_inventory_form(function(formname, formspec))
    intercepts forms sent from the server
register_on_open_nodemeta_form(function(pos, formname, formspec))
    intercepts forms that are to be shown locally

nodemeta intecept probably doesn't intercept all ways it can be opened
This commit is contained in:
cron 2020-11-20 02:23:13 +00:00
parent 5d59ede9c4
commit 832558fb9e
5 changed files with 59 additions and 20 deletions

View File

@ -53,7 +53,7 @@ function core.override_item(name, redefinition)
end end
local nodedef = core.get_node_def(name) local nodedef = core.get_node_def(name)
table.combine(itemdef, nodedef) table.combine(itemdef, nodedef)
for k, v in pairs(redefinition) do for k, v in pairs(redefinition) do
rawset(itemdef, k, v) rawset(itemdef, k, v)
end end
@ -94,3 +94,5 @@ core.registered_on_item_use, core.register_on_item_use = make_registration()
core.registered_on_modchannel_message, core.register_on_modchannel_message = make_registration() core.registered_on_modchannel_message, core.register_on_modchannel_message = make_registration()
core.registered_on_modchannel_signal, core.register_on_modchannel_signal = make_registration() core.registered_on_modchannel_signal, core.register_on_modchannel_signal = make_registration()
core.registered_on_inventory_open, core.register_on_inventory_open = make_registration() core.registered_on_inventory_open, core.register_on_inventory_open = make_registration()
core.registered_on_receiving_inventory_form, core.register_on_receiving_inventory_form = make_registration()
core.registered_on_open_nodemeta_form, core.register_on_open_nodemeta_form = make_registration()

View File

@ -2624,20 +2624,23 @@ bool Game::nodePlacement(const ItemDefinition &selected_def,
if (nodedef_manager->get(map.getNode(nodepos)).rightclickable) if (nodedef_manager->get(map.getNode(nodepos)).rightclickable)
client->interact(INTERACT_PLACE, pointed); client->interact(INTERACT_PLACE, pointed);
infostream << "Launching custom inventory view" << std::endl; std::string formspec_str = meta->getString("formspec");
if (!client->getScript()->on_nodemeta_form_open(nodepos, "", formspec_str)) {
infostream << "Launching custom inventory view" << std::endl;
InventoryLocation inventoryloc; InventoryLocation inventoryloc;
inventoryloc.setNodeMeta(nodepos); inventoryloc.setNodeMeta(nodepos);
NodeMetadataFormSource *fs_src = new NodeMetadataFormSource( NodeMetadataFormSource *fs_src = new NodeMetadataFormSource(
&client->getEnv().getClientMap(), nodepos); &client->getEnv().getClientMap(), nodepos);
TextDest *txt_dst = new TextDestNodeMetadata(nodepos, client); TextDest *txt_dst = new TextDestNodeMetadata(nodepos, client);
auto *&formspec = m_game_ui->updateFormspec(""); auto *&formspec = m_game_ui->updateFormspec("");
GUIFormSpecMenu::create(formspec, client, &input->joystick, fs_src, GUIFormSpecMenu::create(formspec, client, &input->joystick, fs_src,
txt_dst, client->getFormspecPrepend()); txt_dst, client->getFormspecPrepend());
formspec->setFormSpec(meta->getString("formspec"), inventoryloc); formspec->setFormSpec(formspec_str, inventoryloc);
}
return false; return false;
} }

View File

@ -979,13 +979,15 @@ void Client::handleCommand_ShowFormSpec(NetworkPacket* pkt)
*pkt >> formname; *pkt >> formname;
ClientEvent *event = new ClientEvent(); if (!m_script->on_receiving_inventory_form(formname, formspec)) {
event->type = CE_SHOW_FORMSPEC; ClientEvent *event = new ClientEvent();
// pointer is required as event is a struct only! event->type = CE_SHOW_FORMSPEC;
// adding a std:string to a struct isn't possible // pointer is required as event is a struct only!
event->show_formspec.formspec = new std::string(formspec); // adding a std:string to a struct isn't possible
event->show_formspec.formname = new std::string(formname); event->show_formspec.formspec = new std::string(formspec);
m_client_event_queue.push(event); event->show_formspec.formname = new std::string(formname);
m_client_event_queue.push(event);
}
} }
void Client::handleCommand_SpawnParticle(NetworkPacket* pkt) void Client::handleCommand_SpawnParticle(NetworkPacket* pkt)

View File

@ -237,17 +237,46 @@ bool ScriptApiClient::on_inventory_open(Inventory *inventory)
void ScriptApiClient::open_special_inventory() void ScriptApiClient::open_special_inventory()
{ {
SCRIPTAPI_PRECHECKHEADER SCRIPTAPI_PRECHECKHEADER
PUSH_ERROR_HANDLER(L); PUSH_ERROR_HANDLER(L);
int error_handler = lua_gettop(L) - 1; int error_handler = lua_gettop(L) - 1;
lua_insert(L, error_handler); lua_insert(L, error_handler);
lua_getglobal(L, "core"); lua_getglobal(L, "core");
lua_getfield(L, -1, "open_special_inventory"); lua_getfield(L, -1, "open_special_inventory");
if (lua_isfunction(L, -1)) if (lua_isfunction(L, -1))
lua_pcall(L, 0, 0, error_handler); lua_pcall(L, 0, 0, error_handler);
} }
bool ScriptApiClient::on_receiving_inventory_form(std::string formname, std::string formspec)
{
SCRIPTAPI_PRECHECKHEADER
lua_getglobal(L, "core");
lua_getfield(L, -1, "registered_on_receiving_inventory_form");
lua_pushstring(L, formname.c_str());
lua_pushstring(L, formspec.c_str());
runCallbacks(2, RUN_CALLBACKS_MODE_OR);
return readParam<bool>(L, -1);
}
bool ScriptApiClient::on_nodemeta_form_open(v3s16 position, std::string formname, std::string formspec)
{
SCRIPTAPI_PRECHECKHEADER
lua_getglobal(L, "core");
lua_getfield(L, -1, "registered_on_nodemeta_form_open");
push_v3s16(L, position);
lua_pushstring(L, formname.c_str());
lua_pushstring(L, formspec.c_str());
runCallbacks(2, RUN_CALLBACKS_MODE_OR);
return readParam<bool>(L, -1);
}
void ScriptApiClient::setEnv(ClientEnvironment *env) void ScriptApiClient::setEnv(ClientEnvironment *env)
{ {
ScriptApiBase::setEnv(env); ScriptApiBase::setEnv(env);

View File

@ -61,5 +61,8 @@ public:
bool on_inventory_open(Inventory *inventory); bool on_inventory_open(Inventory *inventory);
void open_special_inventory(); void open_special_inventory();
bool on_receiving_inventory_form(std::string formname, std::string formspec);
bool on_nodemeta_form_open(v3s16 position, std::string formname, std::string formspec);
void setEnv(ClientEnvironment *env); void setEnv(ClientEnvironment *env);
}; };