1
0
Fork 0
mirror of https://github.com/luanti-org/luanti.git synced 2025-08-06 17:41:04 +00:00
luanti/src/client/particles.h

267 lines
6.8 KiB
C
Raw Normal View History

// Luanti
// SPDX-License-Identifier: LGPL-2.1-or-later
// Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>
2012-12-31 19:33:36 +01:00
#pragma once
2012-12-31 19:33:36 +01:00
#include <vector>
#include <unordered_map>
2012-12-31 19:33:36 +01:00
#include "irrlichttypes_extrabloated.h"
#include "irr_ptr.h"
#include "../particles.h"
2012-12-31 19:33:36 +01:00
struct ClientEvent;
class ParticleManager;
class ClientEnvironment;
struct MapNode;
struct ContentFeatures;
class LocalPlayer;
class ITextureSource;
class IGameDef;
class Client;
2023-10-11 17:07:30 +02:00
struct ClientParticleTexture
{
/* per-spawner structure used to store the ParticleTexture structs
2023-10-11 17:07:30 +02:00
* that spawned particles will refer to through ClientParticleTexRef */
ParticleTexture tex;
video::ITexture *ref = nullptr;
2023-10-11 17:07:30 +02:00
ClientParticleTexture() = default;
ClientParticleTexture(const ServerParticleTexture& p, ITextureSource *tsrc);
};
2023-10-11 17:07:30 +02:00
struct ClientParticleTexRef
{
/* per-particle structure used to avoid massively duplicating the
* fairly large ParticleTexture struct */
2023-10-11 17:07:30 +02:00
ParticleTexture *tex = nullptr;
video::ITexture *ref = nullptr;
ClientParticleTexRef() = default;
/* constructor used by particles spawned from a spawner */
2023-10-11 17:07:30 +02:00
explicit ClientParticleTexRef(ClientParticleTexture &t):
tex(&t.tex), ref(t.ref) {};
/* constructor used for node particles */
2023-10-11 17:07:30 +02:00
explicit ClientParticleTexRef(video::ITexture *tp): ref(tp) {};
};
class ParticleSpawner;
class ParticleBuffer;
class Particle
2012-12-31 19:33:36 +01:00
{
public:
2012-12-31 19:33:36 +01:00
Particle(
const ParticleParameters &p,
2023-10-11 17:07:30 +02:00
const ClientParticleTexRef &texture,
v2f texpos,
v2f texsize,
2023-10-11 17:07:30 +02:00
video::SColor color,
ParticleSpawner *parent = nullptr,
std::unique_ptr<ClientParticleTexture> owned_texture = nullptr
2012-12-31 19:33:36 +01:00
);
~Particle();
2012-12-31 19:33:36 +01:00
DISABLE_CLASS_COPY(Particle)
2012-12-31 19:33:36 +01:00
void step(float dtime, ClientEnvironment *env);
2012-12-31 19:33:36 +01:00
bool isExpired () const
{ return m_expiration < m_time; }
2012-12-31 19:33:36 +01:00
ParticleSpawner *getParent() const { return m_parent; }
2012-12-31 19:33:36 +01:00
const ClientParticleTexRef &getTextureRef() const { return m_texture; }
2012-12-31 19:33:36 +01:00
ParticleParamTypes::BlendMode getBlendMode() const
{ return m_texture.tex ? m_texture.tex->blendmode : m_p.texture.blendmode; }
ParticleBuffer *getBuffer() const { return m_buffer; }
bool attachToBuffer(ParticleBuffer *buffer);
2012-12-31 19:33:36 +01:00
private:
video::SColor updateLight(ClientEnvironment *env);
void updateVertices(ClientEnvironment *env, video::SColor color);
ParticleBuffer *m_buffer = nullptr;
u16 m_index; // index in m_buffer
float m_time = 0.0f;
2012-12-31 19:33:36 +01:00
float m_expiration;
// Color without lighting
video::SColor m_base_color;
2023-10-11 17:07:30 +02:00
ClientParticleTexRef m_texture;
v2f m_texpos;
v2f m_texsize;
2012-12-31 19:33:36 +01:00
v3f m_pos;
v3f m_velocity;
v3f m_acceleration;
2023-10-11 17:07:30 +02:00
const ParticleParameters m_p;
float m_animation_time = 0.0f;
int m_animation_frame = 0;
2023-10-11 17:07:30 +02:00
ParticleSpawner *m_parent = nullptr;
// Used if not spawned from a particlespawner
std::unique_ptr<ClientParticleTexture> m_owned_texture;
};
class ParticleSpawner
{
public:
ParticleSpawner(LocalPlayer *player,
2023-10-11 17:07:30 +02:00
const ParticleSpawnerParameters &params,
2016-08-04 13:09:21 -07:00
u16 attached_id,
2023-10-11 17:07:30 +02:00
std::vector<ClientParticleTexture> &&texpool,
ParticleManager *p_manager);
void step(float dtime, ClientEnvironment *env);
bool getExpired() const
2023-10-11 17:07:30 +02:00
{ return p.amount <= 0 && p.time != 0; }
bool hasActive() const { return m_active != 0; }
void decrActive() { m_active -= 1; }
private:
void spawnParticle(ClientEnvironment *env, float radius,
const core::matrix4 *attached_absolute_pos_rot_matrix);
2023-10-11 17:07:30 +02:00
size_t m_active;
ParticleManager *m_particlemanager;
float m_time;
LocalPlayer *m_player;
ParticleSpawnerParameters p;
2023-10-11 17:07:30 +02:00
std::vector<ClientParticleTexture> m_texpool;
std::vector<float> m_spawntimes;
2016-08-04 13:09:21 -07:00
u16 m_attached_id;
2012-12-31 19:33:36 +01:00
};
class ParticleBuffer : public scene::ISceneNode
{
friend class ParticleManager;
public:
ParticleBuffer(ClientEnvironment *env, const video::SMaterial &material);
// for pointer stability
DISABLE_CLASS_COPY(ParticleBuffer)
/// Reserves one more slot for a particle (4 vertices, 6 indices)
/// @return particle index within buffer
std::optional<u16> allocate();
/// Frees the particle at `index`
void release(u16 index);
/// @return video::S3DVertex[4]
video::S3DVertex *getVertices(u16 index);
inline bool isEmpty() const {
return m_free_list.size() == m_count;
}
virtual video::SMaterial &getMaterial(u32 num) override {
return m_mesh_buffer->getMaterial();
}
virtual u32 getMaterialCount() const override {
return 1;
}
virtual const core::aabbox3df &getBoundingBox() const override;
virtual void render() override;
virtual void OnRegisterSceneNode() override;
// we have 16-bit indices
static constexpr u16 MAX_PARTICLES_PER_BUFFER = 16000;
private:
irr_ptr<scene::SMeshBuffer> m_mesh_buffer;
// unused (e.g. expired) particle indices for re-use
std::vector<u16> m_free_list;
// for automatic deletion when unused for a while. is reset on allocate().
float m_usage_timer = 0;
// total count of contained particles
u16 m_count = 0;
mutable bool m_bounding_box_dirty = true;
};
/**
* Class doing particle as well as their spawners handling
*/
class ParticleManager
{
friend class ParticleSpawner;
public:
ParticleManager(ClientEnvironment* env);
2023-10-11 17:07:30 +02:00
DISABLE_CLASS_COPY(ParticleManager)
~ParticleManager();
void step (float dtime);
void handleParticleEvent(ClientEvent *event, Client *client,
LocalPlayer *player);
void addDiggingParticles(IGameDef *gamedef, LocalPlayer *player, v3s16 pos,
const MapNode &n, const ContentFeatures &f);
void addNodeParticle(IGameDef *gamedef, LocalPlayer *player, v3s16 pos,
const MapNode &n, const ContentFeatures &f);
void reserveParticleSpace(size_t max_estimate);
/**
* This function is only used by client particle spawners
*
* We don't need to check the particle spawner list because client ID will
* never overlap (u64)
* @return new id
*/
u64 generateSpawnerId()
{
return m_next_particle_spawner_id++;
}
protected:
static bool getNodeParticleParams(const MapNode &n, const ContentFeatures &f,
ParticleParameters &p, video::ITexture **texture, v2f &texpos,
v2f &texsize, video::SColor *color, u8 tilenum = 0);
static video::SMaterial getMaterialForParticle(const Particle *texture);
bool addParticle(std::unique_ptr<Particle> toadd);
2012-12-31 19:33:36 +01:00
private:
2023-10-11 17:07:30 +02:00
void addParticleSpawner(u64 id, std::unique_ptr<ParticleSpawner> toadd);
void deleteParticleSpawner(u64 id);
void stepParticles(float dtime);
void stepSpawners(float dtime);
void stepBuffers(float dtime);
void clearAll();
2023-10-11 17:07:30 +02:00
std::vector<std::unique_ptr<Particle>> m_particles;
std::unordered_map<u64, std::unique_ptr<ParticleSpawner>> m_particle_spawners;
std::vector<std::unique_ptr<ParticleSpawner>> m_dying_particle_spawners;
std::vector<irr_ptr<ParticleBuffer>> m_particle_buffers;
// Start the particle spawner ids generated from here after u32_max.
// lower values are for server sent spawners.
u64 m_next_particle_spawner_id = static_cast<u64>(U32_MAX) + 1;
2023-10-11 17:07:30 +02:00
ClientEnvironment *m_env;
IntervalLimiter m_buffer_gc;
std::mutex m_particle_list_lock;
std::mutex m_spawner_list_lock;
};