2017-01-21 15:02:08 +01:00
|
|
|
/*
|
|
|
|
Minetest
|
|
|
|
Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>
|
|
|
|
Copyright (C) 2017 nerzhul, Loic Blot <loic.blot@unix-experience.fr>
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2.1 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU Lesser General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public License along
|
|
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "l_client.h"
|
2017-07-16 10:47:31 +02:00
|
|
|
#include "chatmessage.h"
|
2017-08-16 22:11:45 +02:00
|
|
|
#include "client.h"
|
2017-08-28 20:02:23 +02:00
|
|
|
#include "client/clientevent.h"
|
2018-03-24 15:45:25 +01:00
|
|
|
#include "client/sound.h"
|
2017-04-06 09:10:59 +02:00
|
|
|
#include "clientenvironment.h"
|
2017-03-31 22:29:34 +02:00
|
|
|
#include "common/c_content.h"
|
|
|
|
#include "common/c_converter.h"
|
2017-01-22 09:05:09 +01:00
|
|
|
#include "cpp_api/s_base.h"
|
2017-01-24 17:26:15 +01:00
|
|
|
#include "gettext.h"
|
2017-03-31 22:29:34 +02:00
|
|
|
#include "l_internal.h"
|
2017-01-31 14:18:52 +01:00
|
|
|
#include "lua_api/l_item.h"
|
2017-04-04 07:41:37 +02:00
|
|
|
#include "lua_api/l_nodemeta.h"
|
2017-11-08 23:56:20 +01:00
|
|
|
#include "gui/mainmenumanager.h"
|
2017-04-04 07:41:37 +02:00
|
|
|
#include "map.h"
|
2017-04-06 09:10:59 +02:00
|
|
|
#include "util/string.h"
|
2017-05-21 14:40:55 +02:00
|
|
|
#include "nodedef.h"
|
2017-01-21 15:02:08 +01:00
|
|
|
|
|
|
|
int ModApiClient::l_get_current_modname(lua_State *L)
|
|
|
|
{
|
|
|
|
lua_rawgeti(L, LUA_REGISTRYINDEX, CUSTOM_RIDX_CURRENT_MOD_NAME);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-01-22 09:05:09 +01:00
|
|
|
// get_last_run_mod()
|
|
|
|
int ModApiClient::l_get_last_run_mod(lua_State *L)
|
|
|
|
{
|
|
|
|
lua_rawgeti(L, LUA_REGISTRYINDEX, CUSTOM_RIDX_CURRENT_MOD_NAME);
|
2018-06-30 17:11:38 +02:00
|
|
|
std::string current_mod = readParam<std::string>(L, -1, "");
|
|
|
|
if (current_mod.empty()) {
|
2017-01-22 09:05:09 +01:00
|
|
|
lua_pop(L, 1);
|
|
|
|
lua_pushstring(L, getScriptApiBase(L)->getOrigin().c_str());
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// set_last_run_mod(modname)
|
|
|
|
int ModApiClient::l_set_last_run_mod(lua_State *L)
|
|
|
|
{
|
2017-01-24 17:26:15 +01:00
|
|
|
if (!lua_isstring(L, 1))
|
|
|
|
return 0;
|
|
|
|
|
2017-01-22 09:05:09 +01:00
|
|
|
const char *mod = lua_tostring(L, 1);
|
|
|
|
getScriptApiBase(L)->setOriginDirect(mod);
|
2017-01-24 17:26:15 +01:00
|
|
|
lua_pushboolean(L, true);
|
|
|
|
return 1;
|
2017-01-22 09:05:09 +01:00
|
|
|
}
|
|
|
|
|
2017-05-20 16:46:12 +02:00
|
|
|
// print(text)
|
|
|
|
int ModApiClient::l_print(lua_State *L)
|
|
|
|
{
|
|
|
|
NO_MAP_LOCK_REQUIRED;
|
|
|
|
std::string text = luaL_checkstring(L, 1);
|
|
|
|
rawstream << text << std::endl;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-01-21 22:44:37 +01:00
|
|
|
// display_chat_message(message)
|
|
|
|
int ModApiClient::l_display_chat_message(lua_State *L)
|
|
|
|
{
|
2017-01-24 17:26:15 +01:00
|
|
|
if (!lua_isstring(L, 1))
|
|
|
|
return 0;
|
2017-01-21 22:44:37 +01:00
|
|
|
|
|
|
|
std::string message = luaL_checkstring(L, 1);
|
2017-07-16 10:47:31 +02:00
|
|
|
getClient(L)->pushToChatQueue(new ChatMessage(utf8_to_wide(message)));
|
2017-01-24 17:26:15 +01:00
|
|
|
lua_pushboolean(L, true);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-07-15 09:28:10 +02:00
|
|
|
// send_chat_message(message)
|
|
|
|
int ModApiClient::l_send_chat_message(lua_State *L)
|
|
|
|
{
|
|
|
|
if (!lua_isstring(L, 1))
|
|
|
|
return 0;
|
2017-07-18 21:39:55 +02:00
|
|
|
|
|
|
|
// If server disabled this API, discard
|
2018-06-20 22:36:08 +02:00
|
|
|
|
|
|
|
// clang-format off
|
|
|
|
if (getClient(L)->checkCSMRestrictionFlag(
|
|
|
|
CSMRestrictionFlags::CSM_RF_CHAT_MESSAGES))
|
2017-07-18 21:39:55 +02:00
|
|
|
return 0;
|
2018-06-20 22:36:08 +02:00
|
|
|
// clang-format on
|
2017-07-18 21:39:55 +02:00
|
|
|
|
2017-07-15 09:28:10 +02:00
|
|
|
std::string message = luaL_checkstring(L, 1);
|
|
|
|
getClient(L)->sendChatMessage(utf8_to_wide(message));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// clear_out_chat_queue()
|
|
|
|
int ModApiClient::l_clear_out_chat_queue(lua_State *L)
|
|
|
|
{
|
|
|
|
getClient(L)->clearOutChatQueue();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-03-22 21:13:03 +01:00
|
|
|
// get_player_names()
|
|
|
|
int ModApiClient::l_get_player_names(lua_State *L)
|
|
|
|
{
|
|
|
|
const std::list<std::string> &plist = getClient(L)->getConnectedPlayerNames();
|
|
|
|
lua_createtable(L, plist.size(), 0);
|
|
|
|
int newTable = lua_gettop(L);
|
|
|
|
int index = 1;
|
|
|
|
std::list<std::string>::const_iterator iter;
|
2017-04-25 10:17:53 +02:00
|
|
|
for (iter = plist.begin(); iter != plist.end(); ++iter) {
|
2017-03-22 21:13:03 +01:00
|
|
|
lua_pushstring(L, (*iter).c_str());
|
|
|
|
lua_rawseti(L, newTable, index);
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-01-24 17:26:15 +01:00
|
|
|
// show_formspec(formspec)
|
|
|
|
int ModApiClient::l_show_formspec(lua_State *L)
|
|
|
|
{
|
2017-03-31 22:29:34 +02:00
|
|
|
if (!lua_isstring(L, 1) || !lua_isstring(L, 2))
|
2017-01-24 17:26:15 +01:00
|
|
|
return 0;
|
|
|
|
|
2017-08-28 20:02:23 +02:00
|
|
|
ClientEvent *event = new ClientEvent();
|
|
|
|
event->type = CE_SHOW_LOCAL_FORMSPEC;
|
|
|
|
event->show_formspec.formname = new std::string(luaL_checkstring(L, 1));
|
|
|
|
event->show_formspec.formspec = new std::string(luaL_checkstring(L, 2));
|
2017-01-24 17:26:15 +01:00
|
|
|
getClient(L)->pushToEventQueue(event);
|
|
|
|
lua_pushboolean(L, true);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// send_respawn()
|
|
|
|
int ModApiClient::l_send_respawn(lua_State *L)
|
|
|
|
{
|
|
|
|
getClient(L)->sendRespawn();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-04-01 13:40:56 +02:00
|
|
|
// disconnect()
|
|
|
|
int ModApiClient::l_disconnect(lua_State *L)
|
|
|
|
{
|
|
|
|
// Stops badly written Lua code form causing boot loops
|
|
|
|
if (getClient(L)->isShutdown()) {
|
|
|
|
lua_pushboolean(L, false);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_gamecallback->disconnect();
|
|
|
|
lua_pushboolean(L, true);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-01-24 17:26:15 +01:00
|
|
|
// gettext(text)
|
|
|
|
int ModApiClient::l_gettext(lua_State *L)
|
|
|
|
{
|
|
|
|
std::string text = strgettext(std::string(luaL_checkstring(L, 1)));
|
|
|
|
lua_pushstring(L, text.c_str());
|
|
|
|
|
2017-01-21 22:44:37 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-01-30 20:10:37 +01:00
|
|
|
// get_node(pos)
|
|
|
|
// pos = {x=num, y=num, z=num}
|
|
|
|
int ModApiClient::l_get_node_or_nil(lua_State *L)
|
|
|
|
{
|
|
|
|
// pos
|
|
|
|
v3s16 pos = read_v3s16(L, 1);
|
2017-07-18 21:39:55 +02:00
|
|
|
|
2017-01-30 20:10:37 +01:00
|
|
|
// Do it
|
|
|
|
bool pos_ok;
|
|
|
|
MapNode n = getClient(L)->getNode(pos, &pos_ok);
|
|
|
|
if (pos_ok) {
|
|
|
|
// Return node
|
|
|
|
pushnode(L, n, getClient(L)->ndef());
|
2017-03-31 22:29:34 +02:00
|
|
|
} else {
|
2017-01-30 20:10:37 +01:00
|
|
|
lua_pushnil(L);
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-10-28 08:56:10 +02:00
|
|
|
int ModApiClient::l_get_language(lua_State *L)
|
|
|
|
{
|
|
|
|
char *locale = setlocale(LC_ALL, "");
|
|
|
|
lua_pushstring(L, locale);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-01-31 14:18:52 +01:00
|
|
|
int ModApiClient::l_get_wielded_item(lua_State *L)
|
|
|
|
{
|
|
|
|
Client *client = getClient(L);
|
|
|
|
|
|
|
|
Inventory local_inventory(client->idef());
|
|
|
|
client->getLocalInventory(local_inventory);
|
|
|
|
|
|
|
|
InventoryList *mlist = local_inventory.getList("main");
|
|
|
|
|
|
|
|
if (mlist && client->getPlayerItem() < mlist->getSize()) {
|
|
|
|
LuaItemStack::create(L, mlist->getItem(client->getPlayerItem()));
|
|
|
|
} else {
|
|
|
|
LuaItemStack::create(L, ItemStack());
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-04-04 07:41:37 +02:00
|
|
|
// get_meta(pos)
|
|
|
|
int ModApiClient::l_get_meta(lua_State *L)
|
|
|
|
{
|
|
|
|
v3s16 p = read_v3s16(L, 1);
|
|
|
|
NodeMetadata *meta = getClient(L)->getEnv().getMap().getNodeMetadata(p);
|
|
|
|
NodeMetaRef::createClient(L, meta);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-04-06 08:14:31 +02:00
|
|
|
int ModApiClient::l_sound_play(lua_State *L)
|
|
|
|
{
|
|
|
|
ISoundManager *sound = getClient(L)->getSoundManager();
|
|
|
|
|
|
|
|
SimpleSoundSpec spec;
|
|
|
|
read_soundspec(L, 1, spec);
|
2017-06-11 13:58:26 +02:00
|
|
|
float gain = 1.0f;
|
|
|
|
float pitch = 1.0f;
|
2017-04-06 08:14:31 +02:00
|
|
|
bool looped = false;
|
|
|
|
s32 handle;
|
|
|
|
|
|
|
|
if (lua_istable(L, 2)) {
|
|
|
|
getfloatfield(L, 2, "gain", gain);
|
2017-06-11 13:58:26 +02:00
|
|
|
getfloatfield(L, 2, "pitch", pitch);
|
2017-04-06 08:14:31 +02:00
|
|
|
getboolfield(L, 2, "loop", looped);
|
|
|
|
|
|
|
|
lua_getfield(L, 2, "pos");
|
|
|
|
if (!lua_isnil(L, -1)) {
|
|
|
|
v3f pos = read_v3f(L, -1) * BS;
|
|
|
|
lua_pop(L, 1);
|
2017-04-06 16:03:29 +02:00
|
|
|
handle = sound->playSoundAt(
|
2017-06-11 13:58:26 +02:00
|
|
|
spec.name, looped, gain * spec.gain, pos, pitch);
|
2017-04-06 08:14:31 +02:00
|
|
|
lua_pushinteger(L, handle);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-11 13:58:26 +02:00
|
|
|
handle = sound->playSound(spec.name, looped, gain * spec.gain, 0.0f, pitch);
|
2017-04-06 08:14:31 +02:00
|
|
|
lua_pushinteger(L, handle);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ModApiClient::l_sound_stop(lua_State *L)
|
|
|
|
{
|
|
|
|
u32 handle = luaL_checkinteger(L, 1);
|
|
|
|
|
|
|
|
getClient(L)->getSoundManager()->stopSound(handle);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-04-19 15:16:54 +02:00
|
|
|
// get_server_info()
|
|
|
|
int ModApiClient::l_get_server_info(lua_State *L)
|
2017-04-06 22:50:45 +02:00
|
|
|
{
|
2017-04-19 15:16:54 +02:00
|
|
|
Client *client = getClient(L);
|
|
|
|
Address serverAddress = client->getServerAddress();
|
|
|
|
lua_newtable(L);
|
|
|
|
lua_pushstring(L, client->getAddressName().c_str());
|
|
|
|
lua_setfield(L, -2, "address");
|
|
|
|
lua_pushstring(L, serverAddress.serializeString().c_str());
|
|
|
|
lua_setfield(L, -2, "ip");
|
|
|
|
lua_pushinteger(L, serverAddress.getPort());
|
|
|
|
lua_setfield(L, -2, "port");
|
|
|
|
lua_pushinteger(L, client->getProtoVersion());
|
|
|
|
lua_setfield(L, -2, "protocol_version");
|
2017-04-06 22:50:45 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-05-21 14:40:55 +02:00
|
|
|
// get_item_def(itemstring)
|
|
|
|
int ModApiClient::l_get_item_def(lua_State *L)
|
|
|
|
{
|
|
|
|
IGameDef *gdef = getGameDef(L);
|
|
|
|
assert(gdef);
|
|
|
|
|
|
|
|
IItemDefManager *idef = gdef->idef();
|
|
|
|
assert(idef);
|
2017-05-21 17:30:00 +02:00
|
|
|
|
2018-06-20 22:36:08 +02:00
|
|
|
// clang-format off
|
|
|
|
if (getClient(L)->checkCSMRestrictionFlag(
|
|
|
|
CSMRestrictionFlags::CSM_RF_READ_ITEMDEFS))
|
2017-07-18 21:39:55 +02:00
|
|
|
return 0;
|
2018-06-20 22:36:08 +02:00
|
|
|
// clang-format on
|
2017-07-18 21:39:55 +02:00
|
|
|
|
2017-05-21 14:40:55 +02:00
|
|
|
if (!lua_isstring(L, 1))
|
|
|
|
return 0;
|
|
|
|
|
2018-06-30 17:11:38 +02:00
|
|
|
std::string name = readParam<std::string>(L, 1);
|
2017-05-21 14:40:55 +02:00
|
|
|
if (!idef->isKnown(name))
|
|
|
|
return 0;
|
|
|
|
const ItemDefinition &def = idef->get(name);
|
|
|
|
|
2017-05-21 17:30:00 +02:00
|
|
|
push_item_definition_full(L, def);
|
|
|
|
|
|
|
|
return 1;
|
2017-05-21 14:40:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// get_node_def(nodename)
|
|
|
|
int ModApiClient::l_get_node_def(lua_State *L)
|
|
|
|
{
|
|
|
|
IGameDef *gdef = getGameDef(L);
|
|
|
|
assert(gdef);
|
2017-05-21 17:30:00 +02:00
|
|
|
|
2018-02-10 21:04:16 +01:00
|
|
|
const NodeDefManager *ndef = gdef->ndef();
|
2017-05-21 14:40:55 +02:00
|
|
|
assert(ndef);
|
2017-05-21 17:30:00 +02:00
|
|
|
|
2017-05-21 14:40:55 +02:00
|
|
|
if (!lua_isstring(L, 1))
|
|
|
|
return 0;
|
2017-05-21 17:30:00 +02:00
|
|
|
|
2018-06-20 22:36:08 +02:00
|
|
|
// clang-format off
|
|
|
|
if (getClient(L)->checkCSMRestrictionFlag(
|
|
|
|
CSMRestrictionFlags::CSM_RF_READ_NODEDEFS))
|
2017-07-18 21:39:55 +02:00
|
|
|
return 0;
|
2018-06-20 22:36:08 +02:00
|
|
|
// clang-format on
|
2017-07-18 21:39:55 +02:00
|
|
|
|
2018-06-30 17:11:38 +02:00
|
|
|
std::string name = readParam<std::string>(L, 1);
|
2017-05-21 14:40:55 +02:00
|
|
|
const ContentFeatures &cf = ndef->get(ndef->getId(name));
|
|
|
|
if (cf.name != name) // Unknown node. | name = <whatever>, cf.name = ignore
|
|
|
|
return 0;
|
2017-05-21 17:30:00 +02:00
|
|
|
|
2017-05-21 14:40:55 +02:00
|
|
|
push_content_features(L, cf);
|
2017-05-21 17:30:00 +02:00
|
|
|
|
2017-05-21 14:40:55 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-06-07 09:09:06 +02:00
|
|
|
int ModApiClient::l_get_privilege_list(lua_State *L)
|
|
|
|
{
|
|
|
|
const Client *client = getClient(L);
|
|
|
|
lua_newtable(L);
|
|
|
|
for (const std::string &priv : client->getPrivilegeList()) {
|
|
|
|
lua_pushboolean(L, true);
|
|
|
|
lua_setfield(L, -2, priv.c_str());
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
2017-06-30 20:14:39 +02:00
|
|
|
|
|
|
|
// get_builtin_path()
|
|
|
|
int ModApiClient::l_get_builtin_path(lua_State *L)
|
|
|
|
{
|
|
|
|
lua_pushstring(L, BUILTIN_MOD_NAME ":");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-01-21 15:02:08 +01:00
|
|
|
void ModApiClient::Initialize(lua_State *L, int top)
|
|
|
|
{
|
|
|
|
API_FCT(get_current_modname);
|
2017-05-20 16:46:12 +02:00
|
|
|
API_FCT(print);
|
2017-01-21 22:44:37 +01:00
|
|
|
API_FCT(display_chat_message);
|
2017-07-15 09:28:10 +02:00
|
|
|
API_FCT(send_chat_message);
|
|
|
|
API_FCT(clear_out_chat_queue);
|
2017-03-22 21:13:03 +01:00
|
|
|
API_FCT(get_player_names);
|
2017-01-22 09:05:09 +01:00
|
|
|
API_FCT(set_last_run_mod);
|
|
|
|
API_FCT(get_last_run_mod);
|
2017-01-24 17:26:15 +01:00
|
|
|
API_FCT(show_formspec);
|
|
|
|
API_FCT(send_respawn);
|
|
|
|
API_FCT(gettext);
|
2017-01-30 20:10:37 +01:00
|
|
|
API_FCT(get_node_or_nil);
|
2017-01-31 14:18:52 +01:00
|
|
|
API_FCT(get_wielded_item);
|
2017-04-01 13:40:56 +02:00
|
|
|
API_FCT(disconnect);
|
2017-04-04 07:41:37 +02:00
|
|
|
API_FCT(get_meta);
|
2017-04-06 08:14:31 +02:00
|
|
|
API_FCT(sound_play);
|
|
|
|
API_FCT(sound_stop);
|
2017-04-19 15:16:54 +02:00
|
|
|
API_FCT(get_server_info);
|
2017-05-21 14:40:55 +02:00
|
|
|
API_FCT(get_item_def);
|
|
|
|
API_FCT(get_node_def);
|
2017-06-07 09:09:06 +02:00
|
|
|
API_FCT(get_privilege_list);
|
2017-06-30 20:14:39 +02:00
|
|
|
API_FCT(get_builtin_path);
|
2017-10-28 08:56:10 +02:00
|
|
|
API_FCT(get_language);
|
2017-01-21 15:02:08 +01:00
|
|
|
}
|