Files
PixelDefense/engine/libs/flecs/test/api/src/ComponentLifecycle.c

3101 lines
72 KiB
C

#include <api.h>
static ECS_COMPONENT_DECLARE(Position);
static ECS_COMPONENT_DECLARE(String);
static ECS_COMPONENT_DECLARE(Entity);
void ComponentLifecycle_setup(void) {
ecs_log_set_level(-3);
}
typedef struct xtor_ctx {
ecs_entity_t component;
size_t size;
int32_t count;
int32_t invoked;
} xtor_ctx;
typedef struct copy_ctx {
ecs_entity_t component;
size_t size;
int32_t count;
int32_t invoked;
} copy_ctx;
typedef struct cl_ctx {
xtor_ctx ctor;
xtor_ctx dtor;
copy_ctx copy;
copy_ctx move;
} cl_ctx;
static
void comp_ctor(
void *ptr,
int32_t count,
const ecs_type_info_t *info)
{
cl_ctx *data = info->hooks.ctx;
data->ctor.component = info->component;
data->ctor.size = info->size;
data->ctor.count += count;
data->ctor.invoked ++;
Position *p = ptr;
int i;
for (i = 0; i < count; i ++) {
p[i].x = 10;
p[i].y = 20;
}
}
static
void comp_dtor(
void *ptr,
int32_t count,
const ecs_type_info_t *info)
{
cl_ctx *data = info->hooks.ctx;
data->dtor.component = info->component;
data->dtor.size = info->size;
data->dtor.count += count;
data->dtor.invoked ++;
}
static
void comp_copy(
void *dst_ptr,
const void *src_ptr,
int32_t count,
const ecs_type_info_t *info)
{
cl_ctx *data = info->hooks.ctx;
data->copy.component = info->component;
data->copy.size = info->size;
data->copy.count += count;
data->copy.invoked ++;
memcpy(dst_ptr, src_ptr, info->size * count);
}
static
void comp_move(
void *dst_ptr,
void *src_ptr,
int32_t count,
const ecs_type_info_t *info)
{
cl_ctx *data = info->hooks.ctx;
data->move.component = info->component;
data->move.size = info->size;
data->move.count = count;
data->move.invoked ++;
memcpy(dst_ptr, src_ptr, info->size * count);
}
/* Position */
static int ctor_position = 0;
static
ECS_CTOR(Position, ptr, {
ptr->x = 0;
ptr->y = 0;
ctor_position ++;
})
static int dtor_position = 0;
static
ECS_DTOR(Position, ptr, {
dtor_position ++;
})
static int copy_position = 0;
static
ECS_COPY(Position, dst, src, {
copy_position ++;
})
static int move_position = 0;
static
ECS_MOVE(Position, dst, src, {
move_position ++;
})
static int on_add_position = 0;
static void ecs_on_add(Position)(ecs_iter_t *it) {
test_assert(it->count >= 1);
test_assert(it->event == EcsOnAdd);
Position *p = ecs_field(it, Position, 1);
for (int i = 0; i < it->count; i ++) {
on_add_position ++;
test_int(p[i].x, 0);
test_int(p[i].y, 0);
}
}
static void on_add_position_emplace(ecs_iter_t *it) {
on_add_position += it->count;
}
/* Velocity */
static int ctor_velocity = 0;
static
ECS_CTOR(Velocity, ptr, {
ctor_velocity ++;
})
static int dtor_velocity = 0;
static
ECS_DTOR(Velocity, ptr, {
dtor_velocity ++;
})
static int copy_velocity = 0;
static
ECS_COPY(Velocity, dst, src, {
copy_velocity ++;
})
static int move_velocity = 0;
static
ECS_MOVE(Velocity, dst, src, {
move_velocity ++;
})
/* Mass */
static int ctor_mass = 0;
static
ECS_CTOR(Mass, ptr, {
ctor_mass ++;
})
static int dtor_mass = 0;
static
ECS_DTOR(Mass, ptr, {
dtor_mass ++;
})
static int copy_mass = 0;
static
ECS_COPY(Mass, dst, src, {
copy_mass ++;
})
static int move_mass = 0;
static
ECS_MOVE(Mass, dst, src, {
move_mass ++;
})
/* Rotation */
static int ctor_rotation = 0;
static
ECS_CTOR(Rotation, ptr, {
ctor_rotation ++;
})
static int dtor_rotation = 0;
static
ECS_DTOR(Rotation, ptr, {
dtor_rotation ++;
})
static int copy_rotation = 0;
static
ECS_COPY(Rotation, dst, src, {
copy_rotation ++;
})
static int move_rotation = 0;
static
ECS_MOVE(Rotation, dst, src, {
move_rotation ++;
})
void ComponentLifecycle_ctor_on_add(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.ctor = comp_ctor,
.ctx = &ctx
});
ecs_entity_t e = ecs_new(world, 0);
test_int(ctx.ctor.invoked, 0);
ecs_add(world, e, Position);
test_assert(ctx.ctor.invoked != 0);
test_int(ctx.ctor.component, ecs_id(Position));
test_int(ctx.ctor.size, sizeof(Position));
test_int(ctx.ctor.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_ctor_on_new(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.ctor = comp_ctor,
.ctx = &ctx
});
ecs_new(world, Position);
test_assert(ctx.ctor.invoked != 0);
test_int(ctx.ctor.component, ecs_id(Position));
test_int(ctx.ctor.size, sizeof(Position));
test_int(ctx.ctor.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_dtor_on_remove(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.dtor = comp_dtor,
.ctx = &ctx
});
ecs_entity_t e = ecs_new(world, Position);
test_int(ctx.dtor.invoked, 0);
ecs_remove(world, e, Position);
test_assert(ctx.dtor.invoked != 0);
test_int(ctx.dtor.component, ecs_id(Position));
test_int(ctx.dtor.size, sizeof(Position));
test_int(ctx.dtor.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_dtor_on_delete(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.dtor = comp_dtor,
.ctx = &ctx
});
ecs_entity_t e = ecs_new(world, Position);
test_int(ctx.dtor.invoked, 0);
ecs_delete(world, e);
test_assert(ctx.dtor.invoked != 0);
test_int(ctx.dtor.component, ecs_id(Position));
test_int(ctx.dtor.size, sizeof(Position));
test_int(ctx.dtor.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_copy_on_set(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.copy = comp_copy,
.ctx = &ctx
});
ecs_entity_t e = ecs_new(world, 0);
test_int(ctx.copy.invoked, 0);
ecs_set(world, e, Position, {0, 0});
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Position));
test_int(ctx.copy.size, sizeof(Position));
test_int(ctx.copy.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_copy_on_override(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.copy = comp_copy,
.ctx = &ctx
});
ecs_entity_t base = ecs_new(world, Position);
test_int(ctx.copy.invoked, 0);
ecs_entity_t e = ecs_new_w_pair(world, EcsIsA, base);
test_int(ctx.copy.invoked, 0);
ecs_add(world, e, Position);
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Position));
test_int(ctx.copy.size, sizeof(Position));
test_int(ctx.copy.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_copy_on_clone(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.copy = comp_copy,
.ctx = &ctx
});
ecs_entity_t e = ecs_set(world, 0, Position, {10, 20});
test_assert(ctx.copy.invoked != 0);
memset(&ctx, 0, sizeof(ctx));
ecs_clone(world, 0, e, true);
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Position));
test_int(ctx.copy.size, sizeof(Position));
test_int(ctx.copy.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_no_copy_on_move(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Velocity);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.copy = comp_copy,
.ctx = &ctx
});
ecs_entity_t e = ecs_set(world, 0, Position, {10, 20});
test_assert(ctx.copy.invoked != 0);
memset(&ctx, 0, sizeof(ctx));
ecs_add(world, e, Velocity);
test_int(ctx.copy.invoked, 0);
ecs_fini(world);
}
void ComponentLifecycle_copy_on_snapshot(void) {
test_quarantine("13 March 2022");
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.copy = comp_copy,
.ctx = &ctx
});
const ecs_entity_t *ids = ecs_bulk_new(world, Position, 10);
test_assert(ids != NULL);
int32_t i;
for (i = 0; i < 10; i ++) {
test_assert(ecs_has(world, ids[i], Position));
ecs_set(world, ids[i], Position, {i, i * 2});
}
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Position));
test_int(ctx.copy.size, sizeof(Position));
test_int(ctx.copy.count, 10);
ctx = (cl_ctx){ { 0 } };
ecs_snapshot_t *s = ecs_snapshot_take(world);
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Position));
test_int(ctx.copy.size, sizeof(Position));
test_int(ctx.copy.count, 10);
ecs_snapshot_restore(world, s);
ecs_snapshot_free(s);
for (i = 0; i < 10; i ++) {
test_assert(ecs_has(world, ids[i], Position));
const Position *p = ecs_get(world, ids[i], Position);
test_assert(p != NULL);
test_int(p->x, i);
test_int(p->y, i * 2);
}
ecs_fini(world);
}
void ComponentLifecycle_ctor_copy_on_snapshot(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.ctor = comp_ctor,
.copy = comp_copy,
.ctx = &ctx
});
const ecs_entity_t *ids = ecs_bulk_new(world, Position, 10);
test_assert(ids != NULL);
int32_t i;
for (i = 0; i < 10; i ++) {
test_assert(ecs_has(world, ids[i], Position));
ecs_set(world, ids[i], Position, {i, i * 2});
}
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Position));
test_int(ctx.copy.size, sizeof(Position));
test_int(ctx.copy.count, 10);
ctx = (cl_ctx){ { 0 } };
ecs_snapshot_t *s = ecs_snapshot_take(world);
test_assert(ctx.ctor.invoked != 0);
test_int(ctx.ctor.component, ecs_id(Position));
test_int(ctx.ctor.size, sizeof(Position));
test_int(ctx.ctor.count, 10);
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Position));
test_int(ctx.copy.size, sizeof(Position));
test_int(ctx.copy.count, 10);
ecs_snapshot_restore(world, s);
for (i = 0; i < 10; i ++) {
test_assert(ecs_has(world, ids[i], Position));
const Position *p = ecs_get(world, ids[i], Position);
test_assert(p != NULL);
test_int(p->x, i);
test_int(p->y, i * 2);
}
ecs_fini(world);
}
void ComponentLifecycle_dtor_on_restore(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.dtor = comp_dtor,
.ctx = &ctx
});
const ecs_entity_t *temp_ids = ecs_bulk_new(world, Position, 10);
test_assert(temp_ids != NULL);
ecs_entity_t ids[10];
memcpy(ids, temp_ids, sizeof(ecs_entity_t) * 10);
int32_t i;
for (i = 0; i < 10; i ++) {
test_assert(ecs_has(world, ids[i], Position));
ecs_set(world, ids[i], Position, {i, i * 2});
}
test_int(ctx.dtor.invoked, 0);
ctx = (cl_ctx){ { 0 } };
ecs_snapshot_t *s = ecs_snapshot_take(world);
test_int(ctx.dtor.invoked, 0);
/* Delete one entity, so we have more confidence we're destructing the right
* entities */
ecs_delete(world, ids[0]);
test_assert(ctx.dtor.invoked != 0);
ctx = (cl_ctx){ { 0 } };
ecs_snapshot_restore(world, s);
test_assert(ctx.dtor.invoked != 0);
test_int(ctx.dtor.component, ecs_id(Position));
test_int(ctx.dtor.size, sizeof(Position));
test_int(ctx.dtor.count, 9);
for (i = 0; i < 10; i ++) {
test_assert(ecs_has(world, ids[i], Position));
const Position *p = ecs_get(world, ids[i], Position);
test_assert(p != NULL);
test_int(p->x, i);
test_int(p->y, i * 2);
}
ecs_fini(world);
}
void ComponentLifecycle_ctor_on_tag(void) {
install_test_abort();
ecs_world_t *world = ecs_mini();
ECS_TAG(world, Tag);
test_expect_abort();
ecs_set_hooks_id(world, Tag, &(ecs_type_hooks_t){
.ctor = comp_ctor
});
ecs_fini(world);
}
void ComponentLifecycle_dtor_on_tag(void) {
install_test_abort();
ecs_world_t *world = ecs_mini();
ECS_TAG(world, Tag);
test_expect_abort();
ecs_set_hooks_id(world, Tag, &(ecs_type_hooks_t){
.dtor = comp_dtor
});
ecs_fini(world);
}
void ComponentLifecycle_copy_on_tag(void) {
install_test_abort();
ecs_world_t *world = ecs_mini();
ECS_TAG(world, Tag);
test_expect_abort();
ecs_set_hooks_id(world, Tag, &(ecs_type_hooks_t){
.copy = comp_copy
});
ecs_fini(world);
}
void ComponentLifecycle_move_on_tag(void) {
install_test_abort();
ecs_world_t *world = ecs_mini();
ECS_TAG(world, Tag);
test_expect_abort();
ecs_set_hooks_id(world, Tag, &(ecs_type_hooks_t){
.move = comp_move
});
ecs_fini(world);
}
void ComponentLifecycle_merge_to_different_table(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Velocity);
ECS_COMPONENT(world, Mass);
ECS_COMPONENT(world, Rotation);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.dtor = ecs_dtor(Position),
.copy = ecs_copy(Position),
.move = ecs_move(Position)
});
ecs_set_hooks(world, Velocity, {
.ctor = ecs_ctor(Velocity),
.dtor = ecs_dtor(Velocity),
.copy = ecs_copy(Velocity),
.move = ecs_move(Velocity)
});
ecs_set_hooks(world, Mass, {
.ctor = ecs_ctor(Mass),
.dtor = ecs_dtor(Mass),
.copy = ecs_copy(Mass),
.move = ecs_move(Mass)
});
ecs_set_hooks(world, Rotation, {
.ctor = ecs_ctor(Rotation),
.dtor = ecs_dtor(Rotation),
.copy = ecs_copy(Rotation),
.move = ecs_move(Rotation)
});
ECS_ENTITY(world, e, Position, Velocity, Rotation);
ctor_position = 0;
dtor_position = 0;
copy_position = 0;
move_position = 0;
ctor_velocity = 0;
dtor_velocity = 0;
copy_velocity = 0;
move_velocity = 0;
ctor_rotation = 0;
dtor_rotation = 0;
copy_rotation = 0;
move_rotation = 0;
ecs_defer_begin(world);
ecs_remove(world, e, Position);
test_int(ctor_position, 0);
test_int(dtor_position, 0);
test_int(copy_position, 0);
test_int(move_position, 0);
test_int(ctor_velocity, 0);
test_int(dtor_velocity, 0);
test_int(copy_velocity, 0);
test_int(move_velocity, 0);
test_int(ctor_rotation, 0);
test_int(dtor_rotation, 0);
test_int(copy_rotation, 0);
test_int(move_rotation, 0);
ecs_add(world, e, Mass);
test_int(ctor_mass, 0);
test_int(ctor_velocity, 0);
test_int(dtor_velocity, 0);
test_int(copy_velocity, 0);
test_int(move_velocity, 0);
test_int(ctor_rotation, 0);
test_int(dtor_rotation, 0);
test_int(copy_rotation, 0);
test_int(move_rotation, 0);
ecs_remove(world, e, Rotation);
test_int(ctor_rotation, 0);
test_int(dtor_rotation, 0);
test_int(copy_rotation, 0);
test_int(move_rotation, 0);
test_int(ctor_velocity, 0);
test_int(dtor_velocity, 0);
test_int(copy_velocity, 0);
test_int(move_velocity, 0);
ecs_defer_end(world);
test_assert(!ecs_has(world, e, Position));
test_assert(ecs_has(world, e, Velocity));
test_assert(ecs_has(world, e, Mass));
test_assert(!ecs_has(world, e, Rotation));
test_int(ctor_position, 0);
test_int(dtor_position, 1); // removed first, no moves
test_int(copy_position, 0);
test_int(move_position, 0);
test_assert(ctor_velocity != 0);
test_assert(dtor_velocity != 0);
test_int(copy_velocity, 0);
test_assert(move_velocity != 0);
test_assert(ctor_rotation == 0);
test_assert(dtor_rotation != 0);
test_int(copy_rotation, 0);
test_assert(move_rotation == 0);
test_assert(ctor_mass != 0); // got added, moved once
test_assert(dtor_mass == 0);
test_int(copy_mass, 0);
test_assert(move_mass == 0);
ecs_fini(world);
}
void ComponentLifecycle_merge_to_new_table(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_entity_t e = ecs_new(world, 0);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.dtor = ecs_dtor(Position),
.copy = ecs_copy(Position),
.move = ecs_move(Position)
});
ecs_defer_begin(world);
ecs_add(world, e, Position);
ecs_defer_end(world);
test_int(ctor_position, 1);
test_int(move_position, 0);
ecs_fini(world);
}
void ComponentLifecycle_delete_in_stage(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Velocity);
ECS_COMPONENT(world, Mass);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.dtor = ecs_dtor(Position),
.copy = ecs_copy(Position),
.move = ecs_move(Position)
});
ecs_set_hooks(world, Velocity, {
.ctor = ecs_ctor(Velocity),
.dtor = ecs_dtor(Velocity),
.copy = ecs_copy(Velocity),
.move = ecs_move(Velocity)
});
ecs_set_hooks(world, Mass, {
.ctor = ecs_ctor(Mass),
.dtor = ecs_dtor(Mass),
.copy = ecs_copy(Mass),
.move = ecs_move(Mass)
});
ECS_ENTITY(world, e, Position, Velocity, Mass);
ctor_position = 0;
dtor_position = 0;
copy_position = 0;
move_position = 0;
ctor_velocity = 0;
dtor_velocity = 0;
copy_velocity = 0;
move_velocity = 0;
ctor_mass = 0;
dtor_mass = 0;
copy_mass = 0;
move_mass = 0;
ecs_defer_begin(world);
/* None of the components should be destructed while in the stage as they
* were never copied to the stage */
ecs_delete(world, e);
test_int(dtor_position, 0);
test_int(dtor_velocity, 0);
test_int(dtor_mass, 0);
ecs_defer_end(world);
test_assert(ecs_exists(world, e));
test_assert(!ecs_is_alive(world, e));
/* Position is destroyed in main stage */
test_int(ctor_position, 0);
test_int(dtor_position, 1);
test_int(copy_position, 0);
test_int(move_position, 0);
/* Velocity is destroyed in main stage */
test_int(ctor_velocity, 0);
test_int(dtor_velocity, 1);
test_int(copy_velocity, 0);
test_int(move_velocity, 0);
/* Mass is destroyed in main stage */
test_int(ctor_mass, 0);
test_int(dtor_mass, 1);
test_int(copy_mass, 0);
test_int(move_mass, 0);
ecs_fini(world);
}
typedef struct Pair {
float value_1;
float value_2;
} Pair;
void ComponentLifecycle_ctor_on_add_pair(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Pair);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Pair, {
.ctor = comp_ctor,
.ctx = &ctx
});
ecs_entity_t e = ecs_new(world, 0);
test_int(ctx.ctor.invoked, 0);
ecs_add_pair(world, e, ecs_id(Pair), ecs_id(Position));
test_assert(ctx.ctor.invoked != 0);
test_int(ctx.ctor.component, ecs_id(Pair));
test_int(ctx.ctor.size, sizeof(Pair));
test_int(ctx.ctor.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_ctor_on_add_pair_tag(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_TAG(world, Pair);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.ctor = comp_ctor,
.ctx = &ctx
});
ecs_entity_t e = ecs_new(world, 0);
test_int(ctx.ctor.invoked, 0);
ecs_add_pair(world, e, Pair, ecs_id(Position));
test_assert(ctx.ctor.invoked != 0);
test_int(ctx.ctor.component, ecs_id(Position));
test_int(ctx.ctor.size, sizeof(Position));
test_int(ctx.ctor.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_ctor_on_move_pair(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Pair);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Pair, {
.ctor = comp_ctor,
.ctx = &ctx
});
/* Create entity in existing table */
ecs_entity_t e = ecs_new(world, Position);
test_int(ctx.ctor.invoked, 0);
/* Add pair to existing table */
ecs_add_pair(world, e, ecs_id(Pair), ecs_id(Position));
test_assert(ctx.ctor.invoked != 0);
test_int(ctx.ctor.component, ecs_id(Pair));
test_int(ctx.ctor.size, sizeof(Pair));
test_int(ctx.ctor.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_move_on_realloc(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.ctor = comp_ctor,
.move = comp_move,
.ctx = &ctx
});
ecs_entity_t e = ecs_new(world, 0);
test_int(ctx.ctor.invoked, 0);
ecs_add(world, e, Position);
test_assert(ctx.ctor.invoked != 0);
test_int(ctx.move.invoked, 0);
test_int(ctx.ctor.component, ecs_id(Position));
test_int(ctx.ctor.size, sizeof(Position));
test_int(ctx.ctor.count, 1);
ctx = (cl_ctx){ { 0 } };
/* Trigger realloc & move */
ecs_new(world, Position);
ecs_new(world, Position);
test_assert(ctx.ctor.invoked != 0);
test_assert(ctx.move.invoked != 0);
ecs_fini(world);
}
void ComponentLifecycle_move_on_bulk_new(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.ctor = comp_ctor,
.move = comp_move,
.ctx = &ctx
});
ecs_entity_t e = ecs_new(world, 0);
test_int(ctx.ctor.invoked, 0);
ecs_add(world, e, Position);
test_assert(ctx.ctor.invoked != 0);
test_int(ctx.move.invoked, 0);
test_int(ctx.ctor.component, ecs_id(Position));
test_int(ctx.ctor.size, sizeof(Position));
test_int(ctx.ctor.count, 1);
ctx = (cl_ctx){ { 0 } };
ecs_bulk_new(world, Position, 1000);
test_assert(ctx.ctor.invoked != 0);
test_assert(ctx.move.invoked != 0);
ecs_fini(world);
}
void ComponentLifecycle_on_add_on_bulk_new(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.on_add = ecs_on_add(Position)
});
ecs_bulk_new(world, Position, 1000);
test_int(ctor_position, 1000);
test_int(on_add_position, 1000);
ecs_fini(world);
}
void ComponentLifecycle_move_on_delete(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.ctor = comp_ctor,
.move = comp_move,
.ctx = &ctx
});
ecs_entity_t e1 = ecs_new(world, Position);
ecs_new(world, Position);
test_assert(ctx.ctor.invoked != 0);
ctx = (cl_ctx){ { 0 } };
ecs_delete(world, e1); /* Should trigger move of e2 */
test_int(ctx.ctor.invoked, 0);
test_assert(ctx.move.invoked != 0);
test_int(ctx.move.component, ecs_id(Position));
test_int(ctx.move.size, sizeof(Position));
test_int(ctx.move.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_move_dtor_on_delete(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.dtor = comp_dtor,
.move = comp_move,
.ctx = &ctx
});
ecs_entity_t e1 = ecs_new(world, Position);
ecs_new(world, Position);
ctx = (cl_ctx){ { 0 } };
ecs_delete(world, e1); /* Should trigger move of e2 */
test_assert(ctx.dtor.invoked != 0);
test_assert(ctx.move.invoked != 0);
test_int(ctx.dtor.component, ecs_id(Position));
test_int(ctx.dtor.size, sizeof(Position));
test_int(ctx.dtor.count, 1);
test_int(ctx.move.component, ecs_id(Position));
test_int(ctx.move.size, sizeof(Position));
test_int(ctx.move.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_copy_on_override_pair(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Pair);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Pair, {
.ctor = comp_ctor,
.copy = comp_copy,
.ctx = &ctx
});
ecs_entity_t base = ecs_new(world, 0);
ecs_add_pair(world, base, ecs_id(Pair), ecs_id(Position));
test_assert(ctx.ctor.invoked != 0);
test_int(ctx.copy.invoked, 0);
ecs_entity_t instance = ecs_new_w_pair(world, EcsIsA, base);
test_assert(ctx.ctor.invoked != 0); /* No change */
test_int(ctx.copy.invoked, 0);
ctx = (cl_ctx){ { 0 } };
/* Override */
ecs_add_pair(world, instance, ecs_id(Pair), ecs_id(Position));
test_assert(ctx.ctor.invoked != 0);
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Pair));
test_int(ctx.copy.size, sizeof(Pair));
test_int(ctx.copy.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_copy_on_override_pair_tag(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_TAG(world, Pair);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.ctor = comp_ctor,
.copy = comp_copy,
.ctx = &ctx
});
ecs_entity_t base = ecs_new(world, 0);
ecs_add_pair(world, base, Pair, ecs_id(Position));
test_assert(ctx.ctor.invoked != 0);
test_int(ctx.copy.invoked, 0);
ecs_entity_t instance = ecs_new_w_pair(world, EcsIsA, base);
test_assert(ctx.ctor.invoked != 0); /* No change */
test_int(ctx.copy.invoked, 0);
ctx = (cl_ctx){ { 0 } };
/* Override */
ecs_add_pair(world, instance, Pair, ecs_id(Position));
test_assert(ctx.ctor.invoked != 0);
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Position));
test_int(ctx.copy.size, sizeof(Position));
test_int(ctx.copy.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_copy_on_set_pair(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Pair);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Pair, {
.copy = comp_copy,
.ctx = &ctx
});
ecs_entity_t e = ecs_new(world, 0);
test_int(ctx.copy.invoked, 0);
ecs_set_pair(world, e, Pair, ecs_id(Position), {0, 0});
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Pair));
test_int(ctx.copy.size, sizeof(Pair));
test_int(ctx.copy.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_copy_on_set_pair_tag(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_TAG(world, Pair);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.copy = comp_copy,
.ctx = &ctx
});
ecs_entity_t e = ecs_new(world, 0);
test_int(ctx.copy.invoked, 0);
ecs_set_pair_object(world, e, Pair, Position, {0, 0});
test_assert(ctx.copy.invoked != 0);
test_int(ctx.copy.component, ecs_id(Position));
test_int(ctx.copy.size, sizeof(Position));
test_int(ctx.copy.count, 1);
ecs_fini(world);
}
void ComponentLifecycle_allow_lifecycle_overwrite_equal_callbacks(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position)
});
/* Same actions, should be ok */
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position)
});
ecs_new(world, Position);
test_int(ctor_position, 1);
ecs_fini(world);
}
static
void AddPosition(ecs_iter_t *it) { }
void ComponentLifecycle_set_lifecycle_after_trigger(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_OBSERVER(world, AddPosition, EcsOnAdd, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position)
});
ecs_new(world, Position);
test_int(ctor_position, 1);
ecs_fini(world);
}
static int dummy_dtor_invoked = 0;
typedef struct Dummy {
ecs_world_t *world;
ecs_entity_t e;
int value;
} Dummy;
static
void dummy_dtor(
void *ptr,
int32_t count,
const ecs_type_info_t *info)
{
Dummy *d = ptr;
int i;
for (i = 0; i < count; i ++) {
test_assert(ecs_is_valid(d[i].world, d[i].e));
}
dummy_dtor_invoked ++;
}
void ComponentLifecycle_valid_entity_in_dtor_after_delete(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Dummy);
ecs_set_hooks(world, Dummy, {
.dtor = dummy_dtor
});
ecs_entity_t e = ecs_new(world, Dummy);
test_assert(e != 0);
ecs_set(world, e, Dummy, {world, e, 0});
ecs_delete(world, e);
test_assert(!ecs_is_valid(world, e));
test_assert(!ecs_is_alive(world, e));
test_int(dummy_dtor_invoked, 1);
ecs_fini(world);
}
void ComponentLifecycle_ctor_w_emplace(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.ctor = comp_ctor,
.ctx = &ctx
});
ecs_entity_t e = ecs_new_id(world);
test_assert(e != 0);
const Position *ptr = ecs_emplace(world, e, Position);
test_assert(ptr != NULL);
test_int(ctx.ctor.invoked, 0);
ecs_fini(world);
}
void ComponentLifecycle_ctor_w_emplace_defer(void) {
ecs_world_t* world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position)
});
test_int(ctor_position, 0);
ecs_entity_t e = ecs_new_id(world);
test_assert(e != 0);
ecs_defer_begin(world);
Position *ptr = ecs_emplace(world, e, Position);
test_assert(ptr != NULL);
test_int(ctor_position, 0);
ptr->x = 10;
ptr->y = 20;
test_assert(!ecs_has(world, e, Position));
test_int(ctor_position, 0);
ecs_defer_end(world);
test_assert(ecs_has(world, e, Position));
test_int(ctor_position, 0);
const Position *p = ecs_get(world, e, Position);
test_assert(p != NULL);
test_int(p->x, 10);
test_int(p->y, 20);
ecs_fini(world);
}
void ComponentLifecycle_on_add_w_emplace(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_add = on_add_position_emplace
});
ecs_entity_t e = ecs_new_id(world);
test_assert(e != 0);
test_int(on_add_position, 0);
const Position *ptr = ecs_emplace(world, e, Position);
test_assert(ptr != NULL);
test_int(on_add_position, 1);
ecs_fini(world);
}
void ComponentLifecycle_on_add_w_emplace_existing(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Velocity);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.on_add = on_add_position_emplace
});
ecs_entity_t e = ecs_new(world, Velocity);
test_assert(e != 0);
test_int(ctor_position, 0);
test_int(on_add_position, 0);
const Position *ptr = ecs_emplace(world, e, Position);
test_assert(ptr != NULL);
test_int(ctor_position, 0);
test_int(on_add_position, 1);
ecs_fini(world);
}
void ComponentLifecycle_on_add_w_emplace_defer(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_add = on_add_position_emplace
});
ecs_entity_t e = ecs_new_id(world);
test_assert(e != 0);
ecs_defer_begin(world);
test_int(on_add_position, 0);
const Position *ptr = ecs_emplace(world, e, Position);
test_assert(ptr != NULL);
test_int(on_add_position, 0);
ecs_defer_end(world);
test_int(on_add_position, 1);
ecs_fini(world);
}
static int move_ctor_position = 0;
static
void position_move_ctor(
void *dst,
void *src,
int32_t count,
const ecs_type_info_t *info)
{
*((Position*)dst) = *((Position*)src);
move_ctor_position ++;
}
void ComponentLifecycle_ctor_w_emplace_defer_use_move_ctor(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.move = ecs_move(Position),
.move_ctor = position_move_ctor
});
ecs_entity_t e = ecs_new_id(world);
test_assert(e != 0);
ecs_defer_begin(world);
test_int(on_add_position, 0);
Position *ptr = ecs_emplace(world, e, Position);
ptr->x = 10;
ptr->y = 20;
test_assert(ptr != NULL);
test_int(ctor_position, 0);
test_int(move_position, 0);
test_int(move_ctor_position, 0);
ecs_defer_end(world);
test_int(ctor_position, 0);
test_int(move_position, 0);
test_int(move_ctor_position, 1);
const Position *p = ecs_get(world, e, Position);
test_int(p->x, 10);
test_int(p->y, 20);
ecs_fini(world);
}
void ComponentLifecycle_merge_async_stage_w_emplace(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.copy = ecs_copy(Position),
.move_ctor = position_move_ctor,
.dtor = ecs_dtor(Position)
});
ecs_entity_t e = ecs_new_id(world);
ecs_world_t *async = ecs_async_stage_new(world);
Position *p = ecs_emplace(async, e, Position);
test_assert(!ecs_has(world, e, Position));
test_int(ctor_position, 0);
test_int(copy_position, 0);
test_int(move_position, 0);
test_int(move_ctor_position, 0);
test_int(dtor_position, 0);
p->x = 10;
p->y = 20;
ecs_merge(async);
test_assert(ecs_has(world, e, Position));
test_int(ctor_position, 0);
test_int(copy_position, 0);
test_int(move_position, 0);
test_int(move_ctor_position, 1);
test_int(dtor_position, 1);
const Position *ptr = ecs_get(world, e, Position);
test_int(ptr->x, 10);
test_int(ptr->y, 20);
ecs_async_stage_free(async);
ecs_fini(world);
}
void ComponentLifecycle_merge_async_stage_w_emplace_to_deferred_world(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.copy = ecs_copy(Position)
});
ecs_entity_t e = ecs_new_id(world);
ecs_world_t *async = ecs_async_stage_new(world);
Position *p = ecs_emplace(async, e, Position);
test_assert(!ecs_has(world, e, Position));
test_int(ctor_position, 0);
p->x = 10;
p->y = 20;
ecs_defer_begin(world);
ecs_merge(async);
test_assert(!ecs_has(world, e, Position));
test_int(ctor_position, 0);
ecs_defer_end(world);
test_assert(ecs_has(world, e, Position));
test_int(ctor_position, 0);
const Position *ptr = ecs_get(world, e, Position);
test_int(ptr->x, 10);
test_int(ptr->y, 20);
ecs_async_stage_free(async);
ecs_fini(world);
}
static void invalid_ctor(void *ptr, int count, const ecs_type_info_t *ti) {
test_assert(false);
}
void ComponentLifecycle_emplace_grow_w_existing_component(void) {
test_quarantine("5 Feb 2023");
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Velocity);
ecs_set_hooks(world, Position, {
.ctor = invalid_ctor,
.copy = ecs_copy(Position),
.move = ecs_move(Position),
.dtor = ecs_dtor(Position)
});
ecs_entity_t e1 = ecs_new(world, Velocity);
ecs_entity_t e2 = ecs_new(world, Velocity);
ecs_entity_t e3 = ecs_new(world, Velocity);
{
Position *p = ecs_emplace(world, e1, Position);
p->x = 10;
p->y = 20;
}
{
Position *p = ecs_emplace(world, e2, Position);
p->x = 30;
p->y = 40;
}
{
Position *p = ecs_emplace(world, e3, Position);
p->x = 50;
p->y = 60;
}
test_assert(ecs_has(world, e1, Position));
test_assert(ecs_has(world, e2, Position));
test_assert(ecs_has(world, e3, Position));
{
const Position *p = ecs_get(world, e1, Position);
test_assert(p != NULL);
test_int(p->x, 10);
test_int(p->y, 20);
}
{
const Position *p = ecs_get(world, e2, Position);
test_assert(p != NULL);
test_int(p->x, 30);
test_int(p->y, 40);
}
{
const Position *p = ecs_get(world, e3, Position);
test_assert(p != NULL);
test_int(p->x, 50);
test_int(p->y, 60);
}
ecs_fini(world);
}
void ComponentLifecycle_dtor_on_fini(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Dummy);
ecs_set_hooks(world, Dummy, {
.dtor = dummy_dtor
});
ecs_entity_t e = ecs_new(world, Dummy);
test_assert(e != 0);
ecs_set(world, e, Dummy, {world, e});
test_int(dummy_dtor_invoked, 0);
ecs_fini(world);
test_int(dummy_dtor_invoked, 1);
}
static
void type_dtor(
void *ptr,
int32_t count,
const ecs_type_info_t *info)
{
test_int(count, 1);
Dummy *d = ptr;
ecs_world_t *world = d->world;
ecs_entity_t e = d->e;
test_assert(world != NULL);
test_assert(e != 0);
const ecs_type_t *type = ecs_get_type(world, e);
test_assert(type != NULL);
test_int(type->count, 3);
dummy_dtor_invoked = 1;
}
typedef struct {
ecs_world_t *world;
ecs_entity_t e;
ecs_entity_t other;
} Entity;
typedef struct {
ecs_world_t *world;
ecs_entity_t e;
char *str;
} String;
static int other_dtor_invoked;
static int other_dtor_valid_entity;
static
void other_type_dtor(
void *ptr,
int32_t count,
const ecs_type_info_t *info)
{
test_int(count, 1);
test_assert(ptr != NULL);
Entity *comp = ptr;
ecs_world_t *world = comp->world;
ecs_entity_t e = comp->e;
test_assert(e != 0);
test_assert(world != NULL);
test_assert(ecs_is_valid(world, e));
test_assert(comp->other != 0);
if (ecs_is_valid(world, comp->other)) {
const ecs_type_t *type = ecs_get_type(world, comp->other);
test_assert(type != NULL);
test_int(type->count, 2);
other_dtor_valid_entity ++;
}
other_dtor_invoked ++;
}
static
void other_comp_dtor(
void *ptr,
int32_t count,
const ecs_type_info_t *info)
{
test_int(count, 1);
test_assert(ptr != NULL);
Entity *comp = ptr;
ecs_world_t *world = comp->world;
ecs_entity_t e = comp->e;
test_assert(e != 0);
test_assert(ecs_is_valid(world, e));
test_assert(comp->other != 0);
if (ecs_is_valid(world, comp->other)) {
if (ecs_has(world, comp->other, String)) {
const String *str_ptr = ecs_get(world, comp->other, String);
test_assert(str_ptr != NULL);
test_assert(str_ptr->str != NULL);
test_str(str_ptr->str, "Foo");
other_dtor_valid_entity ++;
} else {
test_assert(ecs_get(world, comp->other, String) == NULL);
}
}
other_dtor_invoked ++;
}
ECS_MOVE(Entity, dst, src, {
dst->world = src->world;
dst->e = src->e;
dst->other = src->other;
src->world = NULL;
src->e = 0;
src->other = 0;
})
static
void other_delete_dtor(
void *ptr,
int32_t count,
const ecs_type_info_t *info)
{
test_int(count, 1);
test_assert(ptr != NULL);
Entity *comp = ptr;
ecs_world_t *world = comp->world;
ecs_entity_t e = comp->e;
test_assert(e != 0);
test_assert(ecs_is_valid(world, e));
test_assert(comp->other != 0);
if (ecs_is_valid(world, comp->other)) {
ecs_delete(world, comp->other);
other_dtor_valid_entity ++;
test_assert(ecs_is_valid(world, e));
test_assert(ecs_get(world, e, Entity) == comp);
test_assert(ecs_is_valid(world, comp->other));
}
other_dtor_invoked ++;
}
static
void self_delete_dtor(
void *ptr,
int32_t count,
const ecs_type_info_t *info)
{
test_int(count, 1);
test_assert(ptr != NULL);
Dummy *d = ptr;
ecs_world_t *world = d->world;
ecs_entity_t e = d->e;
test_assert(world != 0);
test_assert(e != 0);
if (ecs_is_valid(world, e)) {
ecs_delete(world, e);
// Delete should be deferred
test_assert(ecs_is_valid(world, e));
}
dummy_dtor_invoked ++;
}
static
void string_dtor(
void *ptr,
int32_t count,
const ecs_type_info_t *info)
{
test_int(count, 1);
test_assert(ptr != NULL);
String *comp = ptr;
ecs_world_t *world = comp->world;
ecs_entity_t e = comp->e;
test_assert(e != 0);
test_assert(ecs_is_valid(world, e));
test_assert(comp->str != NULL);
ecs_os_free(comp->str);
}
void ComponentLifecycle_valid_type_in_dtor_on_fini(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Velocity);
ECS_COMPONENT(world, Dummy);
ecs_set_hooks(world, Dummy, {
.dtor = type_dtor
});
ecs_entity_t e = ecs_new(world, Dummy);
test_assert(e != 0);
ecs_set(world, e, Dummy, {world, e});
ecs_add(world, e, Position);
ecs_add(world, e, Velocity);
test_int(dummy_dtor_invoked, 0);
ecs_fini(world);
test_int(dummy_dtor_invoked, 1);
}
void ComponentLifecycle_valid_other_type_of_entity_in_dtor_on_fini(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Velocity);
ECS_COMPONENT_DEFINE(world, Entity);
ecs_set_hooks(world, Entity, {
.dtor = other_type_dtor
});
ecs_entity_t e1 = ecs_new_id(world);
ecs_entity_t e2 = ecs_new(world, Position);
ecs_set(world, e2, Entity, {world, e2, e1});
ecs_add(world, e1, Velocity);
ecs_set(world, e1, Entity, {world, e1, e2});
test_int(other_dtor_invoked, 0);
ecs_fini(world);
test_int(other_dtor_invoked, 2);
test_int(other_dtor_valid_entity, 1);
}
void ComponentLifecycle_delete_in_dtor_other_type_on_fini(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT_DEFINE(world, String);
ECS_COMPONENT_DEFINE(world, Entity);
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Velocity);
ecs_set_hooks(world, String, {
.dtor = string_dtor
});
ecs_set_hooks(world, Entity, {
.dtor = other_delete_dtor
});
ecs_entity_t e1 = ecs_new(world, Position);
ecs_entity_t e2 = ecs_new(world, Velocity);
ecs_set(world, e2, Entity, {world, e2, e1});
ecs_set(world, e1, Entity, {world, e1, e2});
ecs_set(world, e1, String, {world, e1, ecs_os_strdup("Foo")});
ecs_set(world, e2, String, {world, e2, ecs_os_strdup("Foo")});
test_int(other_dtor_invoked, 0);
ecs_fini(world);
test_int(other_dtor_invoked, 2);
test_int(other_dtor_valid_entity, 1);
}
void ComponentLifecycle_delete_in_dtor_other_type_on_delete_parent(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT_DEFINE(world, String);
ECS_COMPONENT_DEFINE(world, Entity);
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Velocity);
ecs_set_hooks(world, String, {
.dtor = string_dtor
});
ecs_set_hooks(world, Entity, {
.dtor = other_delete_dtor
});
ecs_entity_t parent = ecs_new_id(world);
ecs_entity_t e1 = ecs_new(world, Position);
ecs_entity_t e2 = ecs_new(world, Velocity);
ecs_add_pair(world, e1, EcsChildOf, parent);
ecs_add_pair(world, e2, EcsChildOf, parent);
ecs_set(world, e2, Entity, {world, e2, e1});
ecs_set(world, e1, Entity, {world, e1, e2});
ecs_set(world, e1, String, {world, e1, ecs_os_strdup("Foo")});
ecs_set(world, e2, String, {world, e2, ecs_os_strdup("Foo")});
test_int(other_dtor_invoked, 0);
ecs_delete(world, parent);
test_int(other_dtor_invoked, 2);
test_int(other_dtor_valid_entity, 1);
test_assert(!ecs_is_alive(world, e1));
test_assert(!ecs_is_alive(world, e2));
ecs_fini(world);
}
void ComponentLifecycle_delete_in_dtor_other_type_on_delete(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT_DEFINE(world, String);
ECS_COMPONENT_DEFINE(world, Entity);
ECS_COMPONENT(world, Position);
ECS_COMPONENT(world, Velocity);
ecs_set_hooks(world, String, {
.dtor = string_dtor
});
ecs_set_hooks(world, Entity, {
.dtor = other_delete_dtor
});
ecs_entity_t e1 = ecs_new(world, Position);
ecs_entity_t e2 = ecs_new(world, Velocity);
ecs_set(world, e2, Entity, {world, e2, e1});
ecs_set(world, e1, Entity, {world, e1, e2});
ecs_set(world, e1, String, {world, e1, ecs_os_strdup("Foo")});
ecs_set(world, e2, String, {world, e2, ecs_os_strdup("Foo")});
test_int(other_dtor_invoked, 0);
ecs_delete(world, e1);
ecs_delete(world, e2);
test_int(other_dtor_invoked, 2);
test_int(other_dtor_valid_entity, 1);
ecs_fini(world);
}
void ComponentLifecycle_delete_self_in_dtor_on_delete(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Dummy);
ecs_set_hooks(world, Dummy, {
.dtor = self_delete_dtor
});
ecs_entity_t e1 = ecs_new(world, Dummy);
ecs_entity_t e2 = ecs_new(world, Dummy);
ecs_entity_t e3 = ecs_new(world, Dummy);
ecs_set(world, e1, Dummy, {world, e1});
ecs_set(world, e2, Dummy, {world, e2});
ecs_set(world, e3, Dummy, {world, e3});
test_int(dummy_dtor_invoked, 0);
ecs_delete(world, e1);
ecs_delete(world, e2);
ecs_delete(world, e3);
test_int(dummy_dtor_invoked, 3);
ecs_fini(world);
}
static int position_on_set_invoked;
static int velocity_on_set_invoked;
ECS_ON_SET(Position, ptr, {
ptr->x += 1;
ptr->y += 2;
position_on_set_invoked ++;
})
ECS_ON_SET(Velocity, ptr, {
ptr->x += 1;
ptr->y += 2;
velocity_on_set_invoked ++;
})
void ComponentLifecycle_on_set_after_set(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.on_set = ecs_on_set(Position)
});
ecs_entity_t e = ecs_new(world, Position);
test_int(ctor_position, 1);
test_int(position_on_set_invoked, 0);
const Position *p = ecs_get(world, e, Position);
test_assert(p != NULL);
test_int(p->x, 0);
test_int(p->y, 0);
ecs_set(world, e, Position, {10, 20});
test_int(ctor_position, 1);
test_int(position_on_set_invoked, 1);
test_assert(p == ecs_get(world, e, Position));
test_int(p->x, 11);
test_int(p->y, 22);
ecs_fini(world);
}
void ComponentLifecycle_on_add_after_new(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_default_ctor,
.on_add = ecs_on_add(Position)
});
ecs_new(world, Position);
test_int(on_add_position, 1);
ecs_new(world, Position);
test_int(on_add_position, 2);
ecs_new(world, Position);
test_int(on_add_position, 3);
ecs_fini(world);
}
void ComponentLifecycle_on_add_after_add(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_default_ctor,
.on_add = ecs_on_add(Position)
});
ecs_entity_t e1 = ecs_new_id(world);
ecs_entity_t e2 = ecs_new_id(world);
ecs_entity_t e3 = ecs_new_id(world);
ecs_add(world, e1, Position);
test_int(on_add_position, 1);
ecs_add(world, e2, Position);
test_int(on_add_position, 2);
ecs_add(world, e3, Position);
test_int(on_add_position, 3);
ecs_fini(world);
}
void ComponentLifecycle_on_add_after_set(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_default_ctor,
.on_add = ecs_on_add(Position)
});
ecs_set(world, 0, Position, {10, 20});
test_int(on_add_position, 1);
ecs_set(world, 0, Position, {10, 20});
test_int(on_add_position, 2);
ecs_set(world, 0, Position, {10, 20});
test_int(on_add_position, 3);
ecs_fini(world);
}
static int on_remove_position = 0;
static void ecs_on_remove(Position)(ecs_iter_t *it) {
test_assert(it->count >= 1);
test_assert(it->event == EcsOnRemove);
Position *p = ecs_field(it, Position, 1);
for (int i = 0; i < it->count; i ++) {
on_remove_position ++;
test_int(p[i].x, 10);
test_int(p[i].y, 20);
}
}
void ComponentLifecycle_on_remove_after_remove(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_remove = ecs_on_remove(Position)
});
ecs_entity_t e1 = ecs_set(world, 0, Position, {10, 20});
ecs_entity_t e2 = ecs_set(world, 0, Position, {10, 20});
ecs_entity_t e3 = ecs_set(world, 0, Position, {10, 20});
ecs_remove(world, e1, Position);
test_int(on_remove_position, 1);
ecs_remove(world, e2, Position);
test_int(on_remove_position, 2);
ecs_remove(world, e3, Position);
test_int(on_remove_position, 3);
ecs_fini(world);
}
void ComponentLifecycle_on_remove_after_clear(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_remove = ecs_on_remove(Position)
});
ecs_entity_t e1 = ecs_set(world, 0, Position, {10, 20});
ecs_entity_t e2 = ecs_set(world, 0, Position, {10, 20});
ecs_entity_t e3 = ecs_set(world, 0, Position, {10, 20});
ecs_clear(world, e1);
test_int(on_remove_position, 1);
ecs_clear(world, e2);
test_int(on_remove_position, 2);
ecs_clear(world, e3);
test_int(on_remove_position, 3);
ecs_fini(world);
}
void ComponentLifecycle_on_remove_after_delete(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_remove = ecs_on_remove(Position)
});
ecs_entity_t e1 = ecs_set(world, 0, Position, {10, 20});
ecs_entity_t e2 = ecs_set(world, 0, Position, {10, 20});
ecs_entity_t e3 = ecs_set(world, 0, Position, {10, 20});
ecs_delete(world, e1);
test_int(on_remove_position, 1);
ecs_delete(world, e2);
test_int(on_remove_position, 2);
ecs_delete(world, e3);
test_int(on_remove_position, 3);
ecs_fini(world);
}
static int on_remove_tag_set_position_invoked = 0;
static
void on_remove_tag_set_position(ecs_iter_t *it) {
for (int i = 0; i < it->count; i ++) {
ecs_set(it->world, it->entities[i], Position, {10, 20});
on_remove_tag_set_position_invoked ++;
}
}
static
void on_remove_tag_set_position_pair(ecs_iter_t *it) {
for (int i = 0; i < it->count; i ++) {
ecs_set_pair(it->world, it->entities[i],
Position, ecs_new_id(it->world), {10, 20});
on_remove_tag_set_position_invoked ++;
}
}
static
void on_remove_tag_set_position_obj_pair(ecs_iter_t *it) {
for (int i = 0; i < it->count; i ++) {
ecs_set_pair_object(it->world, it->entities[i],
ecs_new_id(it->world), Position, {10, 20});
on_remove_tag_set_position_invoked ++;
}
}
void ComponentLifecycle_free_component_new_id_while_fini(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT_DEFINE(world, Position);
ECS_TAG(world, Tag);
ecs_observer_init(world, &(ecs_observer_desc_t){
.filter.terms[0].id = Tag,
.events = {EcsOnRemove},
.callback = on_remove_tag_set_position
});
ecs_new(world, Tag);
ecs_fini(world);
test_int(on_remove_tag_set_position_invoked, 1);
}
void ComponentLifecycle_dtor_component_new_id_while_fini(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT_DEFINE(world, Position);
ECS_TAG(world, Tag);
ecs_observer_init(world, &(ecs_observer_desc_t){
.filter.terms[0].id = Tag,
.events = {EcsOnRemove},
.callback = on_remove_tag_set_position
});
ecs_set_hooks(world, Position, {
.ctor = ecs_default_ctor,
.dtor = ecs_dtor(Position)
});
ecs_new(world, Tag);
test_int(dtor_position, 0);
ecs_fini(world);
test_int(on_remove_tag_set_position_invoked, 1);
test_int(dtor_position, 1);
}
void ComponentLifecycle_free_component_new_pair_id_while_fini(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT_DEFINE(world, Position);
ECS_TAG(world, Tag);
ecs_observer_init(world, &(ecs_observer_desc_t){
.filter.terms[0].id = Tag,
.events = {EcsOnRemove},
.callback = on_remove_tag_set_position_pair
});
ecs_new(world, Tag);
ecs_fini(world);
test_int(on_remove_tag_set_position_invoked, 1);
}
void ComponentLifecycle_dtor_component_new_pair_id_while_fini(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT_DEFINE(world, Position);
ECS_TAG(world, Tag);
ecs_observer_init(world, &(ecs_observer_desc_t){
.filter.terms[0].id = Tag,
.events = {EcsOnRemove},
.callback = on_remove_tag_set_position_pair
});
ecs_set_hooks(world, Position, {
.ctor = ecs_default_ctor,
.dtor = ecs_dtor(Position)
});
ecs_new(world, Tag);
test_int(dtor_position, 0);
ecs_fini(world);
test_int(on_remove_tag_set_position_invoked, 1);
test_int(dtor_position, 1);
}
void ComponentLifecycle_free_component_new_obj_pair_id_while_fini(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT_DEFINE(world, Position);
ECS_TAG(world, Tag);
ecs_observer_init(world, &(ecs_observer_desc_t){
.filter.terms[0].id = Tag,
.events = {EcsOnRemove},
.callback = on_remove_tag_set_position_obj_pair
});
ecs_new(world, Tag);
ecs_fini(world);
test_int(on_remove_tag_set_position_invoked, 1);
}
void ComponentLifecycle_dtor_component_new_obj_pair_id_while_fini(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT_DEFINE(world, Position);
ECS_TAG(world, Tag);
ecs_observer_init(world, &(ecs_observer_desc_t){
.filter.terms[0].id = Tag,
.events = {EcsOnRemove},
.callback = on_remove_tag_set_position_obj_pair
});
ecs_set_hooks(world, Position, {
.ctor = ecs_default_ctor,
.dtor = ecs_dtor(Position)
});
ecs_new(world, Tag);
test_int(dtor_position, 0);
ecs_fini(world);
test_int(on_remove_tag_set_position_invoked, 1);
test_int(dtor_position, 1);
}
void ComponentLifecycle_ctor_move_dtor_after_resize(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
cl_ctx ctx = { { 0 } };
ecs_set_hooks(world, Position, {
.ctor = comp_ctor,
.copy = comp_copy,
.move = comp_move,
.dtor = comp_dtor,
.ctx = &ctx
});
ecs_entity_t e1 = ecs_new_id(world);
ecs_entity_t e2 = ecs_new_id(world);
ecs_entity_t e3 = ecs_new_id(world);
ecs_add(world, e1, Position);
test_int(ctx.ctor.count, 1);
test_int(ctx.copy.count, 0);
test_int(ctx.move.count, 0);
test_int(ctx.dtor.count, 0);
ecs_os_zeromem(&ctx);
ecs_add(world, e2, Position);
test_int(ctx.ctor.count, 1);
test_int(ctx.copy.count, 0);
test_int(ctx.move.count, 0);
test_int(ctx.dtor.count, 0);
ecs_os_zeromem(&ctx);
ecs_add(world, e3, Position);
test_int(ctx.ctor.count, 3);
test_int(ctx.copy.count, 0);
test_int(ctx.move.count, 2);
test_int(ctx.dtor.count, 2);
ecs_fini(world);
}
static int component_lifecycle_ctx = 0;
static int component_lifecycle_binding_ctx = 0;
static void component_lifecycle_ctx_free(void *ctx) {
test_assert(ctx == &component_lifecycle_ctx);
component_lifecycle_ctx ++;
}
void ComponentLifecycle_ctx_free(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctx = &component_lifecycle_ctx,
.ctx_free = component_lifecycle_ctx_free
});
ecs_fini(world);
test_int(1, component_lifecycle_ctx);
}
void ComponentLifecycle_binding_ctx_free(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.binding_ctx = &component_lifecycle_ctx,
.binding_ctx_free = component_lifecycle_ctx_free
});
ecs_fini(world);
test_int(1, component_lifecycle_ctx);
}
void ComponentLifecycle_ctx_free_after_delete_component(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctx = &component_lifecycle_ctx,
.ctx_free = component_lifecycle_ctx_free
});
ecs_remove_pair(world, ecs_id(Position), EcsOnDelete, EcsPanic);
ecs_delete(world, ecs_id(Position));
test_int(1, component_lifecycle_ctx);
ecs_fini(world);
}
void ComponentLifecycle_binding_ctx_free_after_delete_component(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.binding_ctx = &component_lifecycle_ctx,
.binding_ctx_free = component_lifecycle_ctx_free
});
ecs_remove_pair(world, ecs_id(Position), EcsOnDelete, EcsPanic);
ecs_delete(world, ecs_id(Position));
test_int(1, component_lifecycle_ctx);
ecs_fini(world);
}
static void test_lifecycle_ctx(ecs_iter_t *it) {
test_assert(it->ctx == &component_lifecycle_ctx);
test_assert(it->binding_ctx == &component_lifecycle_binding_ctx);
component_lifecycle_ctx ++;
component_lifecycle_binding_ctx ++;
}
void ComponentLifecycle_on_add_ctx(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_add = test_lifecycle_ctx,
.ctx = &component_lifecycle_ctx,
.binding_ctx = &component_lifecycle_binding_ctx
});
ecs_new(world, Position);
test_int(1, component_lifecycle_ctx);
test_int(1, component_lifecycle_binding_ctx);
ecs_fini(world);
test_int(1, component_lifecycle_ctx);
test_int(1, component_lifecycle_binding_ctx);
}
void ComponentLifecycle_on_remove_ctx(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_remove = test_lifecycle_ctx,
.ctx = &component_lifecycle_ctx,
.binding_ctx = &component_lifecycle_binding_ctx
});
ecs_entity_t e = ecs_new(world, Position);
test_int(0, component_lifecycle_ctx);
test_int(0, component_lifecycle_binding_ctx);
ecs_remove(world, e, Position);
test_int(1, component_lifecycle_ctx);
test_int(1, component_lifecycle_binding_ctx);
ecs_fini(world);
test_int(1, component_lifecycle_ctx);
test_int(1, component_lifecycle_binding_ctx);
}
void ComponentLifecycle_on_set_ctx(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_set = test_lifecycle_ctx,
.ctx = &component_lifecycle_ctx,
.binding_ctx = &component_lifecycle_binding_ctx
});
ecs_entity_t e = ecs_new(world, Position);
test_int(0, component_lifecycle_ctx);
test_int(0, component_lifecycle_binding_ctx);
ecs_set(world, e, Position, {10, 20});
test_int(1, component_lifecycle_ctx);
test_int(1, component_lifecycle_binding_ctx);
ecs_fini(world);
test_int(1, component_lifecycle_ctx);
test_int(1, component_lifecycle_binding_ctx);
}
static int test_on_event_invoked = 0;
static void test_on_event(ecs_iter_t *it) {
test_assert(it->ctx != NULL);
test_assert(*(ecs_entity_t*)it->ctx == it->event);
test_on_event_invoked ++;
}
void ComponentLifecycle_on_add_w_existing_component(void) {
ecs_world_t* world = ecs_mini();
ECS_TAG(world, Tag);
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_add = test_on_event,
.ctx = (void*)&EcsOnAdd
});
ecs_entity_t e = ecs_new_entity(world, "Foo");
ecs_add(world, e, Position);
test_int(1, test_on_event_invoked);
ecs_fini(world);
}
void ComponentLifecycle_on_remove_w_existing_component(void) {
ecs_world_t* world = ecs_mini();
ECS_TAG(world, Tag);
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_remove = test_on_event,
.ctx = (void*)&EcsOnRemove
});
ecs_entity_t e = ecs_new_entity(world, "Foo");
ecs_add(world, e, Position);
test_int(0, test_on_event_invoked);
ecs_remove(world, e, Position);
test_int(1, test_on_event_invoked);
ecs_fini(world);
}
static int on_add_count = 0;
static int on_remove_count = 0;
static void test_on_add(ecs_iter_t *it) {
on_add_count ++;
}
static void test_on_remove(ecs_iter_t *it) {
on_remove_count ++;
}
void ComponentLifecycle_component_init_set_hooks(void) {
ecs_world_t* world = ecs_mini();
ecs_entity_t c = ecs_component_init(world, &(ecs_component_desc_t){
.entity = ecs_entity(world, {.name = "Position"}),
.type = {
.size = ECS_SIZEOF(Position),
.alignment = ECS_ALIGNOF(Position),
.hooks = {
.on_add = test_on_add,
.on_remove = test_on_remove
}
}
});
const ecs_type_info_t *ti = ecs_get_type_info(world, c);
test_assert(ti != NULL);
test_uint(ti->size, ECS_SIZEOF(Position));
test_uint(ti->alignment, ECS_ALIGNOF(Position));
test_assert(ti->hooks.on_add == test_on_add);
test_assert(ti->hooks.on_remove == test_on_remove);
test_int(0, on_add_count);
test_int(0, on_remove_count);
ecs_entity_t e = ecs_new_id(world);
ecs_add_id(world, e, c);
test_int(1, on_add_count);
test_int(0, on_remove_count);
ecs_remove_id(world, e, c);
test_int(1, on_add_count);
test_int(1, on_remove_count);
ecs_fini(world);
test_int(1, on_add_count);
test_int(1, on_remove_count);
}
static int ctor_before_on_add_count = 0;
static int on_add_after_ctor_count = 0;
static void ctor_before_on_add(
void *ptr, int32_t count, const ecs_type_info_t *info)
{
test_int(0, on_add_after_ctor_count);
ctor_before_on_add_count ++;
}
static void on_add_after_ctor(ecs_iter_t *it)
{
test_int(1, ctor_before_on_add_count);
on_add_after_ctor_count ++;
}
void ComponentLifecycle_on_add_after_ctor_w_add(void) {
ecs_world_t* world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ctor_before_on_add,
.on_add = on_add_after_ctor
});
test_int(0, ctor_before_on_add_count);
test_int(0, on_add_after_ctor_count);
ecs_new(world, Position);
test_int(1, ctor_before_on_add_count);
test_int(1, on_add_after_ctor_count);
ecs_fini(world);
}
void ComponentLifecycle_on_add_after_ctor_w_add_to(void) {
ecs_world_t* world = ecs_mini();
ECS_COMPONENT(world, Position);
ECS_TAG(world, Tag);
ecs_set_hooks(world, Position, {
.ctor = ctor_before_on_add,
.on_add = on_add_after_ctor
});
test_int(0, ctor_before_on_add_count);
test_int(0, on_add_after_ctor_count);
ecs_entity_t e = ecs_new(world, Tag);
test_int(0, ctor_before_on_add_count);
test_int(0, on_add_after_ctor_count);
ecs_add(world, e, Position);
test_int(1, ctor_before_on_add_count);
test_int(1, on_add_after_ctor_count);
ecs_fini(world);
}
void ComponentLifecycle_with_before_hooks(void) {
ecs_world_t* world = ecs_mini();
ecs_entity_t pos_id = ecs_new_id(world);
ecs_entity_t tag = ecs_new_w_pair(world, EcsWith, pos_id);
ecs_entity_t ecs_id(Position) =
ecs_component_init(world, &(ecs_component_desc_t){
.entity = pos_id,
.type = {
.size = ECS_SIZEOF(Position),
.alignment = ECS_ALIGNOF(Position),
.hooks = {
.ctor = ecs_ctor(Position)
}
}
});
test_assert(ecs_id(Position) == pos_id);
ecs_entity_t e = ecs_new_w_id(world, tag);
test_assert(ecs_has(world, e, Position));
test_assert(ecs_has_id(world, e, tag));
test_assert(ecs_get(world, e, Position) != NULL);
test_int(ctor_position, 1);
const ecs_type_info_t *ti = ecs_get_type_info(world, ecs_id(Position));
test_assert(ti != NULL);
test_assert(ti->hooks.ctor == ecs_ctor(Position));
ecs_fini(world);
}
void ComponentLifecycle_with_component_on_add(void) {
ecs_world_t *world = ecs_mini();
ecs_entity_t ecs_id(Position) =
ecs_component(world, {
.type = {
.size = ECS_SIZEOF(Position),
.alignment = ECS_ALIGNOF(Position),
.hooks = {
.ctor = ecs_default_ctor,
.on_add = ecs_on_add(Position)
}
}
});
ECS_TAG(world, Foo);
ecs_add_pair(world, Foo, EcsWith, ecs_id(Position));
test_int(on_add_position, 0);
ecs_entity_t e = ecs_new(world, Foo);
test_assert(ecs_has(world, e, Foo));
test_assert(ecs_has(world, e, Position));
test_int(on_add_position, 1);
ecs_fini(world);
}
void ComponentLifecycle_move_ctor_on_move(void) {
ecs_world_t* world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.dtor = ecs_dtor(Position),
.move = ecs_move(Position),
.move_ctor = position_move_ctor,
});
ecs_entity_t e1 = ecs_new_id(world);
ecs_entity_t e2 = ecs_new_id(world);
ecs_entity_t p = ecs_new_id(world);
Position *p1 = ecs_emplace(world, e1, Position);
test_assert(p1 != NULL);
Position *p2 = ecs_emplace(world, e2, Position);
test_assert(p2 != NULL);
test_int(ctor_position, 0);
ecs_add_pair(world, e1, EcsChildOf, p);
test_int(ctor_position, 0);
test_int(move_ctor_position, 1); // move e1 to other table
test_int(move_position, 1); // move e2 to old position of e1
test_int(dtor_position, 1); // dtor old position for e2
ecs_add_pair(world, e2, EcsChildOf, p);
ecs_fini(world);
}
typedef struct {
void *ptr;
} TestSelf;
ECS_COPY(TestSelf, dst, src, {
test_assert(dst->ptr == dst);
})
ECS_MOVE(TestSelf, dst, src, {
test_assert(dst->ptr == dst);
test_assert(src->ptr == src);
})
ECS_CTOR(TestSelf, ptr, {
ptr->ptr = ptr;
})
ECS_DTOR(TestSelf, ptr, {
test_assert(ptr->ptr == ptr);
})
void ComponentLifecycle_ptr_to_self(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, TestSelf);
ecs_set_hooks(world, TestSelf, {
.ctor = ecs_ctor(TestSelf),
.copy = ecs_copy(TestSelf),
.move = ecs_move(TestSelf),
.dtor = ecs_dtor(TestSelf)
});
ecs_entity_t role = ecs_new_entity(world, "MyRole");
ecs_entity_t e1 = ecs_new_id(world);
ecs_set(world, e1, TestSelf, {"a"});
ecs_entity_t e2 = ecs_new_id(world);
ecs_set(world, e2, TestSelf, {"a"});
ecs_entity_t e3 = ecs_new_id(world);
ecs_add_pair(world, e2, e3, role);
ecs_entity_t e4 = ecs_new_id(world);
ecs_set(world, e4, TestSelf, {"a"});
ecs_delete(world, role);
ecs_delete_with(world, ecs_id(TestSelf));
ecs_fini(world);
}
void ComponentLifecycle_ctor_move_dtor_from_move_ctor(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.ctor = ecs_ctor(Position),
.copy = ecs_copy(Position),
.move_ctor = position_move_ctor
});
ecs_entity_t e = ecs_new_id(world);
ecs_world_t *async = ecs_async_stage_new(world);
Position *p = ecs_emplace(async, e, Position);
test_assert(!ecs_has(world, e, Position));
test_int(ctor_position, 0);
test_int(copy_position, 0);
test_int(move_ctor_position, 0);
p->x = 10;
p->y = 20;
ecs_merge(async);
test_assert(ecs_has(world, e, Position));
test_int(ctor_position, 0);
test_int(copy_position, 0);
test_int(move_ctor_position, 1);
const Position *ptr = ecs_get(world, e, Position);
test_int(ptr->x, 10);
test_int(ptr->y, 20);
ecs_async_stage_free(async);
ecs_fini(world);
}
static int hook_w_offset_invoked = 0;
static int hook_w_offset_offset = 0;
static Position hook_w_offset_position;
static
void hook_w_offset(ecs_iter_t *it) {
Position *p = ecs_field(it, Position, 1);
test_int(it->count, 1);
hook_w_offset_offset = it->offset;
hook_w_offset_invoked ++;
hook_w_offset_position = *p;
}
void ComponentLifecycle_on_add_hook_check_offset(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_add = hook_w_offset
});
ecs_set(world, 0, Position, {10, 20});
test_int(hook_w_offset_invoked, 1);
test_int(hook_w_offset_offset, 0);
ecs_set(world, 0, Position, {30, 40});
test_int(hook_w_offset_invoked, 2);
test_int(hook_w_offset_offset, 1);
ecs_set(world, 0, Position, {50, 60});
test_int(hook_w_offset_invoked, 3);
test_int(hook_w_offset_offset, 2);
ecs_fini(world);
}
void ComponentLifecycle_on_remove_hook_check_offset(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_remove = hook_w_offset
});
ecs_entity_t e1 = ecs_set(world, 0, Position, {10, 20});
ecs_entity_t e2 = ecs_set(world, 0, Position, {30, 40});
ecs_entity_t e3 = ecs_set(world, 0, Position, {50, 60});
test_int(hook_w_offset_invoked, 0);
ecs_remove(world, e3, Position);
test_int(hook_w_offset_invoked, 1);
test_int(hook_w_offset_offset, 2);
test_int(hook_w_offset_position.x, 50);
test_int(hook_w_offset_position.y, 60);
ecs_remove(world, e2, Position);
test_int(hook_w_offset_invoked, 2);
test_int(hook_w_offset_offset, 1);
test_int(hook_w_offset_position.x, 30);
test_int(hook_w_offset_position.y, 40);
ecs_remove(world, e1, Position);
test_int(hook_w_offset_invoked, 3);
test_int(hook_w_offset_offset, 0);
test_int(hook_w_offset_position.x, 10);
test_int(hook_w_offset_position.y, 20);
ecs_fini(world);
}
void ComponentLifecycle_on_set_hook_check_offset(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_set = hook_w_offset
});
ecs_set(world, 0, Position, {10, 20});
test_int(hook_w_offset_invoked, 1);
test_int(hook_w_offset_offset, 0);
test_int(hook_w_offset_position.x, 10);
test_int(hook_w_offset_position.y, 20);
ecs_set(world, 0, Position, {30, 40});
test_int(hook_w_offset_invoked, 2);
test_int(hook_w_offset_offset, 1);
test_int(hook_w_offset_position.x, 30);
test_int(hook_w_offset_position.y, 40);
ecs_set(world, 0, Position, {50, 60});
test_int(hook_w_offset_invoked, 3);
test_int(hook_w_offset_offset, 2);
test_int(hook_w_offset_position.x, 50);
test_int(hook_w_offset_position.y, 60);
ecs_fini(world);
}
static int on_set_position_invoked = 0;
static
void on_set_position(ecs_iter_t *it) {
Position *p = ecs_field(it, Position, 1);
test_int(it->count, 1);
test_int(p->x, 10);
test_int(p->y, 20);
on_set_position_invoked ++;
}
void ComponentLifecycle_on_set_hook_on_override(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_set = on_set_position
});
ecs_entity_t p = ecs_set(world, 0, Position, {10, 20});
ecs_add_id(world, p, EcsPrefab);
test_int(on_set_position_invoked, 1);
ecs_entity_t i = ecs_new_w_pair(world, EcsIsA, p);
test_int(on_set_position_invoked, 1);
ecs_add(world, i, Position);
test_int(on_set_position_invoked, 2);
{
const Position *p = ecs_get(world, i, Position);
test_assert(p != NULL);
test_int(p->x, 10);
test_int(p->y, 20);
}
ecs_fini(world);
}
void ComponentLifecycle_on_set_hook_on_auto_override(void) {
ecs_world_t *world = ecs_mini();
ECS_COMPONENT(world, Position);
ecs_set_hooks(world, Position, {
.on_set = on_set_position
});
ecs_entity_t p = ecs_set(world, 0, Position, {10, 20});
ecs_add_id(world, p, ECS_OVERRIDE | ecs_id(Position));
ecs_add_id(world, p, EcsPrefab);
test_int(on_set_position_invoked, 1);
ecs_entity_t i = ecs_new_w_pair(world, EcsIsA, p);
test_int(on_set_position_invoked, 2);
{
const Position *p = ecs_get(world, i, Position);
test_assert(p != NULL);
test_int(p->x, 10);
test_int(p->y, 20);
}
ecs_fini(world);
}