taisei/src/projectile.c
2020-03-04 21:50:51 +02:00

1111 lines
26 KiB
C

/*
* This software is licensed under the terms of the MIT License.
* See COPYING for further information.
* ---
* Copyright (c) 2011-2019, Lukas Weber <laochailan@web.de>.
* Copyright (c) 2012-2019, Andrei Alexeyev <akari@taisei-project.org>.
*/
#include "taisei.h"
#include "projectile.h"
#include "global.h"
#include "list.h"
#include "stageobjects.h"
ht_ptr2int_t shader_sublayer_map;
static ProjArgs defaults_proj = {
.sprite = "proj/",
.draw_rule = ProjDraw,
.dest = &global.projs,
.type = PROJ_ENEMY,
.damage_type = DMG_ENEMY_SHOT,
.color = RGB(1, 1, 1),
.blend = BLEND_PREMUL_ALPHA,
.shader = "sprite_bullet",
.layer = LAYER_BULLET,
};
static ProjArgs defaults_part = {
.sprite = "part/",
.draw_rule = ProjDraw,
.dest = &global.particles,
.type = PROJ_PARTICLE,
.damage_type = DMG_UNDEFINED,
.color = RGB(1, 1, 1),
.blend = BLEND_PREMUL_ALPHA,
.shader = "sprite_default",
.layer = LAYER_PARTICLE_MID,
};
static void process_projectile_args(ProjArgs *args, ProjArgs *defaults) {
if(args->proto && args->proto->process_args) {
args->proto->process_args(args->proto, args);
return;
}
// TODO: move this stuff into prototypes along with the defaults?
if(args->sprite) {
args->sprite_ptr = prefix_get_sprite(args->sprite, defaults->sprite);
}
if(!args->shader_ptr) {
if(args->shader) {
args->shader_ptr = r_shader_get(args->shader);
} else {
args->shader_ptr = defaults->shader_ptr;
}
}
if(!args->draw_rule) {
args->draw_rule = ProjDraw;
}
if(!args->blend) {
args->blend = defaults->blend;
}
if(!args->dest) {
args->dest = defaults->dest;
}
if(!args->type) {
args->type = defaults->type;
}
if(!args->color) {
args->color = defaults->color;
}
if(!args->max_viewport_dist && (args->type == PROJ_PARTICLE || args->type == PROJ_PLAYER)) {
args->max_viewport_dist = 300;
}
if(!args->layer) {
if(args->type == PROJ_PLAYER) {
args->layer = LAYER_PLAYER_SHOT;
} else {
args->layer = defaults->layer;
}
}
if(args->damage_type == DMG_UNDEFINED) {
args->damage_type = defaults->damage_type;
if(args->type == PROJ_PLAYER && args->damage_type == DMG_ENEMY_SHOT) {
args->damage_type = DMG_PLAYER_SHOT;
}
}
assert(args->type <= PROJ_PLAYER);
}
static void projectile_size(Projectile *p, double *w, double *h) {
if(p->type == PROJ_PARTICLE && p->sprite != NULL) {
*w = p->sprite->w;
*h = p->sprite->h;
} else {
*w = creal(p->size);
*h = cimag(p->size);
}
assert(*w > 0);
assert(*h > 0);
}
static void ent_draw_projectile(EntityInterface *ent);
static inline char* event_name(int ev) {
switch(ev) {
case EVENT_BIRTH: return "EVENT_BIRTH";
case EVENT_DEATH: return "EVENT_DEATH";
default: UNREACHABLE;
}
}
static Projectile* spawn_bullet_spawning_effect(Projectile *p);
static inline int proj_call_rule(Projectile *p, int t) {
int result = ACTION_NONE;
if(p->timeout > 0 && t >= p->timeout) {
result = ACTION_DESTROY;
} else if(p->rule != NULL) {
result = p->rule(p, t);
if(t < 0 && result != ACTION_ACK) {
set_debug_info(&p->debug);
log_fatal(
"Projectile rule didn't acknowledge %s (returned %i, expected %i)",
event_name(t),
result,
ACTION_ACK
);
}
} else if(t >= 0) {
p->pos += move_update(&p->move);
if(p->pos != p->prevpos) {
p->angle = carg(p->pos - p->prevpos);
}
}
if(/*t == 0 ||*/ t == EVENT_BIRTH) {
p->prevpos = p->pos;
}
if(t == 0) {
spawn_bullet_spawning_effect(p);
}
return result;
}
void projectile_set_prototype(Projectile *p, ProjPrototype *proto) {
if(p->proto && p->proto->deinit_projectile) {
p->proto->deinit_projectile(p->proto, p);
}
if(proto && proto->init_projectile) {
proto->init_projectile(proto, p);
}
p->proto = proto;
}
cmplx projectile_graze_size(Projectile *p) {
if(
p->type == PROJ_ENEMY &&
!(p->flags & (PFLAG_NOGRAZE | PFLAG_NOCOLLISION)) &&
p->graze_counter < 3 &&
global.frames >= p->graze_cooldown
) {
cmplx s = (p->size * 420 /* graze it */) / (2 * p->graze_counter + 1);
return sqrt(creal(s)) + sqrt(cimag(s)) * I;
}
return 0;
}
static double projectile_rect_area(Projectile *p) {
double w, h;
projectile_size(p, &w, &h);
return w * h;
}
static Projectile* _create_projectile(ProjArgs *args) {
if(IN_DRAW_CODE) {
log_fatal("Tried to spawn a projectile while in drawing code");
}
Projectile *p = (Projectile*)objpool_acquire(stage_object_pools.projectiles);
p->birthtime = global.frames;
p->pos = p->pos0 = p->prevpos = args->pos;
p->angle = args->angle;
p->rule = args->rule;
p->draw_rule = args->draw_rule;
p->shader = args->shader_ptr;
p->blend = args->blend;
p->sprite = args->sprite_ptr;
p->type = args->type;
p->color = *args->color;
p->max_viewport_dist = args->max_viewport_dist;
p->size = args->size;
p->collision_size = args->collision_size;
p->flags = args->flags;
p->timeout = args->timeout;
p->damage = args->damage;
p->damage_type = args->damage_type;
p->clear_flags = 0;
p->move = args->move;
if(args->shader_params != NULL) {
p->shader_params = *args->shader_params;
}
memcpy(p->args, args->args, sizeof(p->args));
p->ent.draw_layer = args->layer;
p->ent.draw_func = ent_draw_projectile;
projectile_set_prototype(p, args->proto);
// p->collision_size *= 10;
// p->size *= 5;
if((p->type == PROJ_ENEMY || p->type == PROJ_PLAYER) && (creal(p->size) <= 0 || cimag(p->size) <= 0)) {
log_fatal("Tried to spawn a projectile with invalid size %f x %f", creal(p->size), cimag(p->size));
}
if(!(p->ent.draw_layer & LAYER_LOW_MASK)) {
drawlayer_low_t sublayer;
switch(p->type) {
case PROJ_ENEMY:
// 1. Large projectiles go below smaller ones.
sublayer = LAYER_LOW_MASK - (drawlayer_low_t)projectile_rect_area(p);
sublayer = (sublayer << 4) & LAYER_LOW_MASK;
// 2. Group by shader (hardcoded precedence).
sublayer |= ht_get(&shader_sublayer_map, p->shader, 0) & 0xf;
// If specific blending order is required, then you should set up the sublayer manually.
p->ent.draw_layer |= sublayer;
break;
case PROJ_PARTICLE:
// 1. Group by shader (hardcoded precedence).
sublayer = ht_get(&shader_sublayer_map, p->shader, 0) & 0xf;
sublayer <<= 4;
sublayer |= 0x100;
// If specific blending order is required, then you should set up the sublayer manually.
p->ent.draw_layer |= sublayer;
break;
default:
break;
}
}
coevent_init(&p->events.killed);
ent_register(&p->ent, ENT_PROJECTILE);
// TODO: Maybe allow ACTION_DESTROY here?
// But in that case, code that uses this function's return value must be careful to not dereference a NULL pointer.
proj_call_rule(p, EVENT_BIRTH);
alist_append(args->dest, p);
return p;
}
Projectile* create_projectile(ProjArgs *args) {
process_projectile_args(args, &defaults_proj);
return _create_projectile(args);
}
Projectile* create_particle(ProjArgs *args) {
process_projectile_args(args, &defaults_part);
return _create_projectile(args);
}
#ifdef PROJ_DEBUG
Projectile* _proj_attach_dbginfo(Projectile *p, DebugInfo *dbg, const char *callsite_str) {
// log_debug("Spawn: [%s]", callsite_str);
memcpy(&p->debug, dbg, sizeof(DebugInfo));
set_debug_info(dbg);
return p;
}
#endif
static void *_delete_projectile(ListAnchor *projlist, List *proj, void *arg) {
Projectile *p = (Projectile*)proj;
proj_call_rule(p, EVENT_DEATH);
coevent_signal_once(&p->events.killed);
ent_unregister(&p->ent);
objpool_release(stage_object_pools.projectiles, alist_unlink(projlist, proj));
return NULL;
}
static void delete_projectile(ProjectileList *projlist, Projectile *proj) {
_delete_projectile((ListAnchor*)projlist, (List*)proj, NULL);
}
void delete_projectiles(ProjectileList *projlist) {
alist_foreach(projlist, _delete_projectile, NULL);
}
void calc_projectile_collision(Projectile *p, ProjCollisionResult *out_col) {
assert(out_col != NULL);
out_col->type = PCOL_NONE;
out_col->entity = NULL;
out_col->fatal = false;
out_col->location = p->pos;
out_col->damage.amount = p->damage;
out_col->damage.type = p->damage_type;
if(p->flags & PFLAG_NOCOLLISION) {
goto skip_collision;
}
if(p->type == PROJ_ENEMY) {
Ellipse e_proj = {
.axes = p->collision_size,
.angle = p->angle + M_PI/2,
};
LineSegment seg = {
.a = global.plr.pos - global.plr.velocity - p->prevpos,
.b = global.plr.pos - p->pos
};
attr_unused double seglen = cabs(seg.a - seg.b);
if(seglen > 30) {
log_debug(
seglen > VIEWPORT_W
? "Lerp over HUGE distance %f; this is ABSOLUTELY a bug! Player speed was %f. Spawned at %s:%d (%s); proj time = %d"
: "Lerp over large distance %f; this is either a bug or a very fast projectile, investigate. Player speed was %f. Spawned at %s:%d (%s); proj time = %d",
seglen,
cabs(global.plr.velocity),
p->debug.file,
p->debug.line,
p->debug.func,
global.frames - p->birthtime
);
}
if(lineseg_ellipse_intersect(seg, e_proj)) {
out_col->type = PCOL_ENTITY;
out_col->entity = &global.plr.ent;
out_col->fatal = true;
} else {
e_proj.axes = projectile_graze_size(p);
if(creal(e_proj.axes) > 1 && lineseg_ellipse_intersect(seg, e_proj)) {
out_col->type = PCOL_PLAYER_GRAZE;
out_col->entity = &global.plr.ent;
out_col->location = p->pos;
}
}
} else if(p->type == PROJ_PLAYER) {
for(Enemy *e = global.enemies.first; e; e = e->next) {
if(e->hp != ENEMY_IMMUNE && cabs(e->pos - p->pos) < 30) {
out_col->type = PCOL_ENTITY;
out_col->entity = &e->ent;
out_col->fatal = true;
return;
}
}
if(global.boss && cabs(global.boss->pos - p->pos) < 42) {
if(boss_is_vulnerable(global.boss)) {
out_col->type = PCOL_ENTITY;
out_col->entity = &global.boss->ent;
out_col->fatal = true;
}
}
}
skip_collision:
if(out_col->type == PCOL_NONE && !projectile_in_viewport(p)) {
out_col->type = PCOL_VOID;
out_col->fatal = true;
}
}
void apply_projectile_collision(ProjectileList *projlist, Projectile *p, ProjCollisionResult *col) {
switch(col->type) {
case PCOL_NONE:
case PCOL_VOID:
break;
case PCOL_PLAYER_GRAZE: {
player_graze(ENT_CAST(col->entity, Player), col->location, 10 + 10 * p->graze_counter, 3 + p->graze_counter, &p->color);
p->graze_counter++;
p->graze_cooldown = global.frames + 12;
p->graze_counter_reset_timer = global.frames;
break;
}
case PCOL_ENTITY: {
ent_damage(col->entity, &col->damage);
break;
}
default:
UNREACHABLE;
}
if(col->fatal) {
delete_projectile(projlist, p);
}
}
static void ent_draw_projectile(EntityInterface *ent) {
Projectile *proj = ENT_CAST(ent, Projectile);
r_blend(proj->blend);
r_shader_ptr(proj->shader);
#ifdef PROJ_DEBUG
static Projectile prev_state;
memcpy(&prev_state, proj, sizeof(Projectile));
proj->draw_rule(proj, global.frames - proj->birthtime);
if(memcmp(&prev_state, proj, sizeof(Projectile))) {
set_debug_info(&proj->debug);
log_fatal("Projectile modified its state in draw rule");
}
#else
proj->draw_rule(proj, global.frames - proj->birthtime);
#endif
}
bool projectile_in_viewport(Projectile *proj) {
double w, h;
int e = proj->max_viewport_dist;
projectile_size(proj, &w, &h);
return !(creal(proj->pos) + w/2 + e < 0 || creal(proj->pos) - w/2 - e > VIEWPORT_W
|| cimag(proj->pos) + h/2 + e < 0 || cimag(proj->pos) - h/2 - e > VIEWPORT_H);
}
Projectile* spawn_projectile_collision_effect(Projectile *proj) {
if(proj->flags & PFLAG_NOCOLLISIONEFFECT) {
return NULL;
}
if(proj->sprite == NULL) {
return NULL;
}
return PARTICLE(
.sprite_ptr = proj->sprite,
.size = proj->size,
.pos = proj->pos,
.color = &proj->color,
.flags = proj->flags | PFLAG_NOREFLECT | PFLAG_REQUIREDPARTICLE,
.layer = LAYER_PARTICLE_HIGH,
.shader_ptr = proj->shader,
.rule = linear,
.draw_rule = DeathShrink,
.angle = proj->angle,
.args = { 5*cexp(I*proj->angle) },
.timeout = 10,
);
}
static void really_clear_projectile(ProjectileList *projlist, Projectile *proj) {
Projectile *effect = spawn_projectile_clear_effect(proj);
Item *clear_item = NULL;
if(!(proj->flags & PFLAG_NOCLEARBONUS)) {
clear_item = create_clear_item(proj->pos, proj->clear_flags);
}
if(clear_item != NULL && effect != NULL) {
effect->args[0] = add_ref(clear_item);
}
delete_projectile(projlist, proj);
}
bool clear_projectile(Projectile *proj, uint flags) {
switch(proj->type) {
case PROJ_PLAYER:
case PROJ_PARTICLE:
return false;
default: break;
}
if(!(flags & CLEAR_HAZARDS_FORCE) && !projectile_is_clearable(proj)) {
return false;
}
proj->type = PROJ_DEAD;
proj->clear_flags |= flags;
return true;
}
void kill_projectile(Projectile* proj) {
coevent_signal_once(&proj->events.killed);
proj->flags |= PFLAG_INTERNAL_DEAD | PFLAG_NOCOLLISION | PFLAG_NOCLEAR;
proj->draw_rule = ProjNoDraw;
}
void process_projectiles(ProjectileList *projlist, bool collision) {
ProjCollisionResult col = { 0 };
char killed = 0;
int action;
bool stage_cleared = stage_is_cleared();
for(Projectile *proj = projlist->first, *next; proj; proj = next) {
next = proj->next;
proj->prevpos = proj->pos;
if(proj->flags & PFLAG_INTERNAL_DEAD) {
delete_projectile(projlist, proj);
continue;
}
if(stage_cleared) {
clear_projectile(proj, CLEAR_HAZARDS_BULLETS | CLEAR_HAZARDS_FORCE);
}
action = proj_call_rule(proj, global.frames - proj->birthtime);
if(proj->graze_counter && proj->graze_counter_reset_timer - global.frames <= -90) {
proj->graze_counter--;
proj->graze_counter_reset_timer = global.frames;
}
if(proj->type == PROJ_DEAD && killed < 10 && !(proj->clear_flags & CLEAR_HAZARDS_NOW)) {
proj->clear_flags |= CLEAR_HAZARDS_NOW;
killed++;
}
if(action == ACTION_DESTROY) {
memset(&col, 0, sizeof(col));
col.fatal = true;
} else if(collision) {
calc_projectile_collision(proj, &col);
if(col.fatal && col.type != PCOL_VOID) {
spawn_projectile_collision_effect(proj);
}
} else {
memset(&col, 0, sizeof(col));
if(!projectile_in_viewport(proj)) {
col.fatal = true;
}
}
apply_projectile_collision(projlist, proj, &col);
}
for(Projectile *proj = projlist->first, *next; proj; proj = next) {
next = proj->next;
if(proj->type == PROJ_DEAD && (proj->clear_flags & CLEAR_HAZARDS_NOW)) {
really_clear_projectile(projlist, proj);
}
}
}
int trace_projectile(Projectile *p, ProjCollisionResult *out_col, ProjCollisionType stopflags, int timeofs) {
int t;
for(t = timeofs; p; ++t) {
int action = p->rule(p, t);
calc_projectile_collision(p, out_col);
if(out_col->type & stopflags || action == ACTION_DESTROY) {
return t;
}
}
return t;
}
bool projectile_is_clearable(Projectile *p) {
if(p->type == PROJ_DEAD) {
return true;
}
if(p->type == PROJ_ENEMY) {
return (p->flags & PFLAG_NOCLEAR) != PFLAG_NOCLEAR;
}
return false;
}
int linear(Projectile *p, int t) { // sure is physics in here; a[0]: velocity
if(t == EVENT_DEATH) {
return ACTION_ACK;
}
p->angle = carg(p->args[0]);
if(t == EVENT_BIRTH) {
return ACTION_ACK;
}
p->pos = p->pos0 + p->args[0]*t;
return ACTION_NONE;
}
int accelerated(Projectile *p, int t) {
if(t == EVENT_DEATH) {
return ACTION_ACK;
}
p->angle = carg(p->args[0]);
if(t == EVENT_BIRTH) {
return ACTION_ACK;
}
p->pos += p->args[0];
p->args[0] += p->args[1];
return 1;
}
int asymptotic(Projectile *p, int t) { // v = a[0]*(a[1] + 1); a[1] -> 0
if(t == EVENT_DEATH) {
return ACTION_ACK;
}
p->angle = carg(p->args[0]);
if(t == EVENT_BIRTH) {
return ACTION_ACK;
}
p->args[1] *= 0.8;
p->pos += p->args[0]*(p->args[1] + 1);
return 1;
}
static inline bool proj_uses_spawning_effect(Projectile *proj, ProjFlags effect_flag) {
if(proj->type != PROJ_ENEMY) {
return false;
}
if((proj->flags & effect_flag) == effect_flag) {
return false;
}
return true;
}
static float proj_spawn_effect_factor(Projectile *proj, int t) {
static const int maxt = 16;
if(t >= maxt || !proj_uses_spawning_effect(proj, PFLAG_NOSPAWNFADE)) {
return 1;
}
return t / (float)maxt;
}
static inline void apply_common_transforms(Projectile *proj, int t) {
r_mat_mv_translate(creal(proj->pos), cimag(proj->pos), 0);
r_mat_mv_rotate(proj->angle + M_PI/2, 0, 0, 1);
/*
float s = 0.75 + 0.25 * proj_spawn_effect_factor(proj, t);
if(s != 1) {
r_mat_mv_scale(s, s, 1);
}
*/
}
static void bullet_highlight_draw(Projectile *p, int t) {
float timefactor = t / p->timeout;
float sx = creal(p->args[0]);
float sy = cimag(p->args[0]);
float opacity = pow(1 - timefactor, 2);
opacity = min(1, 1.5 * opacity) * min(1, timefactor * 10);
r_mat_tex_push();
r_mat_tex_translate(0.5, 0.5, 0);
r_mat_tex_rotate(p->args[1], 0, 0, 1);
r_mat_tex_translate(-0.5, -0.5, 0);
r_draw_sprite(&(SpriteParams) {
.sprite_ptr = p->sprite,
.shader_ptr = p->shader,
.shader_params = &(ShaderCustomParams) {{ 1 - opacity }},
.color = &p->color,
.scale = { .x = sx, .y = sy },
.rotation.angle = p->angle + M_PI * 0.5,
.pos = { creal(p->pos), cimag(p->pos) }
});
r_mat_tex_pop();
}
static Projectile* spawn_projectile_highlight_effect_internal(Projectile *p, bool flare) {
if(!p->sprite) {
return NULL;
}
Color clr = p->color;
clr.r = fmax(0.1, clr.r);
clr.g = fmax(0.1, clr.g);
clr.b = fmax(0.1, clr.b);
float h, s, l;
color_get_hsl(&clr, &h, &s, &l);
s = s > 0 ? 0.75 : 0;
l = 0.5;
color_hsla(&clr, h, s, l, 0.05);
float sx, sy;
if(flare) {
sx = pow(p->sprite->w, 0.7);
sy = pow(p->sprite->h, 0.7);
PARTICLE(
.sprite = "stardust_green",
.shader = "sprite_bullet",
.size = p->size * 4.5,
.layer = LAYER_PARTICLE_HIGH | 0x40,
.draw_rule = ScaleSquaredFade,
.args = { 0, 0, (0 + 2*I) * 0.1 * fmax(sx, sy) * (1 - 0.2 * frand()) },
.angle = frand() * M_PI * 2,
.pos = p->pos + frand() * 8 * cexp(I*M_PI*2*frand()),
.flags = PFLAG_NOREFLECT,
.timeout = 24 + 2 * nfrand(),
.color = &clr,
);
}
sx = pow((1.5 * p->sprite->w + 0.5 * p->sprite->h) * 0.5, 0.65);
sy = pow((1.5 * p->sprite->h + 0.5 * p->sprite->w) * 0.5, 0.65);
clr.a = 0.2;
return PARTICLE(
.sprite = "bullet_cloud",
.size = p->size * 4.5,
.shader = "sprite_bullet",
.layer = LAYER_PARTICLE_HIGH | 0x80,
.draw_rule = bullet_highlight_draw,
.args = { 0.125 * (sx + I * sy), frand() * M_PI * 2 },
.angle = p->angle,
.pos = p->pos + frand() * 5 * cexp(I*M_PI*2*frand()),
.flags = PFLAG_NOREFLECT | PFLAG_REQUIREDPARTICLE,
.timeout = 32 + 2 * nfrand(),
.color = &clr,
);
}
Projectile* spawn_projectile_highlight_effect(Projectile *p) {
return spawn_projectile_highlight_effect_internal(p, true);
}
static Projectile* spawn_bullet_spawning_effect(Projectile *p) {
if(proj_uses_spawning_effect(p, PFLAG_NOSPAWNFLARE)) {
return spawn_projectile_highlight_effect(p);
}
return NULL;
}
static void projectile_clear_effect_draw(Projectile *p, int t) {
r_mat_mv_push();
apply_common_transforms(p, t);
float timefactor = t / p->timeout;
float plrfactor = clamp(1 - (cabs(p->pos - global.plr.pos) - 64) / 128, 0, 1);
plrfactor *= clamp(timefactor * 10, 0, 1);
float f = 1 - (1 - timefactor) * (1 - plrfactor);
Sprite spr = *p->sprite;
Sprite *ispr = get_sprite("item/bullet_point");
spr.w = f * ispr->w + (1 - f) * spr.w;
spr.h = f * ispr->h + (1 - f) * spr.h;
r_draw_sprite(&(SpriteParams) {
.sprite_ptr = &spr,
.color = RGBA(p->color.r, p->color.g, p->color.b, p->color.a * (1 - 0)),
.shader_params = &(ShaderCustomParams){{ f }},
// .scale.both = 1 + 0.5 * sqrt(tf),
});
r_mat_mv_pop();
}
static int projectile_clear_effect_logic(Projectile *p, int t) {
if(t == EVENT_DEATH) {
free_ref(p->args[0]);
return ACTION_ACK;
}
if(t == EVENT_BIRTH) {
return ACTION_ACK;
}
if((int)p->args[0] < 0) {
return ACTION_NONE;
}
Item *i = REF(p->args[0]);
if(i != NULL) {
p->pos = i->pos;
}
return ACTION_NONE;
}
Projectile* spawn_projectile_clear_effect(Projectile *proj) {
if((proj->flags & PFLAG_NOCLEAREFFECT) || proj->sprite == NULL) {
return NULL;
}
// spawn_projectile_highlight_effect_internal(proj, false);
ShaderProgram *shader = proj->shader;
uint32_t layer = LAYER_PARTICLE_BULLET_CLEAR;
if(proj->shader == defaults_proj.shader_ptr) {
// HACK
shader = r_shader_get("sprite_bullet_dead");
layer |= 0x1;
}
return PARTICLE(
.sprite_ptr = proj->sprite,
.size = proj->size,
.pos = proj->pos,
.color = &proj->color,
.flags = proj->flags | PFLAG_NOREFLECT | PFLAG_REQUIREDPARTICLE,
.shader_ptr = shader,
.rule = projectile_clear_effect_logic,
.draw_rule = projectile_clear_effect_draw,
.angle = proj->angle,
.timeout = 24,
.args = { -1 },
.layer = layer,
);
}
void ProjDrawCore(Projectile *proj, const Color *c) {
r_draw_sprite(&(SpriteParams) {
.sprite_ptr = proj->sprite,
.color = c,
.shader_params = &proj->shader_params,
});
}
void ProjDraw(Projectile *proj, int t) {
r_mat_mv_push();
apply_common_transforms(proj, t);
float eff = proj_spawn_effect_factor(proj, t);
/*
if(eff < 1 && proj->color.a > 0) {
Color c = proj->color;
c.a *= sqrt(eff);
ProjDrawCore(proj, &c);
} else {
ProjDrawCore(proj, &proj->color);
}
*/
if(eff < 1) {
float o = proj->shader_params.vector[0];
float a = min(1, eff * 2);
proj->shader_params.vector[0] = 1 - (1 - o) * a;
Color c = proj->color;
c.a *= eff;//sqrt(eff);
ProjDrawCore(proj, &c);
proj->shader_params.vector[0] = o;
} else {
ProjDrawCore(proj, &proj->color);
}
r_mat_mv_pop();
}
void ProjNoDraw(Projectile *proj, int t) {
}
void Blast(Projectile *p, int t) {
r_mat_mv_push();
r_mat_mv_translate(creal(p->pos), cimag(p->pos), 0);
r_mat_mv_rotate(creal(p->args[1]) * DEG2RAD, cimag(p->args[1]), creal(p->args[2]), cimag(p->args[2]));
if(t != p->timeout && p->timeout != 0) {
r_mat_mv_scale(t/(double)p->timeout, t/(double)p->timeout, 1);
}
float fade = 1.0 - t / (double)p->timeout;
r_color(RGBA_MUL_ALPHA(0.3, 0.6, 1.0, fade));
SpriteParams sp = {
.sprite_ptr = p->sprite,
.color = RGBA_MUL_ALPHA(0.3, 0.6, 1.0, fade),
};
r_draw_sprite(&sp);
r_mat_mv_scale(0.5+creal(p->args[2]), 0.5+creal(p->args[2]), 1);
r_color4(0.3 * fade, 0.6 * fade, 1.0 * fade, 0);
sp.color = RGBA(0.3 * fade, 0.6 * fade, 1.0 * fade, 0);
r_draw_sprite(&sp);
r_mat_mv_pop();
}
void Shrink(Projectile *p, int t) {
r_mat_mv_push();
apply_common_transforms(p, t);
float s = 2.0-t/(double)p->timeout*2;
if(s != 1) {
r_mat_mv_scale(s, s, 1);
}
ProjDrawCore(p, &p->color);
r_mat_mv_pop();
}
void DeathShrink(Projectile *p, int t) {
r_mat_mv_push();
apply_common_transforms(p, t);
float s = 2.0-t/(double)p->timeout*2;
if(s != 1) {
r_mat_mv_scale(s, 1, 1);
}
ProjDrawCore(p, &p->color);
r_mat_mv_pop();
}
void GrowFade(Projectile *p, int t) {
r_mat_mv_push();
apply_common_transforms(p, t);
set_debug_info(&p->debug);
assert(p->timeout != 0);
float s = t/(double)p->timeout*(1 + (creal(p->args[2])? p->args[2] : p->args[1]));
if(s != 1) {
r_mat_mv_scale(s, s, 1);
}
ProjDrawCore(p, color_mul_scalar(COLOR_COPY(&p->color), 1 - t/(double)p->timeout));
r_mat_mv_pop();
}
void Fade(Projectile *p, int t) {
r_mat_mv_push();
apply_common_transforms(p, t);
ProjDrawCore(p, color_mul_scalar(COLOR_COPY(&p->color), 1 - t/(double)p->timeout));
r_mat_mv_pop();
}
static void ScaleFadeImpl(Projectile *p, int t, int fade_exponent) {
r_mat_mv_push();
apply_common_transforms(p, t);
double scale_min = creal(p->args[2]);
double scale_max = cimag(p->args[2]);
double timefactor = t / (double)p->timeout;
double scale = scale_min * (1 - timefactor) + scale_max * timefactor;
double alpha = pow(fabs(1 - timefactor), fade_exponent);
r_mat_mv_scale(scale, scale, 1);
ProjDrawCore(p, color_mul_scalar(COLOR_COPY(&p->color), alpha));
r_mat_mv_pop();
}
void ScaleFade(Projectile *p, int t) {
ScaleFadeImpl(p, t, 1);
}
void ScaleSquaredFade(Projectile *p, int t) {
ScaleFadeImpl(p, t, 2);
}
void Petal(Projectile *p, int t) {
float x = creal(p->args[2]);
float y = cimag(p->args[2]);
float z = creal(p->args[3]);
float r = sqrt(x*x+y*y+z*z);
x /= r; y /= r; z /= r;
r_disable(RCAP_CULL_FACE);
r_mat_mv_push();
r_mat_mv_translate(creal(p->pos), cimag(p->pos),0);
r_mat_mv_rotate(DEG2RAD * (t*4.0 + cimag(p->args[3])), x, y, z);
ProjDrawCore(p, &p->color);
r_mat_mv_pop();
}
void petal_explosion(int n, cmplx pos) {
for(int i = 0; i < n; i++) {
tsrand_fill(6);
float t = frand();
PARTICLE(
.sprite = "petal",
.pos = pos,
.color = RGBA(sin(5*t) * t, cos(5*t) * t, 0.5 * t, 0),
.rule = asymptotic,
.draw_rule = Petal,
.args = {
(3+5*afrand(2))*cexp(I*M_PI*2*afrand(3)),
5,
afrand(4) + afrand(5)*I,
afrand(1) + 360.0*I*afrand(0),
},
// TODO: maybe remove this noreflect, there shouldn't be a cull mode mess anymore
.flags = PFLAG_NOREFLECT | (n % 2 ? 0 : PFLAG_REQUIREDPARTICLE),
.layer = LAYER_PARTICLE_PETAL,
);
}
}
void projectiles_preload(void) {
const char *shaders[] = {
// This array defines a shader-based fallback draw order
"sprite_silhouette",
defaults_proj.shader,
defaults_part.shader,
"sprite_bullet_dead",
};
const uint num_shaders = sizeof(shaders)/sizeof(*shaders);
for(uint i = 0; i < num_shaders; ++i) {
preload_resource(RES_SHADER_PROGRAM, shaders[i], RESF_PERMANENT);
}
// FIXME: Why is this here?
preload_resources(RES_TEXTURE, RESF_PERMANENT,
"part/lasercurve",
NULL);
// TODO: Maybe split this up into stage-specific preloads too?
// some of these are ubiquitous, but some only appear in very specific parts.
preload_resources(RES_SPRITE, RESF_PERMANENT,
"part/blast",
"part/bullet_cloud",
"part/flare",
"part/graze",
"part/lightning0",
"part/lightning1",
"part/lightningball",
"part/petal",
"part/smoke",
"part/smoothdot",
"part/stain",
"part/stardust_green",
NULL);
preload_resources(RES_SFX, RESF_PERMANENT,
"shot1",
"shot2",
"shot3",
"shot1_loop",
"shot_special1",
"redirect",
NULL);
#define PP(name) (_pp_##name).preload(&_pp_##name);
#include "projectile_prototypes/all.inc.h"
ht_create(&shader_sublayer_map);
for(uint i = 0; i < num_shaders; ++i) {
ht_set(&shader_sublayer_map, r_shader_get(shaders[i]), i + 1);
}
defaults_proj.shader_ptr = r_shader_get(defaults_proj.shader);
defaults_part.shader_ptr = r_shader_get(defaults_part.shader);
}
void projectiles_free(void) {
ht_destroy(&shader_sublayer_map);
}