123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266 |
- // Luanti
- // SPDX-License-Identifier: LGPL-2.1-or-later
- // Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>
- #pragma once
- #include <vector>
- #include <unordered_map>
- #include "irrlichttypes_extrabloated.h"
- #include "irr_ptr.h"
- #include "../particles.h"
- struct ClientEvent;
- class ParticleManager;
- class ClientEnvironment;
- struct MapNode;
- struct ContentFeatures;
- class LocalPlayer;
- class ITextureSource;
- class IGameDef;
- class Client;
- struct ClientParticleTexture
- {
- /* per-spawner structure used to store the ParticleTexture structs
- * that spawned particles will refer to through ClientParticleTexRef */
- ParticleTexture tex;
- video::ITexture *ref = nullptr;
- ClientParticleTexture() = default;
- ClientParticleTexture(const ServerParticleTexture& p, ITextureSource *tsrc);
- };
- struct ClientParticleTexRef
- {
- /* per-particle structure used to avoid massively duplicating the
- * fairly large ParticleTexture struct */
- ParticleTexture *tex = nullptr;
- video::ITexture *ref = nullptr;
- ClientParticleTexRef() = default;
- /* constructor used by particles spawned from a spawner */
- explicit ClientParticleTexRef(ClientParticleTexture &t):
- tex(&t.tex), ref(t.ref) {};
- /* constructor used for node particles */
- explicit ClientParticleTexRef(video::ITexture *tp): ref(tp) {};
- };
- class ParticleSpawner;
- class ParticleBuffer;
- class Particle
- {
- public:
- Particle(
- const ParticleParameters &p,
- const ClientParticleTexRef &texture,
- v2f texpos,
- v2f texsize,
- video::SColor color,
- ParticleSpawner *parent = nullptr,
- std::unique_ptr<ClientParticleTexture> owned_texture = nullptr
- );
- ~Particle();
- DISABLE_CLASS_COPY(Particle)
- void step(float dtime, ClientEnvironment *env);
- bool isExpired () const
- { return m_expiration < m_time; }
- ParticleSpawner *getParent() const { return m_parent; }
- const ClientParticleTexRef &getTextureRef() const { return m_texture; }
- 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);
- 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;
- float m_expiration;
- // Color without lighting
- video::SColor m_base_color;
- ClientParticleTexRef m_texture;
- v2f m_texpos;
- v2f m_texsize;
- v3f m_pos;
- v3f m_velocity;
- v3f m_acceleration;
- const ParticleParameters m_p;
- float m_animation_time = 0.0f;
- int m_animation_frame = 0;
- ParticleSpawner *m_parent = nullptr;
- // Used if not spawned from a particlespawner
- std::unique_ptr<ClientParticleTexture> m_owned_texture;
- };
- class ParticleSpawner
- {
- public:
- ParticleSpawner(LocalPlayer *player,
- const ParticleSpawnerParameters ¶ms,
- u16 attached_id,
- std::vector<ClientParticleTexture> &&texpool,
- ParticleManager *p_manager);
- void step(float dtime, ClientEnvironment *env);
- bool getExpired() const
- { 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);
- size_t m_active;
- ParticleManager *m_particlemanager;
- float m_time;
- LocalPlayer *m_player;
- ParticleSpawnerParameters p;
- std::vector<ClientParticleTexture> m_texpool;
- std::vector<float> m_spawntimes;
- u16 m_attached_id;
- };
- 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);
- 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);
- private:
- 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();
- 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;
- ClientEnvironment *m_env;
- IntervalLimiter m_buffer_gc;
- std::mutex m_particle_list_lock;
- std::mutex m_spawner_list_lock;
- };
|