forked from Mirrorlandia_minetest/minetest
Implement spawning particles with node texture appearance
This commit is contained in:
parent
15ba75e4cf
commit
9d6e7e48d6
@ -164,6 +164,7 @@ LOCAL_SRC_FILES := \
|
|||||||
../../../src/noise.cpp \
|
../../../src/noise.cpp \
|
||||||
../../../src/objdef.cpp \
|
../../../src/objdef.cpp \
|
||||||
../../../src/object_properties.cpp \
|
../../../src/object_properties.cpp \
|
||||||
|
../../../src/particles.cpp \
|
||||||
../../../src/pathfinder.cpp \
|
../../../src/pathfinder.cpp \
|
||||||
../../../src/player.cpp \
|
../../../src/player.cpp \
|
||||||
../../../src/porting.cpp \
|
../../../src/porting.cpp \
|
||||||
|
@ -7835,6 +7835,8 @@ Used by `minetest.add_particle`.
|
|||||||
|
|
||||||
size = 1,
|
size = 1,
|
||||||
-- Scales the visual size of the particle texture.
|
-- Scales the visual size of the particle texture.
|
||||||
|
-- If `node` is set, size can be set to 0 to spawn a randomly-sized
|
||||||
|
-- particle (just like actual node dig particles).
|
||||||
|
|
||||||
collisiondetection = false,
|
collisiondetection = false,
|
||||||
-- If true collides with `walkable` nodes and, depending on the
|
-- If true collides with `walkable` nodes and, depending on the
|
||||||
@ -7853,6 +7855,7 @@ Used by `minetest.add_particle`.
|
|||||||
-- If true faces player using y axis only
|
-- If true faces player using y axis only
|
||||||
|
|
||||||
texture = "image.png",
|
texture = "image.png",
|
||||||
|
-- The texture of the particle
|
||||||
|
|
||||||
playername = "singleplayer",
|
playername = "singleplayer",
|
||||||
-- Optional, if specified spawns particle only on the player's client
|
-- Optional, if specified spawns particle only on the player's client
|
||||||
@ -7863,6 +7866,17 @@ Used by `minetest.add_particle`.
|
|||||||
glow = 0
|
glow = 0
|
||||||
-- Optional, specify particle self-luminescence in darkness.
|
-- Optional, specify particle self-luminescence in darkness.
|
||||||
-- Values 0-14.
|
-- Values 0-14.
|
||||||
|
|
||||||
|
node = {name = "ignore", param2 = 0},
|
||||||
|
-- Optional, if specified the particle will have the same appearance as
|
||||||
|
-- node dig particles for the given node.
|
||||||
|
-- `texture` and `animation` will be ignored if this is set.
|
||||||
|
|
||||||
|
node_tile = 0,
|
||||||
|
-- Optional, only valid in combination with `node`
|
||||||
|
-- If set to a valid number 1-6, specifies the tile from which the
|
||||||
|
-- particle texture is picked.
|
||||||
|
-- Otherwise, the default behavior is used. (currently: any random tile)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -7892,7 +7906,9 @@ Used by `minetest.add_particlespawner`.
|
|||||||
maxsize = 1,
|
maxsize = 1,
|
||||||
-- The particles' properties are random values between the min and max
|
-- The particles' properties are random values between the min and max
|
||||||
-- values.
|
-- values.
|
||||||
-- pos, velocity, acceleration, expirationtime, size
|
-- applies to: pos, velocity, acceleration, expirationtime, size
|
||||||
|
-- If `node` is set, min and maxsize can be set to 0 to spawn
|
||||||
|
-- randomly-sized particles (just like actual node dig particles).
|
||||||
|
|
||||||
collisiondetection = false,
|
collisiondetection = false,
|
||||||
-- If true collide with `walkable` nodes and, depending on the
|
-- If true collide with `walkable` nodes and, depending on the
|
||||||
@ -7915,6 +7931,7 @@ Used by `minetest.add_particlespawner`.
|
|||||||
-- If true face player using y axis only
|
-- If true face player using y axis only
|
||||||
|
|
||||||
texture = "image.png",
|
texture = "image.png",
|
||||||
|
-- The texture of the particle
|
||||||
|
|
||||||
playername = "singleplayer",
|
playername = "singleplayer",
|
||||||
-- Optional, if specified spawns particles only on the player's client
|
-- Optional, if specified spawns particles only on the player's client
|
||||||
@ -7925,6 +7942,17 @@ Used by `minetest.add_particlespawner`.
|
|||||||
glow = 0
|
glow = 0
|
||||||
-- Optional, specify particle self-luminescence in darkness.
|
-- Optional, specify particle self-luminescence in darkness.
|
||||||
-- Values 0-14.
|
-- Values 0-14.
|
||||||
|
|
||||||
|
node = {name = "ignore", param2 = 0},
|
||||||
|
-- Optional, if specified the particles will have the same appearance as
|
||||||
|
-- node dig particles for the given node.
|
||||||
|
-- `texture` and `animation` will be ignored if this is set.
|
||||||
|
|
||||||
|
node_tile = 0,
|
||||||
|
-- Optional, only valid in combination with `node`
|
||||||
|
-- If set to a valid number 1-6, specifies the tile from which the
|
||||||
|
-- particle texture is picked.
|
||||||
|
-- Otherwise, the default behavior is used. (currently: any random tile)
|
||||||
}
|
}
|
||||||
|
|
||||||
`HTTPRequest` definition
|
`HTTPRequest` definition
|
||||||
|
@ -304,18 +304,37 @@ void ParticleSpawner::spawnParticle(ClientEnvironment *env, float radius,
|
|||||||
}
|
}
|
||||||
|
|
||||||
pp.expirationtime = random_f32(p.minexptime, p.maxexptime);
|
pp.expirationtime = random_f32(p.minexptime, p.maxexptime);
|
||||||
pp.size = random_f32(p.minsize, p.maxsize);
|
|
||||||
|
|
||||||
p.copyCommon(pp);
|
p.copyCommon(pp);
|
||||||
|
|
||||||
|
video::ITexture *texture;
|
||||||
|
v2f texpos, texsize;
|
||||||
|
video::SColor color(0xFFFFFFFF);
|
||||||
|
|
||||||
|
if (p.node.getContent() != CONTENT_IGNORE) {
|
||||||
|
const ContentFeatures &f =
|
||||||
|
m_particlemanager->m_env->getGameDef()->ndef()->get(p.node);
|
||||||
|
if (!ParticleManager::getNodeParticleParams(p.node, f, pp, &texture,
|
||||||
|
texpos, texsize, &color, p.node_tile))
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
texture = m_texture;
|
||||||
|
texpos = v2f(0.0f, 0.0f);
|
||||||
|
texsize = v2f(1.0f, 1.0f);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Allow keeping default random size
|
||||||
|
if (p.maxsize > 0.0f)
|
||||||
|
pp.size = random_f32(p.minsize, p.maxsize);
|
||||||
|
|
||||||
m_particlemanager->addParticle(new Particle(
|
m_particlemanager->addParticle(new Particle(
|
||||||
m_gamedef,
|
m_gamedef,
|
||||||
m_player,
|
m_player,
|
||||||
env,
|
env,
|
||||||
pp,
|
pp,
|
||||||
m_texture,
|
texture,
|
||||||
v2f(0.0, 0.0),
|
texpos,
|
||||||
v2f(1.0, 1.0)
|
texsize,
|
||||||
|
color
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -460,17 +479,35 @@ void ParticleManager::handleParticleEvent(ClientEvent *event, Client *client,
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case CE_SPAWN_PARTICLE: {
|
case CE_SPAWN_PARTICLE: {
|
||||||
const ParticleParameters &p = *event->spawn_particle;
|
ParticleParameters &p = *event->spawn_particle;
|
||||||
video::ITexture *texture =
|
|
||||||
client->tsrc()->getTextureForMesh(p.texture);
|
|
||||||
|
|
||||||
|
video::ITexture *texture;
|
||||||
|
v2f texpos, texsize;
|
||||||
|
video::SColor color(0xFFFFFFFF);
|
||||||
|
|
||||||
|
f32 oldsize = p.size;
|
||||||
|
|
||||||
|
if (p.node.getContent() != CONTENT_IGNORE) {
|
||||||
|
const ContentFeatures &f = m_env->getGameDef()->ndef()->get(p.node);
|
||||||
|
if (!getNodeParticleParams(p.node, f, p, &texture, texpos,
|
||||||
|
texsize, &color, p.node_tile))
|
||||||
|
texture = nullptr;
|
||||||
|
} else {
|
||||||
|
texture = client->tsrc()->getTextureForMesh(p.texture);
|
||||||
|
texpos = v2f(0.0f, 0.0f);
|
||||||
|
texsize = v2f(1.0f, 1.0f);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Allow keeping default random size
|
||||||
|
if (oldsize > 0.0f)
|
||||||
|
p.size = oldsize;
|
||||||
|
|
||||||
|
if (texture) {
|
||||||
Particle *toadd = new Particle(client, player, m_env,
|
Particle *toadd = new Particle(client, player, m_env,
|
||||||
p,
|
p, texture, texpos, texsize, color);
|
||||||
texture,
|
|
||||||
v2f(0.0, 0.0),
|
|
||||||
v2f(1.0, 1.0));
|
|
||||||
|
|
||||||
addParticle(toadd);
|
addParticle(toadd);
|
||||||
|
}
|
||||||
|
|
||||||
delete event->spawn_particle;
|
delete event->spawn_particle;
|
||||||
break;
|
break;
|
||||||
@ -480,15 +517,19 @@ void ParticleManager::handleParticleEvent(ClientEvent *event, Client *client,
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool ParticleManager::getNodeParticleParams(const MapNode &n,
|
bool ParticleManager::getNodeParticleParams(const MapNode &n,
|
||||||
const ContentFeatures &f, ParticleParameters &p,
|
const ContentFeatures &f, ParticleParameters &p, video::ITexture **texture,
|
||||||
video::ITexture **texture, v2f &texpos, v2f &texsize, video::SColor *color)
|
v2f &texpos, v2f &texsize, video::SColor *color, u8 tilenum)
|
||||||
{
|
{
|
||||||
// No particles for "airlike" nodes
|
// No particles for "airlike" nodes
|
||||||
if (f.drawtype == NDT_AIRLIKE)
|
if (f.drawtype == NDT_AIRLIKE)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// Texture
|
// Texture
|
||||||
u8 texid = rand() % 6;
|
u8 texid;
|
||||||
|
if (tilenum > 0 && tilenum <= 6)
|
||||||
|
texid = tilenum - 1;
|
||||||
|
else
|
||||||
|
texid = rand() % 6;
|
||||||
const TileLayer &tile = f.tiles[texid].layers[0];
|
const TileLayer &tile = f.tiles[texid].layers[0];
|
||||||
p.animation.type = TAT_NONE;
|
p.animation.type = TAT_NONE;
|
||||||
|
|
||||||
|
@ -42,7 +42,7 @@ class Particle : public scene::ISceneNode
|
|||||||
video::ITexture *texture,
|
video::ITexture *texture,
|
||||||
v2f texpos,
|
v2f texpos,
|
||||||
v2f texsize,
|
v2f texsize,
|
||||||
video::SColor color = video::SColor(0xFFFFFFFF)
|
video::SColor color
|
||||||
);
|
);
|
||||||
~Particle() = default;
|
~Particle() = default;
|
||||||
|
|
||||||
@ -171,7 +171,7 @@ public:
|
|||||||
protected:
|
protected:
|
||||||
static bool getNodeParticleParams(const MapNode &n, const ContentFeatures &f,
|
static bool getNodeParticleParams(const MapNode &n, const ContentFeatures &f,
|
||||||
ParticleParameters &p, video::ITexture **texture, v2f &texpos,
|
ParticleParameters &p, video::ITexture **texture, v2f &texpos,
|
||||||
v2f &texsize, video::SColor *color);
|
v2f &texsize, video::SColor *color, u8 tilenum = 0);
|
||||||
|
|
||||||
void addParticle(Particle* toadd);
|
void addParticle(Particle* toadd);
|
||||||
|
|
||||||
|
@ -1003,6 +1003,16 @@ void Client::handleCommand_AddParticleSpawner(NetworkPacket* pkt)
|
|||||||
p.glow = readU8(is);
|
p.glow = readU8(is);
|
||||||
p.object_collision = readU8(is);
|
p.object_collision = readU8(is);
|
||||||
|
|
||||||
|
// This is kinda awful
|
||||||
|
do {
|
||||||
|
u16 tmp_param0 = readU16(is);
|
||||||
|
if (is.eof())
|
||||||
|
break;
|
||||||
|
p.node.param0 = tmp_param0;
|
||||||
|
p.node.param2 = readU8(is);
|
||||||
|
p.node_tile = readU8(is);
|
||||||
|
} while (0);
|
||||||
|
|
||||||
auto event = new ClientEvent();
|
auto event = new ClientEvent();
|
||||||
event->type = CE_ADD_PARTICLESPAWNER;
|
event->type = CE_ADD_PARTICLESPAWNER;
|
||||||
event->add_particlespawner.p = new ParticleSpawnerParameters(p);
|
event->add_particlespawner.p = new ParticleSpawnerParameters(p);
|
||||||
|
@ -34,6 +34,9 @@ void ParticleParameters::serialize(std::ostream &os, u16 protocol_ver) const
|
|||||||
animation.serialize(os, 6); /* NOT the protocol ver */
|
animation.serialize(os, 6); /* NOT the protocol ver */
|
||||||
writeU8(os, glow);
|
writeU8(os, glow);
|
||||||
writeU8(os, object_collision);
|
writeU8(os, object_collision);
|
||||||
|
writeU16(os, node.param0);
|
||||||
|
writeU8(os, node.param2);
|
||||||
|
writeU8(os, node_tile);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ParticleParameters::deSerialize(std::istream &is, u16 protocol_ver)
|
void ParticleParameters::deSerialize(std::istream &is, u16 protocol_ver)
|
||||||
@ -50,4 +53,11 @@ void ParticleParameters::deSerialize(std::istream &is, u16 protocol_ver)
|
|||||||
animation.deSerialize(is, 6); /* NOT the protocol ver */
|
animation.deSerialize(is, 6); /* NOT the protocol ver */
|
||||||
glow = readU8(is);
|
glow = readU8(is);
|
||||||
object_collision = readU8(is);
|
object_collision = readU8(is);
|
||||||
|
// This is kinda awful
|
||||||
|
u16 tmp_param0 = readU16(is);
|
||||||
|
if (is.eof())
|
||||||
|
return;
|
||||||
|
node.param0 = tmp_param0;
|
||||||
|
node.param2 = readU8(is);
|
||||||
|
node_tile = readU8(is);
|
||||||
}
|
}
|
||||||
|
@ -22,6 +22,7 @@ with this program; if not, write to the Free Software Foundation, Inc.,
|
|||||||
#include <string>
|
#include <string>
|
||||||
#include "irrlichttypes_bloated.h"
|
#include "irrlichttypes_bloated.h"
|
||||||
#include "tileanimation.h"
|
#include "tileanimation.h"
|
||||||
|
#include "mapnode.h"
|
||||||
|
|
||||||
// This file defines the particle-related structures that both the server and
|
// This file defines the particle-related structures that both the server and
|
||||||
// client need. The ParticleManager and rendering is in client/particles.h
|
// client need. The ParticleManager and rendering is in client/particles.h
|
||||||
@ -34,9 +35,12 @@ struct CommonParticleParams {
|
|||||||
std::string texture;
|
std::string texture;
|
||||||
struct TileAnimationParams animation;
|
struct TileAnimationParams animation;
|
||||||
u8 glow = 0;
|
u8 glow = 0;
|
||||||
|
MapNode node;
|
||||||
|
u8 node_tile = 0;
|
||||||
|
|
||||||
CommonParticleParams() {
|
CommonParticleParams() {
|
||||||
animation.type = TAT_NONE;
|
animation.type = TAT_NONE;
|
||||||
|
node.setContent(CONTENT_IGNORE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This helper is useful for copying params from
|
/* This helper is useful for copying params from
|
||||||
@ -49,6 +53,8 @@ struct CommonParticleParams {
|
|||||||
to.texture = texture;
|
to.texture = texture;
|
||||||
to.animation = animation;
|
to.animation = animation;
|
||||||
to.glow = glow;
|
to.glow = glow;
|
||||||
|
to.node = node;
|
||||||
|
to.node_tile = node_tile;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -111,6 +111,13 @@ int ModApiParticles::l_add_particle(lua_State *L)
|
|||||||
p.texture = getstringfield_default(L, 1, "texture", p.texture);
|
p.texture = getstringfield_default(L, 1, "texture", p.texture);
|
||||||
p.glow = getintfield_default(L, 1, "glow", p.glow);
|
p.glow = getintfield_default(L, 1, "glow", p.glow);
|
||||||
|
|
||||||
|
lua_getfield(L, 1, "node");
|
||||||
|
if (lua_istable(L, -1))
|
||||||
|
p.node = readnode(L, -1, getGameDef(L)->ndef());
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
p.node_tile = getintfield_default(L, 1, "node_tile", p.node_tile);
|
||||||
|
|
||||||
playername = getstringfield_default(L, 1, "playername", "");
|
playername = getstringfield_default(L, 1, "playername", "");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -231,6 +238,13 @@ int ModApiParticles::l_add_particlespawner(lua_State *L)
|
|||||||
p.texture = getstringfield_default(L, 1, "texture", p.texture);
|
p.texture = getstringfield_default(L, 1, "texture", p.texture);
|
||||||
playername = getstringfield_default(L, 1, "playername", "");
|
playername = getstringfield_default(L, 1, "playername", "");
|
||||||
p.glow = getintfield_default(L, 1, "glow", p.glow);
|
p.glow = getintfield_default(L, 1, "glow", p.glow);
|
||||||
|
|
||||||
|
lua_getfield(L, 1, "node");
|
||||||
|
if (lua_istable(L, -1))
|
||||||
|
p.node = readnode(L, -1, getGameDef(L)->ndef());
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
p.node_tile = getintfield_default(L, 1, "node_tile", p.node_tile);
|
||||||
}
|
}
|
||||||
|
|
||||||
u32 id = getServer(L)->addParticleSpawner(p, attached, playername);
|
u32 id = getServer(L)->addParticleSpawner(p, attached, playername);
|
||||||
|
@ -67,6 +67,13 @@ int ModApiParticlesLocal::l_add_particle(lua_State *L)
|
|||||||
p.texture = getstringfield_default(L, 1, "texture", p.texture);
|
p.texture = getstringfield_default(L, 1, "texture", p.texture);
|
||||||
p.glow = getintfield_default(L, 1, "glow", p.glow);
|
p.glow = getintfield_default(L, 1, "glow", p.glow);
|
||||||
|
|
||||||
|
lua_getfield(L, 1, "node");
|
||||||
|
if (lua_istable(L, -1))
|
||||||
|
p.node = readnode(L, -1, getGameDef(L)->ndef());
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
p.node_tile = getintfield_default(L, 1, "node_tile", p.node_tile);
|
||||||
|
|
||||||
ClientEvent *event = new ClientEvent();
|
ClientEvent *event = new ClientEvent();
|
||||||
event->type = CE_SPAWN_PARTICLE;
|
event->type = CE_SPAWN_PARTICLE;
|
||||||
event->spawn_particle = new ParticleParameters(p);
|
event->spawn_particle = new ParticleParameters(p);
|
||||||
@ -134,6 +141,13 @@ int ModApiParticlesLocal::l_add_particlespawner(lua_State *L)
|
|||||||
p.texture = getstringfield_default(L, 1, "texture", p.texture);
|
p.texture = getstringfield_default(L, 1, "texture", p.texture);
|
||||||
p.glow = getintfield_default(L, 1, "glow", p.glow);
|
p.glow = getintfield_default(L, 1, "glow", p.glow);
|
||||||
|
|
||||||
|
lua_getfield(L, 1, "node");
|
||||||
|
if (lua_istable(L, -1))
|
||||||
|
p.node = readnode(L, -1, getGameDef(L)->ndef());
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
p.node_tile = getintfield_default(L, 1, "node_tile", p.node_tile);
|
||||||
|
|
||||||
u64 id = getClient(L)->getParticleManager()->generateSpawnerId();
|
u64 id = getClient(L)->getParticleManager()->generateSpawnerId();
|
||||||
|
|
||||||
auto event = new ClientEvent();
|
auto event = new ClientEvent();
|
||||||
|
@ -1577,6 +1577,7 @@ void Server::SendAddParticleSpawner(session_t peer_id, u16 protocol_version,
|
|||||||
pkt.putRawString(os.str());
|
pkt.putRawString(os.str());
|
||||||
}
|
}
|
||||||
pkt << p.glow << p.object_collision;
|
pkt << p.glow << p.object_collision;
|
||||||
|
pkt << p.node.param0 << p.node.param2 << p.node_tile;
|
||||||
|
|
||||||
Send(&pkt);
|
Send(&pkt);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user