15 "PhysicsOperator created");
24 if (vertices.empty()) {
26 "Cannot initialize PhysicsOperator with zero vertices");
34 "PhysicsOperator initialized with {} points", vertices.size());
42 const std::vector<std::vector<PointVertex>>& collections)
44 for (
const auto& collection : collections) {
49 "PhysicsOperator initialized with {} collections",
54 const std::vector<PointVertex>& vertices,
55 float mass_multiplier)
57 if (vertices.empty()) {
59 "Cannot add collection with zero vertices");
64 group.
collection = std::make_shared<GpuSync::PointCollectionNode>();
68 for (
size_t i = 0; i < vertices.size(); ++i) {
71 .force = glm::vec3(0.0F),
72 .mass = mass_multiplier
82 "Added collection #{} with {} points (mass_mult={:.2f})",
102 group.collection->compute_frame();
112 return Reflect::string_to_enum_case_insensitive<PhysicsParameter>(param);
121 Reflect::string_to_enum_or_throw_case_insensitive<PhysicsParameter>(
122 param,
"PhysicsOperator parameter");
123 }
catch (
const std::invalid_argument& e) {
128 "Unknown physics parameter: '{}'", param);
132 switch (*param_enum) {
143 m_drag = glm::clamp(
static_cast<float>(value), 0.0F, 1.0F);
160 auto& points = group.collection->get_points();
161 for (
auto& pt : points) {
177 if (query ==
"point_count") {
180 if (query ==
"collection_count") {
183 if (query ==
"avg_velocity") {
185 size_t total_points = 0;
188 for (
const auto& state : group.physics_state) {
189 avg += state.velocity;
194 if (total_points > 0) {
195 avg /=
static_cast<float>(total_points);
197 return static_cast<double>(glm::length(avg));
209 std::vector<PointVertex> positions;
212 const auto& points = group.collection->get_points();
213 for (
const auto& pt : points) {
214 positions.push_back(pt);
233 auto span = group.collection->get_vertex_data();
236 span.begin(), span.end());
243 size_t current_offset = 0;
246 size_t group_size = group.physics_state.size();
248 if (global_index < current_offset + group_size) {
249 size_t local_index = global_index - current_offset;
250 return static_cast<double>(glm::length(group.physics_state[local_index].velocity));
253 current_offset += group_size;
265 auto layout_opt =
m_collections[0].collection->get_vertex_layout();
266 if (!layout_opt.has_value()) {
277 total += group.collection->get_vertex_count();
284 return std::ranges::any_of(
286 [](
const auto& group) {
return group.collection->needs_gpu_update(); });
292 group.collection->mark_vertex_data_dirty(
false);
300 total += group.collection->get_point_count();
322 std::string_view param,
323 const std::shared_ptr<NodeNetwork>& source)
327 if (source->get_node_count() != point_count) {
329 "ONE_TO_ONE size mismatch: {} particles vs {} source nodes",
330 point_count, source->get_node_count());
334 if (param ==
"force_x" || param ==
"force_y" || param ==
"force_z") {
336 }
else if (param ==
"mass") {
344 std::string_view param,
345 const std::shared_ptr<NodeNetwork>& source)
347 size_t global_index = 0;
350 for (
auto& i : group.physics_state) {
351 auto val = source->get_node_output(global_index++);
355 auto force =
static_cast<float>(*val);
357 if (param ==
"force_x") {
359 }
else if (param ==
"force_y") {
361 }
else if (param ==
"force_z") {
369 const std::shared_ptr<NodeNetwork>& source)
371 size_t global_index = 0;
374 for (
auto& i : group.physics_state) {
375 auto val = source->get_node_output(global_index++);
379 i.mass = std::max(0.1F,
static_cast<float>(*val));
391 for (
auto& state : group.physics_state) {
410 auto& points = group.collection->get_points();
412 for (
size_t i = 0; i < points.size(); ++i) {
414 group.physics_state[i].force += field(points[i].
position);
426 for (
auto& state : group.physics_state) {
427 state.force += field(glm::vec3(0.0F));
436 auto& points1 = group1.collection->get_points();
438 for (
size_t i = 0; i < points1.size(); ++i) {
439 const auto& pos_i = points1[i].position;
440 auto& state_i = group1.physics_state[i];
444 auto& points2 = group2.collection->get_points();
446 size_t start_j = (g1 == g2) ? i + 1 : 0;
448 for (
size_t j = start_j; j < points2.size(); ++j) {
449 const auto& pos_j = points2[j].position;
450 auto& state_j = group2.physics_state[j];
452 glm::vec3 delta = pos_j - pos_i;
453 float distance = glm::length(delta);
455 if (distance < m_interaction_radius && distance > 0.001F) {
456 glm::vec3 direction = delta / distance;
460 float repulsion_force = 0.0F;
465 glm::vec3 force = direction * (spring_force - repulsion_force);
467 state_i.force += force;
468 state_j.force -= force;
481 auto& points = group.collection->get_points();
483 for (
size_t i = 0; i < points.size(); ++i) {
484 group.physics_state[i].force += field(points[i].
position) * group.physics_state[i].mass;
492 auto& points = group.collection->get_points();
494 for (
size_t i = 0; i < points.size(); ++i) {
495 auto& state = group.physics_state[i];
496 auto& vertex = points[i];
498 glm::vec3 acceleration = state.force / state.mass;
499 state.velocity += acceleration * dt;
500 state.velocity *= (1.0F -
m_drag);
501 vertex.position += state.velocity * dt;
503 state.force = glm::vec3(0.0F);
514 constexpr float damping = 0.8F;
517 auto& points = group.collection->get_points();
519 for (
size_t i = 0; i < points.size(); ++i) {
520 auto& vertex = points[i];
521 auto& state = group.physics_state[i];
523 for (
int axis = 0; axis < 3; ++axis) {
528 state.velocity[axis] *= -damping;
535 state.velocity[axis] = 0.0F;
540 }
else if (vertex.position[axis] >
m_bounds.
max[axis]) {
544 state.velocity[axis] *= -damping;
551 state.velocity[axis] = 0.0F;
565 group.collection->mark_vertex_data_dirty(
true);
574 size_t local_index = global_index - offset;
575 return &group.collection->get_points()[local_index];
577 offset += group.collection->get_point_count();
585 for (
auto& state : group.physics_state) {
586 state.velocity += impulse / state.mass;
596 size_t local_index = index - offset;
597 group.physics_state[local_index].velocity += impulse / group.physics_state[local_index].mass;
600 offset += group.collection->get_point_count();
617 "Cleared all force fields");
#define MF_ERROR(comp, ctx,...)
#define MF_WARN(comp, ctx,...)
#define MF_DEBUG(comp, ctx,...)
void apply_one_to_one(std::string_view param, const std::shared_ptr< NodeNetwork > &source) override
Apply ONE_TO_ONE parameter mapping.
Kinesis::SamplerBounds m_bounds
std::vector< PointVertex > extract_vertices() const
Extract current vertex data as PointVertex array.
size_t get_vertex_count() const override
Get number of vertices (may differ from point count for topology/path)
void apply_impulse(size_t index, const glm::vec3 &impulse)
Apply impulse to specific particle.
size_t get_point_count() const override
Get source point count (before topology expansion)
std::vector< CollectionGroup > m_collections
Kinesis::Stochastic::Stochastic m_random_generator
void mark_vertex_data_clean() override
Clear dirty flag after GPU upload.
void apply_per_particle_force(std::string_view param, const std::shared_ptr< NodeNetwork > &source)
std::optional< double > get_particle_velocity(size_t global_index) const
Get velocity magnitude for specific particle.
Kakshya::VertexLayout get_vertex_layout() const override
Get vertex layout describing vertex structure.
void set_attraction_point(const glm::vec3 &point)
std::vector< uint8_t > m_vertex_data_aggregate
std::span< const uint8_t > get_vertex_data() const override
Get vertex data for GPU upload.
void initialize_collections(const std::vector< std::vector< PointVertex > > &collections)
Initialize multiple physics collections.
bool is_vertex_data_dirty() const override
Check if geometry changed this frame.
@ BOUNCE
Reflect off boundaries with damping.
@ NONE
No bounds checking.
@ WRAP
Teleport to opposite side.
bool m_has_attraction_point
void apply_per_particle_mass(const std::shared_ptr< NodeNetwork > &source)
void add_collection(const std::vector< PointVertex > &vertices, float mass_multiplier=1.0F)
Add a single physics collection.
void set_bounds(const glm::vec3 &min, const glm::vec3 &max)
Set the simulation bounds.
float m_attraction_strength
float m_turbulence_strength
std::vector< Kinesis::VectorField > m_force_fields
bool m_spatial_interactions_enabled
void * get_data_at(size_t global_index) override
Get mutable access to point at global index.
void sync_to_point_collection()
void handle_boundary_conditions()
std::optional< double > query_state(std::string_view query) const override
Query operator internal state.
void process(float dt) override
Process for one batch cycle.
std::span< const uint8_t > get_vertex_data_for_collection(uint32_t idx) const override
Get vertex data for specific collection (if multiple)
void clear_force_fields()
Remove all external force fields.
float m_repulsion_strength
float m_interaction_radius
void add_force_field(Kinesis::VectorField field)
Add an external force field evaluated per-particle per-frame.
void apply_attraction_forces()
void apply_spatial_interactions()
glm::vec3 m_attraction_point
void set_parameter(std::string_view param, double value) override
Set operator parameter.
void apply_one_to_one(std::string_view param, const std::shared_ptr< NodeNetwork > &source) override
Apply ONE_TO_ONE parameter for physics-specific properties.
void initialize(const std::vector< PointVertex > &vertices)
Initialize with a single physics collection.
void apply_global_impulse(const glm::vec3 &impulse)
Apply impulse to all particles.
static std::optional< PhysicsParameter > string_to_parameter(std::string_view param)
@ NodeProcessing
Node graph processing (Nodes::NodeGraphManager)
@ Nodes
DSP Generator and Filter Nodes, graph pipeline, node management.
VectorField turbulence(float strength, Stochastic::Stochastic rng=Stochastic::Stochastic())
Uniform random force field using Stochastic infrastructure.
VectorField point_attractor(const glm::vec3 &anchor, float strength)
Radial attraction/repulsion toward an anchor point.
Complete description of vertex data layout in a buffer.
Typed, composable, stateless callable from domain D to range R.
std::shared_ptr< GpuSync::PointCollectionNode > collection
std::vector< PhysicsState > physics_state
Physics-specific data parallel to PointVertex array.