2010-10-12 10:55:23 +02:00
|
|
|
/*
|
2019-08-03 19:43:48 +02:00
|
|
|
* This software is licensed under the terms of the MIT License.
|
2017-02-10 23:05:22 +01:00
|
|
|
* See COPYING for further information.
|
2011-03-05 13:44:21 +01:00
|
|
|
* ---
|
2019-01-23 21:10:43 +01:00
|
|
|
* Copyright (c) 2011-2019, Lukas Weber <laochailan@web.de>.
|
2019-07-03 20:00:56 +02:00
|
|
|
* Copyright (c) 2012-2019, Andrei Alexeyev <akari@taisei-project.org>.
|
2010-10-12 10:55:23 +02:00
|
|
|
*/
|
|
|
|
|
2017-11-25 20:45:11 +01:00
|
|
|
#include "taisei.h"
|
|
|
|
|
2010-10-12 10:55:23 +02:00
|
|
|
#include "player.h"
|
|
|
|
|
|
|
|
#include "projectile.h"
|
|
|
|
#include "global.h"
|
2011-04-10 11:19:44 +02:00
|
|
|
#include "plrmodes.h"
|
2012-08-17 20:58:23 +02:00
|
|
|
#include "stage.h"
|
2017-09-16 03:15:34 +02:00
|
|
|
#include "stagetext.h"
|
2018-07-04 10:36:16 +02:00
|
|
|
#include "stagedraw.h"
|
2020-04-26 21:27:13 +02:00
|
|
|
#include "stats.h"
|
2018-04-13 21:13:48 +02:00
|
|
|
#include "entity.h"
|
2020-01-04 04:24:24 +01:00
|
|
|
#include "util/glm.h"
|
2010-10-12 10:55:23 +02:00
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
DEFINE_ENTITY_TYPE(PlayerIndicators, {
|
|
|
|
Player *plr;
|
|
|
|
|
|
|
|
struct {
|
|
|
|
Sprite *focus;
|
|
|
|
} sprites;
|
|
|
|
|
|
|
|
float focus_alpha;
|
|
|
|
int focus_time;
|
|
|
|
});
|
|
|
|
|
2017-10-08 13:30:51 +02:00
|
|
|
void player_init(Player *plr) {
|
2011-06-25 12:41:40 +02:00
|
|
|
memset(plr, 0, sizeof(Player));
|
2019-03-05 12:50:29 +01:00
|
|
|
plr->pos = PLR_SPAWN_POS;
|
2017-03-21 11:09:32 +01:00
|
|
|
plr->lives = PLR_START_LIVES;
|
2011-07-02 12:45:32 +02:00
|
|
|
plr->bombs = PLR_START_BOMBS;
|
2019-02-22 00:56:03 +01:00
|
|
|
plr->point_item_value = PLR_START_PIV;
|
|
|
|
plr->power = 100;
|
2011-06-25 12:41:40 +02:00
|
|
|
plr->deathtime = -1;
|
2017-10-29 23:45:24 +01:00
|
|
|
plr->continuetime = -1;
|
2017-10-08 13:30:51 +02:00
|
|
|
plr->mode = plrmode_find(0, 0);
|
2011-07-04 09:14:08 +02:00
|
|
|
}
|
|
|
|
|
2017-10-08 13:30:51 +02:00
|
|
|
void player_stage_pre_init(Player *plr) {
|
2017-02-15 17:09:09 +01:00
|
|
|
plr->recovery = 0;
|
|
|
|
plr->respawntime = 0;
|
|
|
|
plr->deathtime = -1;
|
|
|
|
plr->axis_lr = 0;
|
|
|
|
plr->axis_ud = 0;
|
2017-10-08 23:12:38 +02:00
|
|
|
plrmode_preload(plr->mode);
|
2017-10-08 13:30:51 +02:00
|
|
|
}
|
|
|
|
|
2018-05-02 08:08:32 +02:00
|
|
|
double player_property(Player *plr, PlrProperty prop) {
|
|
|
|
return plr->mode->procs.property(plr, prop);
|
|
|
|
}
|
|
|
|
|
2018-04-13 21:13:48 +02:00
|
|
|
static void ent_draw_player(EntityInterface *ent);
|
2018-07-30 09:04:09 +02:00
|
|
|
static DamageResult ent_damage_player(EntityInterface *ent, const DamageInfo *dmg);
|
|
|
|
|
2020-03-13 21:54:07 +01:00
|
|
|
DECLARE_TASK(player_logic, { BoxedPlayer plr; });
|
2020-04-22 23:17:40 +02:00
|
|
|
DECLARE_TASK(player_indicators, { BoxedPlayer plr; });
|
2020-03-13 21:54:07 +01:00
|
|
|
|
2017-10-08 13:30:51 +02:00
|
|
|
void player_stage_post_init(Player *plr) {
|
|
|
|
assert(plr->mode != NULL);
|
|
|
|
|
|
|
|
// ensure the essential callbacks are there. other code tests only for the optional ones
|
2018-05-02 08:08:32 +02:00
|
|
|
assert(plr->mode->procs.property != NULL);
|
2018-07-15 17:48:22 +02:00
|
|
|
assert(plr->mode->character != NULL);
|
|
|
|
assert(plr->mode->dialog != NULL);
|
2017-10-24 04:57:14 +02:00
|
|
|
|
2019-07-08 02:47:50 +02:00
|
|
|
plrchar_make_bomb_portrait(plr->mode->character, &plr->bomb_portrait);
|
2018-04-12 16:08:48 +02:00
|
|
|
aniplayer_create(&plr->ani, get_ani(plr->mode->character->player_sprite_name), "main");
|
2018-04-13 21:13:48 +02:00
|
|
|
|
|
|
|
plr->ent.draw_layer = LAYER_PLAYER;
|
|
|
|
plr->ent.draw_func = ent_draw_player;
|
2018-07-30 09:04:09 +02:00
|
|
|
plr->ent.damage_func = ent_damage_player;
|
2020-04-17 09:18:53 +02:00
|
|
|
ent_register(&plr->ent, ENT_TYPE_ID(Player));
|
2018-04-13 21:13:48 +02:00
|
|
|
|
2020-03-13 21:54:07 +01:00
|
|
|
COEVENT_INIT_ARRAY(plr->events);
|
|
|
|
|
|
|
|
INVOKE_TASK_DELAYED(1, player_logic, ENT_BOX(plr));
|
2020-04-22 23:17:40 +02:00
|
|
|
INVOKE_TASK_DELAYED(1, player_indicators, ENT_BOX(plr));
|
2020-03-13 21:54:07 +01:00
|
|
|
|
|
|
|
if(plr->mode->procs.init != NULL) {
|
|
|
|
plr->mode->procs.init(plr);
|
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
plr->extralife_threshold = player_next_extralife_threshold(plr->extralives_given);
|
|
|
|
|
|
|
|
while(plr->points >= plr->extralife_threshold) {
|
|
|
|
plr->extralife_threshold = player_next_extralife_threshold(++plr->extralives_given);
|
|
|
|
}
|
2017-02-15 17:09:09 +01:00
|
|
|
}
|
|
|
|
|
2017-11-25 16:51:43 +01:00
|
|
|
void player_free(Player *plr) {
|
2020-03-13 21:54:07 +01:00
|
|
|
COEVENT_CANCEL_ARRAY(plr->events);
|
|
|
|
|
2017-11-25 16:51:43 +01:00
|
|
|
if(plr->mode->procs.free) {
|
|
|
|
plr->mode->procs.free(plr);
|
|
|
|
}
|
2018-04-13 21:13:48 +02:00
|
|
|
|
2019-07-08 02:47:50 +02:00
|
|
|
r_texture_destroy(plr->bomb_portrait.tex);
|
2018-05-08 18:17:18 +02:00
|
|
|
aniplayer_free(&plr->ani);
|
2018-04-13 21:13:48 +02:00
|
|
|
ent_unregister(&plr->ent);
|
2017-11-25 16:51:43 +01:00
|
|
|
}
|
|
|
|
|
2017-03-19 03:56:55 +01:00
|
|
|
static void player_full_power(Player *plr) {
|
2018-01-06 10:24:46 +01:00
|
|
|
stage_clear_hazards(CLEAR_HAZARDS_ALL);
|
2018-07-23 19:07:59 +02:00
|
|
|
stagetext_add("Full Power!", VIEWPORT_W * 0.5 + VIEWPORT_H * 0.33 * I, ALIGN_CENTER, get_font("big"), RGB(1, 1, 1), 0, 60, 20, 20);
|
2017-03-19 03:56:55 +01:00
|
|
|
}
|
|
|
|
|
2017-10-24 04:57:14 +02:00
|
|
|
bool player_set_power(Player *plr, short npow) {
|
2019-02-22 00:56:03 +01:00
|
|
|
int pow_base = clamp(npow, 0, PLR_MAX_POWER);
|
|
|
|
int pow_overflow = clamp(npow - PLR_MAX_POWER, 0, PLR_MAX_POWER_OVERFLOW);
|
2017-03-30 17:11:21 +02:00
|
|
|
|
2017-10-08 13:30:51 +02:00
|
|
|
if(plr->mode->procs.power) {
|
2019-02-22 00:56:03 +01:00
|
|
|
plr->mode->procs.power(plr, pow_base);
|
2011-07-04 09:14:08 +02:00
|
|
|
}
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2017-03-19 03:56:55 +01:00
|
|
|
int oldpow = plr->power;
|
2019-02-22 00:56:03 +01:00
|
|
|
int oldpow_over = plr->power_overflow;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
plr->power = pow_base;
|
|
|
|
plr->power_overflow = pow_overflow;
|
|
|
|
|
|
|
|
if((oldpow + oldpow_over) / 100 < (pow_base + pow_overflow) / 100) {
|
2017-12-26 04:18:57 +01:00
|
|
|
play_sound("powerup");
|
|
|
|
}
|
|
|
|
|
2017-10-24 04:57:14 +02:00
|
|
|
if(plr->power == PLR_MAX_POWER && oldpow < PLR_MAX_POWER) {
|
2017-03-19 03:56:55 +01:00
|
|
|
player_full_power(plr);
|
|
|
|
}
|
2017-09-19 20:46:28 +02:00
|
|
|
|
2020-03-13 21:54:07 +01:00
|
|
|
bool change = (oldpow + oldpow_over) != (plr->power + plr->power_overflow);
|
|
|
|
|
|
|
|
if(change) {
|
|
|
|
coevent_signal(&plr->events.power_changed);
|
|
|
|
}
|
|
|
|
|
|
|
|
return change;
|
2019-02-22 00:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool player_add_power(Player *plr, short pdelta) {
|
|
|
|
return player_set_power(plr, plr->power + plr->power_overflow + pdelta);
|
2011-07-04 09:14:08 +02:00
|
|
|
}
|
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
void player_move(Player *plr, cmplx delta) {
|
2018-05-02 08:08:32 +02:00
|
|
|
delta *= player_property(plr, PLR_PROP_SPEED);
|
2019-11-22 04:37:11 +01:00
|
|
|
cmplx lastpos = plr->pos;
|
2017-11-11 22:44:54 +01:00
|
|
|
double x = clamp(creal(plr->pos) + creal(delta), PLR_MIN_BORDER_DIST, VIEWPORT_W - PLR_MIN_BORDER_DIST);
|
|
|
|
double y = clamp(cimag(plr->pos) + cimag(delta), PLR_MIN_BORDER_DIST, VIEWPORT_H - PLR_MIN_BORDER_DIST);
|
|
|
|
plr->pos = x + y*I;
|
2020-04-22 23:35:06 +02:00
|
|
|
plr->velocity = plr->pos - lastpos;
|
2011-07-05 15:20:19 +02:00
|
|
|
}
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2019-07-03 19:50:43 +02:00
|
|
|
void player_draw_overlay(Player *plr) {
|
|
|
|
// plr->bomb_cutin_alpha = 1 - fmod(global.frames / 200.0, 1.0);
|
|
|
|
float a = 1 - plr->bomb_cutin_alpha;
|
|
|
|
|
|
|
|
if(a <= 0 || a >= 1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
r_state_push();
|
|
|
|
r_shader("sprite_default");
|
|
|
|
|
|
|
|
float char_in = clamp(a * 1.5, 0, 1);
|
|
|
|
float char_out = min(1, 2 - (2 * a));
|
|
|
|
float char_opacity_in = 0.75 * min(1, a * 5);
|
|
|
|
float char_opacity = char_opacity_in * char_out * char_out;
|
|
|
|
float char_xofs = -20 * a;
|
|
|
|
|
2019-07-08 02:47:50 +02:00
|
|
|
// Sprite *char_spr = get_sprite(plr->mode->character->dialog_sprite_name);
|
|
|
|
Sprite *char_spr = &plr->bomb_portrait;
|
2019-07-03 19:50:43 +02:00
|
|
|
|
|
|
|
for(int i = 1; i <= 3; ++i) {
|
|
|
|
float t = a * 200;
|
|
|
|
float dur = 20;
|
|
|
|
float start = 200 - dur * 5;
|
|
|
|
float end = start + dur;
|
|
|
|
float ofs = 0.2 * dur * (i - 1);
|
|
|
|
float o = 1 - smoothstep(start + ofs, end + ofs, t);
|
|
|
|
|
|
|
|
r_draw_sprite(&(SpriteParams) {
|
|
|
|
.sprite_ptr = char_spr,
|
|
|
|
.pos = { char_spr->w * 0.5 + VIEWPORT_W * pow(1 - char_in, 4 - i * 0.3) - i + char_xofs, VIEWPORT_H - char_spr->h * 0.5 },
|
|
|
|
.color = color_mul_scalar(color_add(RGBA(0.2, 0.2, 0.2, 0), RGBA(i==1, i==2, i==3, 0)), char_opacity_in * (1 - char_in * o) * o),
|
|
|
|
.flip.x = true,
|
|
|
|
.scale.both = 1.0 + 0.02 * (min(1, a * 1.2)) + i * 0.5 * pow(1 - o, 2),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
r_draw_sprite(&(SpriteParams) {
|
|
|
|
.sprite_ptr = char_spr,
|
|
|
|
.pos = { char_spr->w * 0.5 + VIEWPORT_W * pow(1 - char_in, 4) + char_xofs, VIEWPORT_H - char_spr->h * 0.5 },
|
|
|
|
.color = RGBA_MUL_ALPHA(1, 1, 1, char_opacity * min(1, char_in * 2) * (1 - min(1, (1 - char_out) * 5))),
|
|
|
|
.flip.x = true,
|
|
|
|
.scale.both = 1.0 + 0.1 * (1 - char_out),
|
|
|
|
});
|
|
|
|
|
|
|
|
float spell_in = min(1, a * 3.0);
|
|
|
|
float spell_out = min(1, 3 - (3 * a));
|
|
|
|
float spell_opacity = min(1, a * 5) * spell_out * spell_out;
|
|
|
|
|
|
|
|
float spell_x = 128 * (1 - pow(1 - spell_in, 5)) + (VIEWPORT_W + 256) * pow(1 - spell_in, 3);
|
|
|
|
float spell_y = VIEWPORT_H - 128 * sqrt(a);
|
|
|
|
|
|
|
|
Sprite *spell_spr = get_sprite("spell");
|
|
|
|
|
|
|
|
r_draw_sprite(&(SpriteParams) {
|
|
|
|
.sprite_ptr = spell_spr,
|
|
|
|
.pos = { spell_x, spell_y },
|
|
|
|
.color = color_mul_scalar(RGBA(1, 1, 1, spell_in * 0.5), spell_opacity),
|
|
|
|
.scale.both = 3 - 2 * (1 - pow(1 - spell_in, 3)) + 2 * (1 - spell_out),
|
|
|
|
});
|
|
|
|
|
|
|
|
Font *font = get_font("standard");
|
|
|
|
|
2019-10-12 14:02:15 +02:00
|
|
|
r_mat_mv_push();
|
|
|
|
r_mat_mv_translate(spell_x - spell_spr->w * 0.5 + 10, spell_y + 5 - font_get_metrics(font)->descent, 0);
|
2019-07-03 19:50:43 +02:00
|
|
|
|
|
|
|
TextParams tp = {
|
|
|
|
// .pos = { spell_x - spell_spr->w * 0.5 + 10, spell_y + 5 - font_get_metrics(font)->descent },
|
|
|
|
.shader = "text_hud",
|
|
|
|
.font_ptr = font,
|
|
|
|
.color = color_mul_scalar(RGBA(1, 1, 1, spell_in), spell_opacity),
|
|
|
|
};
|
|
|
|
|
2019-10-12 14:02:15 +02:00
|
|
|
r_mat_mv_push();
|
|
|
|
r_mat_mv_scale(2 - 1 * spell_opacity, 2 - 1 * spell_opacity, 1);
|
2019-07-03 19:50:43 +02:00
|
|
|
text_draw(plr->mode->spellcard_name, &tp);
|
2019-10-12 14:02:15 +02:00
|
|
|
r_mat_mv_pop();
|
2019-07-03 19:50:43 +02:00
|
|
|
|
2019-10-12 14:02:15 +02:00
|
|
|
r_mat_mv_pop();
|
2019-07-03 19:50:43 +02:00
|
|
|
r_state_pop();
|
|
|
|
}
|
|
|
|
|
2018-04-13 21:13:48 +02:00
|
|
|
static void ent_draw_player(EntityInterface *ent) {
|
|
|
|
Player *plr = ENT_CAST(ent, Player);
|
|
|
|
|
2019-03-05 12:50:29 +01:00
|
|
|
if(plr->deathtime >= global.frames) {
|
2018-05-19 01:24:49 +02:00
|
|
|
return;
|
|
|
|
}
|
2017-03-19 03:14:28 +01:00
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
if(plr->focus_circle_alpha) {
|
2018-04-13 21:13:48 +02:00
|
|
|
r_draw_sprite(&(SpriteParams) {
|
|
|
|
.sprite = "fairy_circle",
|
|
|
|
.rotation.angle = DEG2RAD * global.frames * 10,
|
2020-04-22 23:17:40 +02:00
|
|
|
.color = RGBA_MUL_ALPHA(1, 1, 1, 0.2 * plr->focus_circle_alpha),
|
2018-04-13 21:13:48 +02:00
|
|
|
.pos = { creal(plr->pos), cimag(plr->pos) },
|
|
|
|
});
|
|
|
|
}
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2018-04-13 21:13:48 +02:00
|
|
|
Color c;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2019-01-04 23:59:39 +01:00
|
|
|
if(!player_is_vulnerable(plr)) {
|
|
|
|
float f = 0.3*sin(0.1*global.frames);
|
|
|
|
c = *RGBA_MUL_ALPHA(1.0+f, 1.0, 1.0-f, 0.7+f);
|
2018-04-13 21:13:48 +02:00
|
|
|
} else {
|
2018-07-23 19:07:59 +02:00
|
|
|
c = *RGBA_MUL_ALPHA(1.0, 1.0, 1.0, 1.0);
|
2018-04-13 21:13:48 +02:00
|
|
|
}
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2018-04-13 21:13:48 +02:00
|
|
|
r_draw_sprite(&(SpriteParams) {
|
|
|
|
.sprite_ptr = aniplayer_get_frame(&plr->ani),
|
2020-04-22 23:17:40 +02:00
|
|
|
.pos.as_cmplx = plr->pos,
|
2018-07-23 19:07:59 +02:00
|
|
|
.color = &c,
|
2018-04-13 21:13:48 +02:00
|
|
|
});
|
|
|
|
}
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
static void player_draw_indicators(EntityInterface *ent) {
|
|
|
|
PlayerIndicators *indicators = ENT_CAST(ent, PlayerIndicators);
|
|
|
|
Player *plr = indicators->plr;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
float focus_opacity = indicators->focus_alpha;
|
|
|
|
int t = global.frames - indicators->focus_time;
|
|
|
|
cmplx32 pos = plr->pos;
|
2018-05-08 09:06:33 +02:00
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
if(focus_opacity > 0) {
|
2020-04-22 23:17:40 +02:00
|
|
|
float trans_frames = 12;
|
|
|
|
float trans_factor = 1.0 - fminf(trans_frames, t) / trans_frames;
|
|
|
|
float rot_speed = DEG2RAD * global.frames * (1.0f + 3.0f * trans_factor);
|
|
|
|
float scale = 1.0f + trans_factor;
|
2018-04-13 21:13:48 +02:00
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
SpriteParams sp = {
|
|
|
|
.sprite_ptr = indicators->sprites.focus,
|
2019-02-22 00:56:03 +01:00
|
|
|
.rotation.angle = rot_speed,
|
2020-04-22 23:17:40 +02:00
|
|
|
.color = RGBA_MUL_ALPHA(1, 1, 1, focus_opacity),
|
|
|
|
.pos.as_cmplx = pos,
|
2019-02-22 00:56:03 +01:00
|
|
|
.scale.both = scale,
|
2020-04-22 23:17:40 +02:00
|
|
|
};
|
2019-02-22 00:56:03 +01:00
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
r_draw_sprite(&sp);
|
|
|
|
sp.rotation.angle *= -1;
|
|
|
|
r_draw_sprite(&sp);
|
2019-02-22 00:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
float ps_opacity = 1.0;
|
|
|
|
float ps_fill_factor = 1.0;
|
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
if(player_is_powersurge_active(plr)) {
|
|
|
|
ps_opacity *= clamp((global.frames - plr->powersurge.time.activated) / 30.0, 0, 1);
|
|
|
|
} else if(plr->powersurge.time.expired == 0) {
|
2019-02-22 00:56:03 +01:00
|
|
|
ps_opacity = 0;
|
|
|
|
} else {
|
2020-04-22 23:17:40 +02:00
|
|
|
ps_opacity *= (1 - clamp((global.frames - plr->powersurge.time.expired) / 40.0, 0, 1));
|
2019-02-22 00:56:03 +01:00
|
|
|
ps_fill_factor = pow(ps_opacity, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(ps_opacity > 0) {
|
|
|
|
r_state_push();
|
2019-10-12 14:02:15 +02:00
|
|
|
r_mat_mv_push();
|
2020-04-22 23:17:40 +02:00
|
|
|
r_mat_mv_translate(crealf(pos), cimagf(pos), 0);
|
2019-10-12 14:02:15 +02:00
|
|
|
r_mat_mv_scale(140, 140, 0);
|
2019-02-22 00:56:03 +01:00
|
|
|
r_shader("healthbar_radial");
|
|
|
|
r_uniform_vec4_rgba("borderColor", RGBA(0.5, 0.5, 0.5, 0.5));
|
|
|
|
r_uniform_vec4_rgba("glowColor", RGBA(0.5, 0.5, 0.5, 0.75));
|
|
|
|
r_uniform_vec4_rgba("fillColor", RGBA(1.5, 0.5, 0.0, 0.75));
|
|
|
|
r_uniform_vec4_rgba("altFillColor", RGBA(0.0, 0.5, 1.5, 0.75));
|
|
|
|
r_uniform_vec4_rgba("coreFillColor", RGBA(0.8, 0.8, 0.8, 0.25));
|
2020-04-22 23:17:40 +02:00
|
|
|
r_uniform_vec2("fill", plr->powersurge.positive * ps_fill_factor, plr->powersurge.negative * ps_fill_factor);
|
2019-02-22 00:56:03 +01:00
|
|
|
r_uniform_float("opacity", ps_opacity);
|
|
|
|
r_draw_quad();
|
2019-10-12 14:02:15 +02:00
|
|
|
r_mat_mv_pop();
|
2019-02-22 00:56:03 +01:00
|
|
|
r_state_pop();
|
|
|
|
|
|
|
|
char buf[64];
|
2020-04-22 23:17:40 +02:00
|
|
|
format_huge_num(0, plr->powersurge.bonus.baseline, sizeof(buf), buf);
|
2019-02-22 00:56:03 +01:00
|
|
|
Font *fnt = get_font("monotiny");
|
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
float x = crealf(pos);
|
|
|
|
float y = cimagf(pos) + 80;
|
2019-02-22 00:56:03 +01:00
|
|
|
|
|
|
|
float text_opacity = ps_opacity * 0.75;
|
|
|
|
|
|
|
|
x += text_draw(buf, &(TextParams) {
|
|
|
|
.shader = "text_hud",
|
|
|
|
.font_ptr = fnt,
|
|
|
|
.pos = { x, y },
|
|
|
|
.color = RGBA_MUL_ALPHA(1.0, 1.0, 1.0, text_opacity),
|
|
|
|
.align = ALIGN_CENTER,
|
|
|
|
});
|
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
snprintf(buf, sizeof(buf), " +%u", plr->powersurge.bonus.gain_rate);
|
2019-02-22 00:56:03 +01:00
|
|
|
|
|
|
|
text_draw(buf, &(TextParams) {
|
|
|
|
.shader = "text_hud",
|
|
|
|
.font_ptr = fnt,
|
|
|
|
.pos = { x, y },
|
|
|
|
.color = RGBA_MUL_ALPHA(0.3, 0.6, 1.0, text_opacity),
|
|
|
|
.align = ALIGN_LEFT,
|
|
|
|
});
|
|
|
|
|
|
|
|
r_shader("sprite_filled_circle");
|
|
|
|
r_uniform_vec4("color_inner", 0, 0, 0, 0);
|
|
|
|
r_uniform_vec4("color_outer", 0, 0, 0.1 * ps_opacity, 0.1 * ps_opacity);
|
|
|
|
}
|
2018-04-13 21:13:48 +02:00
|
|
|
}
|
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
DEFINE_TASK(player_indicators) {
|
|
|
|
PlayerIndicators *indicators = TASK_HOST_ENT(PlayerIndicators);
|
|
|
|
indicators->plr = TASK_BIND(ARGS.plr);
|
|
|
|
indicators->ent.draw_layer = LAYER_PLAYER_FOCUS;
|
|
|
|
indicators->ent.draw_func = player_draw_indicators;
|
|
|
|
indicators->sprites.focus = get_sprite("focus");
|
|
|
|
|
|
|
|
Player *plr = indicators->plr;
|
|
|
|
|
|
|
|
bool was_focused = false;
|
|
|
|
bool is_focused = false;
|
|
|
|
|
|
|
|
for(;;) {
|
|
|
|
is_focused = (plr->inputflags & INFLAG_FOCUS) && player_is_alive(plr);
|
|
|
|
|
|
|
|
if(is_focused && !was_focused) {
|
|
|
|
indicators->focus_time = global.frames;
|
|
|
|
indicators->focus_alpha = fminf(indicators->focus_alpha, 0.1f);
|
|
|
|
}
|
|
|
|
|
|
|
|
was_focused = is_focused;
|
|
|
|
fapproach_p(&indicators->focus_alpha, is_focused, 1.0f/30.0f);
|
|
|
|
|
|
|
|
YIELD;
|
|
|
|
}
|
2010-10-12 10:55:23 +02:00
|
|
|
}
|
|
|
|
|
2017-04-04 02:57:38 +02:00
|
|
|
static void player_fail_spell(Player *plr) {
|
2018-01-12 19:26:07 +01:00
|
|
|
if( !global.boss ||
|
2017-04-04 02:57:38 +02:00
|
|
|
!global.boss->current ||
|
|
|
|
global.boss->current->finished ||
|
|
|
|
global.boss->current->failtime ||
|
2017-10-01 00:41:32 +02:00
|
|
|
global.boss->current->starttime >= global.frames ||
|
2017-04-04 02:57:38 +02:00
|
|
|
global.stage->type == STAGE_SPELL
|
|
|
|
) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
global.boss->current->failtime = global.frames;
|
|
|
|
|
|
|
|
if(global.boss->current->type == AT_ExtraSpell) {
|
|
|
|
boss_finish_current_attack(global.boss);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-01 22:08:40 +02:00
|
|
|
bool player_should_shoot(Player *plr) {
|
2018-07-31 09:07:34 +02:00
|
|
|
return
|
|
|
|
(plr->inputflags & INFLAG_SHOT) &&
|
2019-07-03 19:50:43 +02:00
|
|
|
!dialog_is_active(global.dialog) &&
|
2020-04-01 22:08:40 +02:00
|
|
|
player_is_alive(&global.plr);
|
2017-10-08 13:30:51 +02:00
|
|
|
}
|
|
|
|
|
2018-07-30 15:14:27 +02:00
|
|
|
void player_placeholder_bomb_logic(Player *plr) {
|
2018-07-31 09:07:34 +02:00
|
|
|
if(!player_is_bomb_active(plr)) {
|
2018-07-30 15:14:27 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
DamageInfo dmg;
|
|
|
|
dmg.amount = 100;
|
|
|
|
dmg.type = DMG_PLAYER_BOMB;
|
|
|
|
|
|
|
|
for(Enemy *en = global.enemies.first; en; en = en->next) {
|
|
|
|
ent_damage(&en->ent, &dmg);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(global.boss) {
|
|
|
|
ent_damage(&global.boss->ent, &dmg);
|
|
|
|
}
|
|
|
|
|
|
|
|
stage_clear_hazards(CLEAR_HAZARDS_ALL);
|
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
static void player_powersurge_expired(Player *plr);
|
|
|
|
|
|
|
|
static void _powersurge_trail_draw(Projectile *p, float t, float cmul) {
|
|
|
|
float nt = t / p->timeout;
|
|
|
|
float s = 1 + (2 + 0.5 * psin((t+global.frames*1.23)/5.0)) * nt * nt;
|
|
|
|
|
|
|
|
r_draw_sprite(&(SpriteParams) {
|
|
|
|
.sprite_ptr = p->sprite,
|
|
|
|
.scale.both = s,
|
|
|
|
.pos = { creal(p->pos), cimag(p->pos) },
|
|
|
|
.color = color_mul_scalar(RGBA(0.8, 0.1 + 0.2 * psin((t+global.frames)/5.0), 0.1, 0.0), 0.5 * (1 - nt) * cmul),
|
|
|
|
.shader_params = &(ShaderCustomParams){{ -2 * nt * nt }},
|
2020-03-05 19:38:48 +01:00
|
|
|
.shader_ptr = p->shader,
|
2019-02-22 00:56:03 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-12-11 10:25:57 +01:00
|
|
|
static void powersurge_trail_draw(Projectile *p, int t, ProjDrawRuleArgs args) {
|
2019-02-22 00:56:03 +01:00
|
|
|
if(t > 0) {
|
|
|
|
_powersurge_trail_draw(p, t - 0.5, 0.25);
|
|
|
|
_powersurge_trail_draw(p, t, 0.25);
|
|
|
|
} else {
|
|
|
|
_powersurge_trail_draw(p, t, 0.5);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int powersurge_trail(Projectile *p, int t) {
|
|
|
|
if(t == EVENT_BIRTH) {
|
|
|
|
return ACTION_ACK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(t == EVENT_DEATH) {
|
|
|
|
return ACTION_ACK;
|
|
|
|
}
|
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
cmplx v = (global.plr.pos - p->pos) * 0.05;
|
2019-02-22 00:56:03 +01:00
|
|
|
p->args[0] += (v - p->args[0]) * (1 - t / p->timeout);
|
|
|
|
p->pos += p->args[0];
|
|
|
|
|
|
|
|
return ACTION_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void player_powersurge_calc_bonus(Player *plr, PowerSurgeBonus *b) {
|
|
|
|
b->gain_rate = round(1000 * plr->powersurge.negative * plr->powersurge.negative);
|
2019-03-28 16:56:06 +01:00
|
|
|
b->baseline = plr->powersurge.power + plr->powersurge.damage_done * 0.4;
|
2019-02-22 00:56:03 +01:00
|
|
|
b->score = b->baseline;
|
|
|
|
b->discharge_power = sqrtf(0.2 * b->baseline + 1024 * log1pf(b->baseline)) * smoothstep(0, 1, 0.0001 * b->baseline);
|
|
|
|
b->discharge_range = 1.2 * b->discharge_power;
|
|
|
|
b->discharge_damage = 10 * pow(b->discharge_power, 1.1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int powersurge_charge_particle(Projectile *p, int t) {
|
|
|
|
if(t == EVENT_BIRTH) {
|
|
|
|
return ACTION_ACK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(t == EVENT_DEATH) {
|
|
|
|
return ACTION_ACK;
|
|
|
|
}
|
|
|
|
|
|
|
|
Player *plr = &global.plr;
|
|
|
|
|
|
|
|
if(player_is_alive(plr)) {
|
|
|
|
p->pos = plr->pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ACTION_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void player_powersurge_logic(Player *plr) {
|
2019-07-03 19:50:43 +02:00
|
|
|
if(dialog_is_active(global.dialog)) {
|
2019-02-28 11:00:35 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
plr->powersurge.positive = max(0, plr->powersurge.positive - lerp(PLR_POWERSURGE_POSITIVE_DRAIN_MIN, PLR_POWERSURGE_POSITIVE_DRAIN_MAX, plr->powersurge.positive));
|
|
|
|
plr->powersurge.negative = max(0, plr->powersurge.negative - lerp(PLR_POWERSURGE_NEGATIVE_DRAIN_MIN, PLR_POWERSURGE_NEGATIVE_DRAIN_MAX, plr->powersurge.negative));
|
|
|
|
|
|
|
|
if(stage_is_cleared()) {
|
|
|
|
player_cancel_powersurge(plr);
|
2019-02-28 11:00:35 +01:00
|
|
|
return;
|
2019-02-22 00:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if(plr->powersurge.positive <= plr->powersurge.negative) {
|
|
|
|
player_powersurge_expired(plr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
player_powersurge_calc_bonus(plr, &plr->powersurge.bonus);
|
|
|
|
|
|
|
|
PARTICLE(
|
2020-01-04 02:03:08 +01:00
|
|
|
.sprite_ptr = aniplayer_get_frame(&plr->ani),
|
2020-03-05 19:38:48 +01:00
|
|
|
.shader = "sprite_silhouette",
|
2019-02-22 00:56:03 +01:00
|
|
|
.pos = plr->pos,
|
|
|
|
.color = RGBA(1, 1, 1, 0.5),
|
|
|
|
.rule = powersurge_trail,
|
|
|
|
.draw_rule = powersurge_trail_draw,
|
|
|
|
.timeout = 15,
|
|
|
|
.layer = LAYER_PARTICLE_HIGH,
|
|
|
|
.flags = PFLAG_NOREFLECT,
|
|
|
|
);
|
|
|
|
|
2020-01-06 06:22:44 +01:00
|
|
|
if(!(global.frames % 6) && plr->powersurge.bonus.discharge_range > 0) {
|
2019-02-22 00:56:03 +01:00
|
|
|
Sprite *spr = get_sprite("part/powersurge_field");
|
|
|
|
double scale = 2 * plr->powersurge.bonus.discharge_range / spr->w;
|
2019-11-29 09:18:42 +01:00
|
|
|
double angle = rng_angle();
|
2019-02-22 00:56:03 +01:00
|
|
|
|
2020-01-06 06:22:44 +01:00
|
|
|
assert(scale > 0);
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
PARTICLE(
|
|
|
|
.sprite_ptr = spr,
|
|
|
|
.pos = plr->pos,
|
2019-11-29 09:18:42 +01:00
|
|
|
.color = color_mul_scalar(rng_bool() ? RGBA(1.5, 0.5, 0.0, 0.1) : RGBA(0.0, 0.5, 1.5, 0.1), 0.25),
|
2019-02-22 00:56:03 +01:00
|
|
|
.rule = powersurge_charge_particle,
|
2020-01-06 06:22:44 +01:00
|
|
|
.draw_rule = pdraw_timeout_fade(1, 0),
|
2019-02-22 00:56:03 +01:00
|
|
|
.timeout = 14,
|
|
|
|
.angle = angle,
|
|
|
|
.layer = LAYER_PLAYER - 1,
|
|
|
|
.flags = PFLAG_NOREFLECT,
|
2020-01-06 06:22:44 +01:00
|
|
|
.scale = scale,
|
2019-02-22 00:56:03 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
PARTICLE(
|
|
|
|
.sprite_ptr = spr,
|
|
|
|
.pos = plr->pos,
|
|
|
|
.color = RGBA(0.5, 0.5, 0.5, 0),
|
|
|
|
.rule = powersurge_charge_particle,
|
2020-01-06 06:22:44 +01:00
|
|
|
.draw_rule = pdraw_timeout_fade(1, 0),
|
2019-02-22 00:56:03 +01:00
|
|
|
.timeout = 3,
|
|
|
|
.angle = angle,
|
|
|
|
.layer = LAYER_PLAYER - 1,
|
|
|
|
.flags = PFLAG_NOREFLECT,
|
2020-01-06 06:22:44 +01:00
|
|
|
.scale = scale,
|
2019-02-22 00:56:03 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
plr->powersurge.power += plr->powersurge.bonus.gain_rate;
|
|
|
|
}
|
|
|
|
|
2020-03-13 21:54:07 +01:00
|
|
|
DEFINE_TASK(player_logic) {
|
|
|
|
Player *plr = TASK_BIND(ARGS.plr);
|
|
|
|
uint prev_inputflags = 0;
|
|
|
|
|
|
|
|
for(;;) {
|
|
|
|
YIELD;
|
|
|
|
|
|
|
|
if(prev_inputflags != plr->inputflags) {
|
|
|
|
coevent_signal(&plr->events.inputflags_changed);
|
|
|
|
prev_inputflags = plr->inputflags;
|
|
|
|
}
|
|
|
|
|
|
|
|
fapproach_p(&plr->bomb_cutin_alpha, 0, 1/200.0);
|
|
|
|
|
|
|
|
if(plr->respawntime - PLR_RESPAWN_TIME/2 == global.frames && plr->lives < 0 && global.replaymode != REPLAY_PLAY) {
|
|
|
|
stage_gameover();
|
|
|
|
}
|
|
|
|
|
|
|
|
if(plr->continuetime == global.frames) {
|
|
|
|
plr->lives = PLR_START_LIVES;
|
|
|
|
plr->bombs = PLR_START_BOMBS;
|
|
|
|
plr->point_item_value = PLR_START_PIV;
|
|
|
|
plr->life_fragments = 0;
|
|
|
|
plr->bomb_fragments = 0;
|
2020-04-26 21:27:13 +02:00
|
|
|
stats_track_continue_used(&plr->stats);
|
2020-03-13 21:54:07 +01:00
|
|
|
player_set_power(plr, 0);
|
|
|
|
stage_clear_hazards(CLEAR_HAZARDS_ALL);
|
|
|
|
spawn_items(plr->deathpos, ITEM_POWER, (int)ceil(PLR_MAX_POWER/(double)POWER_VALUE));
|
|
|
|
}
|
|
|
|
|
|
|
|
aniplayer_update(&plr->ani);
|
|
|
|
|
|
|
|
if(plr->lives < 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(plr->respawntime > global.frames) {
|
|
|
|
double x = PLR_SPAWN_POS_X;
|
|
|
|
double y = lerp(PLR_SPAWN_POS_Y, VIEWPORT_H + 64, smoothstep(0.0, 1.0, (plr->respawntime - global.frames) / (double)PLR_RESPAWN_TIME));
|
|
|
|
plr->pos = CMPLX(x, y);
|
|
|
|
stage_clear_hazards(CLEAR_HAZARDS_ALL | CLEAR_HAZARDS_NOW);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(player_is_powersurge_active(plr)) {
|
|
|
|
player_powersurge_logic(plr);
|
|
|
|
}
|
|
|
|
|
2020-04-22 23:17:40 +02:00
|
|
|
fapproach_p(&plr->focus_circle_alpha, !!(plr->inputflags & INFLAG_FOCUS), 1.0f/15.0f);
|
2020-03-13 21:54:07 +01:00
|
|
|
|
|
|
|
if(plr->mode->procs.think) {
|
|
|
|
plr->mode->procs.think(plr);
|
|
|
|
}
|
|
|
|
|
2020-04-01 22:08:40 +02:00
|
|
|
if(player_should_shoot(plr)) {
|
2020-03-13 21:54:07 +01:00
|
|
|
coevent_signal(&plr->events.shoot);
|
2020-03-25 07:52:18 +01:00
|
|
|
if(plr->mode->procs.shot) {
|
|
|
|
plr->mode->procs.shot(plr);
|
|
|
|
}
|
2020-03-13 21:54:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if(global.frames == plr->deathtime) {
|
|
|
|
player_realdeath(plr);
|
|
|
|
} else if(plr->deathtime > global.frames) {
|
|
|
|
stage_clear_hazards(CLEAR_HAZARDS_ALL | CLEAR_HAZARDS_NOW);
|
|
|
|
}
|
2012-08-03 17:06:25 +02:00
|
|
|
}
|
2020-03-13 21:54:07 +01:00
|
|
|
}
|
2012-08-03 17:06:25 +02:00
|
|
|
|
2020-03-13 21:54:07 +01:00
|
|
|
void player_logic(Player* plr) {
|
2011-06-24 19:16:05 +02:00
|
|
|
}
|
|
|
|
|
2020-04-26 21:27:13 +02:00
|
|
|
static bool player_can_bomb(Player *plr) {
|
|
|
|
return (
|
|
|
|
!player_is_bomb_active(plr)
|
|
|
|
&& (
|
|
|
|
plr->bombs > 0 ||
|
|
|
|
plr->iddqd
|
|
|
|
)
|
|
|
|
&& global.frames >= plr->respawntime
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-01-24 21:21:08 +01:00
|
|
|
static bool player_bomb(Player *plr) {
|
2012-08-16 15:07:14 +02:00
|
|
|
if(global.boss && global.boss->current && global.boss->current->type == AT_ExtraSpell)
|
2017-09-19 20:46:28 +02:00
|
|
|
return false;
|
2012-08-16 15:07:14 +02:00
|
|
|
|
2018-05-02 08:08:32 +02:00
|
|
|
int bomb_time = floor(player_property(plr, PLR_PROP_BOMB_TIME));
|
|
|
|
|
|
|
|
if(bomb_time <= 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-04-26 21:27:13 +02:00
|
|
|
if(player_can_bomb(plr)) {
|
|
|
|
stats_track_bomb_used(&plr->stats);
|
2017-04-04 02:57:38 +02:00
|
|
|
player_fail_spell(plr);
|
2019-02-22 00:56:03 +01:00
|
|
|
// player_cancel_powersurge(plr);
|
2019-03-26 16:58:38 +01:00
|
|
|
// stage_clear_hazards(CLEAR_HAZARDS_ALL);
|
2019-02-22 00:56:03 +01:00
|
|
|
|
2020-03-25 07:52:18 +01:00
|
|
|
coevent_signal(&plr->events.bomb_used);
|
|
|
|
|
|
|
|
if(plr->mode->procs.bomb) {
|
|
|
|
plr->mode->procs.bomb(plr);
|
|
|
|
}
|
|
|
|
|
2011-06-26 13:45:27 +02:00
|
|
|
plr->bombs--;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2019-03-05 12:50:29 +01:00
|
|
|
if(plr->deathtime >= global.frames) {
|
|
|
|
// death bomb - unkill the player!
|
2011-06-25 12:41:40 +02:00
|
|
|
plr->deathtime = -1;
|
2017-03-19 03:05:06 +01:00
|
|
|
|
2018-05-02 08:08:32 +02:00
|
|
|
if(plr->bombs) {
|
2017-03-19 03:05:06 +01:00
|
|
|
plr->bombs--;
|
2018-05-02 08:08:32 +02:00
|
|
|
}
|
2017-02-10 23:05:22 +01:00
|
|
|
}
|
|
|
|
|
2017-02-11 02:24:47 +01:00
|
|
|
if(plr->bombs < 0) {
|
|
|
|
plr->bombs = 0;
|
|
|
|
}
|
|
|
|
|
2018-05-02 08:08:32 +02:00
|
|
|
plr->bombtotaltime = bomb_time;
|
|
|
|
plr->recovery = global.frames + plr->bombtotaltime;
|
2019-07-03 19:50:43 +02:00
|
|
|
plr->bomb_cutin_alpha = 1;
|
2017-09-19 20:46:28 +02:00
|
|
|
|
2019-03-05 12:50:29 +01:00
|
|
|
assert(player_is_alive(plr));
|
2019-02-22 00:56:03 +01:00
|
|
|
collect_all_items(1);
|
2017-09-19 20:46:28 +02:00
|
|
|
return true;
|
2011-03-23 12:26:30 +01:00
|
|
|
}
|
2017-09-19 20:46:28 +02:00
|
|
|
|
|
|
|
return false;
|
2011-03-23 12:26:30 +01:00
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
static bool player_powersurge(Player *plr) {
|
|
|
|
if(
|
|
|
|
!player_is_alive(plr) ||
|
|
|
|
player_is_bomb_active(plr) ||
|
|
|
|
player_is_powersurge_active(plr) ||
|
|
|
|
plr->power + plr->power_overflow < PLR_POWERSURGE_POWERCOST
|
|
|
|
) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
plr->powersurge.positive = 1.0;
|
|
|
|
plr->powersurge.negative = 0.0;
|
|
|
|
plr->powersurge.time.activated = global.frames;
|
|
|
|
plr->powersurge.power = 0;
|
|
|
|
plr->powersurge.damage_accum = 0;
|
|
|
|
player_add_power(plr, -PLR_POWERSURGE_POWERCOST);
|
|
|
|
|
2019-04-16 21:04:02 +02:00
|
|
|
play_sound("powersurge_start");
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
collect_all_items(1);
|
|
|
|
stagetext_add("Power Surge!", plr->pos - 64 * I, ALIGN_CENTER, get_font("standard"), RGBA(0.75, 0.75, 0.75, 0.75), 0, 45, 10, 20);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool player_is_powersurge_active(Player *plr) {
|
|
|
|
return plr->powersurge.positive > plr->powersurge.negative;
|
|
|
|
}
|
|
|
|
|
2018-07-31 09:07:34 +02:00
|
|
|
bool player_is_bomb_active(Player *plr) {
|
|
|
|
return global.frames - plr->recovery < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool player_is_vulnerable(Player *plr) {
|
|
|
|
return global.frames - abs(plr->recovery) >= 0 && !plr->iddqd && player_is_alive(plr);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool player_is_alive(Player *plr) {
|
2019-03-05 12:50:29 +01:00
|
|
|
return plr->deathtime < global.frames && global.frames >= plr->respawntime;
|
2018-07-31 09:07:34 +02:00
|
|
|
}
|
|
|
|
|
2019-04-08 06:34:52 +02:00
|
|
|
static int powersurge_discharge(Projectile *p, int t) {
|
|
|
|
if(t == EVENT_BIRTH) {
|
|
|
|
return ACTION_ACK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(t == EVENT_DEATH) {
|
|
|
|
return ACTION_ACK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// double damage = creal(p->args[0]) / p->timeout;
|
|
|
|
double range = cimag(p->args[0]);
|
|
|
|
|
|
|
|
// ent_area_damage(p->pos, range, &(DamageInfo) { damage, DMG_PLAYER_DISCHARGE }, NULL, NULL);
|
|
|
|
stage_clear_hazards_at(p->pos, range, CLEAR_HAZARDS_ALL | CLEAR_HAZARDS_NOW | CLEAR_HAZARDS_SPAWN_VOLTAGE);
|
|
|
|
|
|
|
|
return ACTION_NONE;
|
|
|
|
}
|
|
|
|
|
2019-12-11 10:25:57 +01:00
|
|
|
static void powersurge_distortion_draw(Projectile *p, int t, ProjDrawRuleArgs args) {
|
2019-04-08 18:57:41 +02:00
|
|
|
if(config_get_int(CONFIG_POSTPROCESS) < 1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-01-06 06:22:44 +01:00
|
|
|
double radius = args[0].as_float[0] * pow(1 - t / p->timeout, 8) * (2 * t / 10.0);
|
2019-04-08 18:57:41 +02:00
|
|
|
|
|
|
|
Framebuffer *fb_aux = stage_get_fbpair(FBPAIR_FG_AUX)->front;
|
|
|
|
Framebuffer *fb_main = r_framebuffer_current();
|
|
|
|
|
|
|
|
r_framebuffer(fb_aux);
|
|
|
|
r_shader("circle_distort");
|
|
|
|
r_uniform_vec3("distortOriginRadius", creal(p->pos), VIEWPORT_H - cimag(p->pos), radius);
|
|
|
|
r_uniform_vec2("viewport", VIEWPORT_W, VIEWPORT_H);
|
|
|
|
r_blend(BLEND_NONE);
|
|
|
|
draw_framebuffer_tex(fb_main, VIEWPORT_W, VIEWPORT_H);
|
|
|
|
|
|
|
|
r_framebuffer(fb_main);
|
|
|
|
r_shader_standard();
|
|
|
|
r_color4(1.0, 0.9, 0.8, 1.0);
|
|
|
|
r_blend(BLEND_PREMUL_ALPHA);
|
2019-04-14 15:15:40 +02:00
|
|
|
stage_draw_begin_noshake();
|
2019-04-08 18:57:41 +02:00
|
|
|
draw_framebuffer_tex(fb_aux, VIEWPORT_W, VIEWPORT_H);
|
2019-04-14 15:15:40 +02:00
|
|
|
stage_draw_end_noshake();
|
2019-04-08 18:57:41 +02:00
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
static void player_powersurge_expired(Player *plr) {
|
|
|
|
plr->powersurge.time.expired = global.frames;
|
|
|
|
|
|
|
|
PowerSurgeBonus bonus;
|
|
|
|
player_powersurge_calc_bonus(plr, &bonus);
|
|
|
|
|
|
|
|
Sprite *blast = get_sprite("part/blast_huge_halo");
|
|
|
|
float scale = 2 * bonus.discharge_range / blast->w;
|
|
|
|
|
2019-04-16 21:04:02 +02:00
|
|
|
play_sound("powersurge_end");
|
|
|
|
|
2019-04-08 18:57:41 +02:00
|
|
|
PARTICLE(
|
|
|
|
.size = 1+I,
|
|
|
|
.pos = plr->pos,
|
|
|
|
.timeout = 60,
|
2020-01-06 06:22:44 +01:00
|
|
|
.draw_rule = {
|
|
|
|
powersurge_distortion_draw,
|
|
|
|
.args[0].as_float = { bonus.discharge_range },
|
|
|
|
},
|
2019-04-08 18:57:41 +02:00
|
|
|
.layer = LAYER_PLAYER,
|
|
|
|
.flags = PFLAG_REQUIREDPARTICLE | PFLAG_NOREFLECT,
|
|
|
|
);
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
PARTICLE(
|
|
|
|
.sprite_ptr = blast,
|
|
|
|
.pos = plr->pos,
|
|
|
|
.color = RGBA(0.6, 1.0, 4.4, 0.0),
|
2020-01-06 06:22:44 +01:00
|
|
|
.draw_rule = pdraw_timeout_scalefade(2, 0, 1, 0),
|
2019-02-22 00:56:03 +01:00
|
|
|
.timeout = 20,
|
2019-11-29 09:18:42 +01:00
|
|
|
.angle = rng_angle(),
|
2020-01-06 06:22:44 +01:00
|
|
|
.scale = scale,
|
2019-04-08 18:57:41 +02:00
|
|
|
.flags = PFLAG_REQUIREDPARTICLE | PFLAG_NOREFLECT,
|
2019-02-22 00:56:03 +01:00
|
|
|
);
|
|
|
|
|
2019-04-07 00:55:13 +02:00
|
|
|
player_add_points(&global.plr, bonus.score, plr->pos);
|
2019-02-22 00:56:03 +01:00
|
|
|
ent_area_damage(plr->pos, bonus.discharge_range, &(DamageInfo) { bonus.discharge_damage, DMG_PLAYER_DISCHARGE }, NULL, NULL);
|
2019-04-08 06:34:52 +02:00
|
|
|
// stage_clear_hazards_at(plr->pos, bonus.discharge_range, CLEAR_HAZARDS_ALL | CLEAR_HAZARDS_NOW | CLEAR_HAZARDS_SPAWN_VOLTAGE);
|
|
|
|
|
|
|
|
PROJECTILE(
|
|
|
|
.pos = plr->pos,
|
|
|
|
.size = 1+I,
|
2019-12-11 10:25:57 +01:00
|
|
|
.layer = LAYER_NODRAW,
|
2019-04-08 06:34:52 +02:00
|
|
|
.timeout = 10,
|
|
|
|
.type = PROJ_PLAYER,
|
|
|
|
.rule = powersurge_discharge,
|
|
|
|
.args = { CMPLX(bonus.discharge_damage, bonus.discharge_range) },
|
|
|
|
.flags = PFLAG_NOCOLLISION,
|
|
|
|
);
|
2019-02-22 00:56:03 +01:00
|
|
|
|
|
|
|
log_debug(
|
|
|
|
"Power Surge expired at %i (duration: %i); baseline = %u; score = %u; discharge = %g, dmg = %g, range = %g",
|
|
|
|
plr->powersurge.time.expired,
|
|
|
|
plr->powersurge.time.expired - plr->powersurge.time.activated,
|
|
|
|
bonus.baseline,
|
|
|
|
bonus.score,
|
|
|
|
bonus.discharge_power,
|
|
|
|
bonus.discharge_damage,
|
|
|
|
bonus.discharge_range
|
|
|
|
);
|
|
|
|
|
|
|
|
plr->powersurge.damage_done = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void player_cancel_powersurge(Player *plr) {
|
|
|
|
if(player_is_powersurge_active(plr)) {
|
|
|
|
plr->powersurge.positive = plr->powersurge.negative;
|
|
|
|
player_powersurge_expired(plr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void player_extend_powersurge(Player *plr, float pos, float neg) {
|
|
|
|
if(player_is_powersurge_active(plr)) {
|
|
|
|
plr->powersurge.positive = clamp(plr->powersurge.positive + pos, 0, 1);
|
|
|
|
plr->powersurge.negative = clamp(plr->powersurge.negative + neg, 0, 1);
|
|
|
|
|
|
|
|
if(plr->powersurge.positive <= plr->powersurge.negative) {
|
|
|
|
player_powersurge_expired(plr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-31 23:12:45 +02:00
|
|
|
double player_get_bomb_progress(Player *plr) {
|
2018-07-31 09:07:34 +02:00
|
|
|
if(!player_is_bomb_active(plr)) {
|
2017-11-26 14:06:35 +01:00
|
|
|
return 1;
|
2017-10-17 22:27:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int end_time = plr->recovery;
|
2019-03-31 23:12:45 +02:00
|
|
|
return (plr->bombtotaltime - (end_time - global.frames))/(double)plr->bombtotaltime;
|
2017-10-17 22:27:35 +02:00
|
|
|
}
|
|
|
|
|
2012-07-20 16:11:24 +02:00
|
|
|
void player_realdeath(Player *plr) {
|
2019-03-05 12:50:29 +01:00
|
|
|
plr->respawntime = global.frames + PLR_RESPAWN_TIME;
|
2017-03-06 15:24:57 +01:00
|
|
|
plr->inputflags &= ~INFLAGS_MOVE;
|
2017-10-29 23:45:24 +01:00
|
|
|
plr->deathpos = plr->pos;
|
2017-02-23 11:39:31 +01:00
|
|
|
plr->pos = VIEWPORT_W/2 + VIEWPORT_H*I+30.0*I;
|
2019-03-05 12:50:29 +01:00
|
|
|
plr->recovery = -(plr->respawntime + 150);
|
2018-01-06 10:24:46 +01:00
|
|
|
stage_clear_hazards(CLEAR_HAZARDS_ALL);
|
2017-04-04 02:57:38 +02:00
|
|
|
player_fail_spell(plr);
|
2017-09-13 15:41:28 +02:00
|
|
|
|
|
|
|
if(global.stage->type != STAGE_SPELL && global.boss && global.boss->current && global.boss->current->type == AT_ExtraSpell) {
|
|
|
|
// deaths in extra spells "don't count"
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
int total_power = plr->power + plr->power_overflow;
|
|
|
|
|
|
|
|
int drop = max(2, (total_power * 0.15) / POWER_VALUE);
|
2019-03-03 12:53:45 +01:00
|
|
|
spawn_items(plr->deathpos, ITEM_POWER, drop);
|
2017-09-13 15:41:28 +02:00
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
player_set_power(plr, total_power * 0.7);
|
2017-09-13 15:41:28 +02:00
|
|
|
plr->bombs = PLR_START_BOMBS;
|
|
|
|
plr->bomb_fragments = 0;
|
2019-02-22 00:56:03 +01:00
|
|
|
plr->voltage *= 0.9;
|
2019-04-08 05:19:42 +02:00
|
|
|
plr->lives--;
|
2020-04-26 21:27:13 +02:00
|
|
|
stats_track_life_used(&plr->stats);
|
2011-06-25 12:41:40 +02:00
|
|
|
}
|
|
|
|
|
2019-12-11 10:25:57 +01:00
|
|
|
static void player_death_effect_draw_overlay(Projectile *p, int t, ProjDrawRuleArgs args) {
|
2018-07-04 10:36:16 +02:00
|
|
|
FBPair *framebuffers = stage_get_fbpair(FBPAIR_FG);
|
|
|
|
r_framebuffer(framebuffers->front);
|
OpenGL ES 3.0 rendering backend (#148)
* First steps towards shader transpilation
Needs to be manually enabled via -Dshader_transpiler=true.
Requires shaderc. https://github.com/google/shaderc
Not yet functional due to missing SPIRV-Cross integration. SPIRV-Cross
currently does not have an official C API, and crossc is too minimal to
be useful. The current plan is to extend crossc and vendor it, while
also sending PRs upstream.
* Integrate crossc; shader transpilation for GLES now works
* fix leak
* gles30 backend now playable on Mesa with 3.2 context
Some rendering issues are present. Identified so far:
- Marisa's lasers are invisible
- Death effect looks wrong
Also, a small pixmap manipulation library has been written, and the
texture uploading API redesigned around it.
* fix marisa lasers in GLES (uniform name clashed with builtin)
* fix player death effect in GLES (another name clash)
* Dump ANGLE's translated shader code in debug log
* fix screenshots
* Drop support for triangle fans, switch to strips
Fans offer no advantage over strips, and they've been removed in D3D10+,
so ANGLE has to emulate them.
* crude workaround for an ANGLE bug
* Re-enable GL debug labels, fix an issue with them that affected ANGLE (but was always technically a bug)
* fix race condition in shaderc initialization
* New SDL_RWops interface for vertex buffers
* Optimize VBO streaming via buffering updates
Measurable performance improvement even with the main gl33 renderer,
drastic improvement with ANGLE.
* Fix the depth texture binding problem under ANGLE
Apparently it hates GL_DEPTH_COMPONENT16 for some reason. Sized internal
formats are not supported in GLES 2.0 anyway, so not using them is
probably a good idea.
* fix GLES2.0 segfault (the backend still doesn't work, though)
* dump GL extensions at info log level, not debug
* get around a Mesa bug; more correct texture format table for GLES2
* Correct GLES3 texture format table according to the spec
Not a Mesa bug after all
* require crossc>=1.5.0, fallback to subproject
* Request at least 8bit per color channel in GL backends
* Forbid lto for static windows builds with shader_transpiler=true
* fix edge case segfault
* Add basic ANGLE bundling support to the build system
Windows only, and no NSIS support yet
* Fix various windows-related build system and installer brokenness
* Disable gles backends by default
* update documentation
2018-10-02 00:36:10 +02:00
|
|
|
r_uniform_sampler("noise_tex", "static");
|
2018-05-18 20:15:11 +02:00
|
|
|
r_uniform_int("frames", global.frames);
|
|
|
|
r_uniform_float("progress", t / p->timeout);
|
|
|
|
r_uniform_vec2("origin", creal(p->pos), VIEWPORT_H - cimag(p->pos));
|
|
|
|
r_uniform_vec2("clear_origin", creal(global.plr.pos), VIEWPORT_H - cimag(global.plr.pos));
|
|
|
|
r_uniform_vec2("viewport", VIEWPORT_W, VIEWPORT_H);
|
2020-02-15 18:45:09 +01:00
|
|
|
r_uniform_float("size", hypotf(VIEWPORT_W, VIEWPORT_H));
|
2018-07-04 10:36:16 +02:00
|
|
|
draw_framebuffer_tex(framebuffers->back, VIEWPORT_W, VIEWPORT_H);
|
|
|
|
fbpair_swap(framebuffers);
|
2018-05-19 01:24:49 +02:00
|
|
|
|
|
|
|
// This change must propagate, hence the r_state salsa. Yes, pop then push, I know what I'm doing.
|
|
|
|
r_state_pop();
|
2018-07-04 10:36:16 +02:00
|
|
|
r_framebuffer(framebuffers->back);
|
2018-05-19 01:24:49 +02:00
|
|
|
r_state_push();
|
|
|
|
}
|
|
|
|
|
2019-12-11 10:25:57 +01:00
|
|
|
static void player_death_effect_draw_sprite(Projectile *p, int t, ProjDrawRuleArgs args) {
|
2018-05-19 01:24:49 +02:00
|
|
|
float s = t / p->timeout;
|
|
|
|
float stretch_range = 3, sx, sy;
|
|
|
|
|
2020-01-04 04:24:24 +01:00
|
|
|
s = glm_ease_quad_in(s);
|
|
|
|
|
|
|
|
sx = (1 - pow(2 * pow(1 - s, 4) - 1, 4));
|
|
|
|
sx = lerp(1 + (stretch_range - 1) * sx, stretch_range * sx, s);
|
|
|
|
sy = 1 + 2 * (stretch_range - 1) * pow(s, 4);
|
2018-05-19 01:24:49 +02:00
|
|
|
|
|
|
|
if(sx <= 0 || sy <= 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-01-04 04:24:24 +01:00
|
|
|
SpriteParamsBuffer spbuf;
|
|
|
|
SpriteParams sp = projectile_sprite_params(p, &spbuf);
|
|
|
|
sp.scale.x *= sx;
|
|
|
|
sp.scale.y *= sy;
|
|
|
|
sp.rotation.angle = 0;
|
|
|
|
r_draw_sprite(&sp);
|
2018-05-19 01:24:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int player_death_effect(Projectile *p, int t) {
|
|
|
|
if(t < 0) {
|
|
|
|
if(t == EVENT_DEATH) {
|
|
|
|
for(int i = 0; i < 12; ++i) {
|
2019-11-29 09:18:42 +01:00
|
|
|
RNG_ARRAY(R, 4);
|
2018-05-19 01:24:49 +02:00
|
|
|
PARTICLE(
|
2019-04-11 06:12:17 +02:00
|
|
|
.proto = pp_blast,
|
2019-11-29 09:18:42 +01:00
|
|
|
.pos = p->pos + vrng_range(R[0], 2, 3) * vrng_dir(R[1]),
|
2018-07-23 19:07:59 +02:00
|
|
|
.color = RGBA(0.15, 0.2, 0.5, 0),
|
2019-11-29 09:18:42 +01:00
|
|
|
.timeout = i + vrng_range(R[2], 10, 14),
|
2020-01-04 03:08:15 +01:00
|
|
|
.draw_rule = pdraw_timeout_scalefade(0, 1, 1, 0),
|
2019-11-29 09:18:42 +01:00
|
|
|
.angle = vrng_angle(R[3]),
|
2018-05-19 01:24:49 +02:00
|
|
|
.flags = PFLAG_NOREFLECT,
|
|
|
|
.layer = LAYER_OVERLAY,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ACTION_ACK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ACTION_NONE;
|
2018-05-18 20:15:11 +02:00
|
|
|
}
|
|
|
|
|
2012-07-20 16:11:24 +02:00
|
|
|
void player_death(Player *plr) {
|
2019-02-22 00:56:03 +01:00
|
|
|
if(!player_is_vulnerable(plr) || stage_is_cleared()) {
|
2018-01-02 08:36:14 +01:00
|
|
|
return;
|
2018-07-31 09:07:34 +02:00
|
|
|
}
|
2018-01-02 08:36:14 +01:00
|
|
|
|
2018-07-31 09:07:34 +02:00
|
|
|
play_sound("death");
|
2017-11-10 21:49:16 +01:00
|
|
|
|
2018-07-31 09:07:34 +02:00
|
|
|
for(int i = 0; i < 60; i++) {
|
2019-11-29 09:18:42 +01:00
|
|
|
RNG_ARRAY(R, 2);
|
2017-11-10 21:49:16 +01:00
|
|
|
PARTICLE(
|
2018-07-31 09:07:34 +02:00
|
|
|
.sprite = "flare",
|
2017-11-10 21:49:16 +01:00
|
|
|
.pos = plr->pos,
|
2018-07-31 09:07:34 +02:00
|
|
|
.timeout = 40,
|
2020-01-03 23:00:00 +01:00
|
|
|
.draw_rule = pdraw_timeout_scale(2, 0.01),
|
2020-01-04 04:24:24 +01:00
|
|
|
.move = move_linear(vrng_range(R[0], 3, 10) * vrng_dir(R[1])),
|
2018-07-31 09:07:34 +02:00
|
|
|
.flags = PFLAG_NOREFLECT,
|
2018-05-18 20:15:11 +02:00
|
|
|
);
|
2018-07-31 09:07:34 +02:00
|
|
|
}
|
2018-05-18 20:15:11 +02:00
|
|
|
|
2018-07-31 09:07:34 +02:00
|
|
|
stage_clear_hazards(CLEAR_HAZARDS_ALL);
|
2018-05-19 01:24:49 +02:00
|
|
|
|
2018-07-31 09:07:34 +02:00
|
|
|
PARTICLE(
|
|
|
|
.sprite = "blast",
|
|
|
|
.pos = plr->pos,
|
|
|
|
.color = RGBA(0.5, 0.15, 0.15, 0),
|
|
|
|
.timeout = 35,
|
2020-01-04 03:08:15 +01:00
|
|
|
.draw_rule = pdraw_timeout_scalefade(0, 3.4, 1, 0),
|
|
|
|
.angle = rng_angle(),
|
|
|
|
.flags = PFLAG_NOREFLECT | PFLAG_REQUIREDPARTICLE | PFLAG_NOMOVE | PFLAG_MANUALANGLE,
|
2018-07-31 09:07:34 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
PARTICLE(
|
|
|
|
.pos = plr->pos,
|
|
|
|
.size = 1+I,
|
|
|
|
.timeout = 90,
|
|
|
|
.draw_rule = player_death_effect_draw_overlay,
|
|
|
|
.blend = BLEND_NONE,
|
|
|
|
.flags = PFLAG_NOREFLECT | PFLAG_REQUIREDPARTICLE,
|
2020-01-04 04:24:24 +01:00
|
|
|
.layer = LAYER_OVERLAY,
|
2018-07-31 09:07:34 +02:00
|
|
|
.shader = "player_death",
|
|
|
|
);
|
|
|
|
|
|
|
|
PARTICLE(
|
|
|
|
.sprite_ptr = aniplayer_get_frame(&plr->ani),
|
|
|
|
.pos = plr->pos,
|
2020-01-04 04:24:24 +01:00
|
|
|
.timeout = 38,
|
2018-07-31 09:07:34 +02:00
|
|
|
.rule = player_death_effect,
|
|
|
|
.draw_rule = player_death_effect_draw_sprite,
|
|
|
|
.flags = PFLAG_NOREFLECT | PFLAG_REQUIREDPARTICLE,
|
|
|
|
.layer = LAYER_PLAYER_FOCUS, // LAYER_OVERLAY | 1,
|
|
|
|
);
|
|
|
|
|
|
|
|
plr->deathtime = global.frames + floor(player_property(plr, PLR_PROP_DEATHBOMB_WINDOW));
|
2019-02-22 00:56:03 +01:00
|
|
|
|
|
|
|
if(player_is_powersurge_active(plr)) {
|
|
|
|
player_cancel_powersurge(plr);
|
|
|
|
// player_bomb(plr);
|
|
|
|
}
|
2011-07-02 11:39:36 +02:00
|
|
|
}
|
2012-07-14 16:37:52 +02:00
|
|
|
|
2018-07-30 09:04:09 +02:00
|
|
|
static DamageResult ent_damage_player(EntityInterface *ent, const DamageInfo *dmg) {
|
|
|
|
Player *plr = ENT_CAST(ent, Player);
|
|
|
|
|
|
|
|
if(
|
2018-07-31 09:07:34 +02:00
|
|
|
!player_is_vulnerable(plr) ||
|
2018-07-30 09:04:09 +02:00
|
|
|
(dmg->type != DMG_ENEMY_SHOT && dmg->type != DMG_ENEMY_COLLISION)
|
|
|
|
) {
|
|
|
|
return DMG_RESULT_IMMUNE;
|
|
|
|
}
|
|
|
|
|
|
|
|
player_death(plr);
|
|
|
|
return DMG_RESULT_OK;
|
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
void player_damage_hook(Player *plr, EntityInterface *target, DamageInfo *dmg) {
|
|
|
|
if(player_is_powersurge_active(plr) && dmg->type == DMG_PLAYER_SHOT) {
|
|
|
|
dmg->amount *= 1.2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-06 15:24:57 +01:00
|
|
|
static PlrInputFlag key_to_inflag(KeyIndex key) {
|
2012-07-14 16:37:52 +02:00
|
|
|
switch(key) {
|
2017-11-21 22:02:32 +01:00
|
|
|
case KEY_UP: return INFLAG_UP;
|
|
|
|
case KEY_DOWN: return INFLAG_DOWN;
|
|
|
|
case KEY_LEFT: return INFLAG_LEFT;
|
|
|
|
case KEY_RIGHT: return INFLAG_RIGHT;
|
|
|
|
case KEY_FOCUS: return INFLAG_FOCUS;
|
|
|
|
case KEY_SHOT: return INFLAG_SHOT;
|
|
|
|
case KEY_SKIP: return INFLAG_SKIP;
|
2017-03-06 15:24:57 +01:00
|
|
|
default: return 0;
|
2012-07-14 16:37:52 +02:00
|
|
|
}
|
2017-03-06 15:24:57 +01:00
|
|
|
}
|
|
|
|
|
2019-01-24 21:21:08 +01:00
|
|
|
static bool player_updateinputflags(Player *plr, PlrInputFlag flags) {
|
2017-10-01 02:48:55 +02:00
|
|
|
if(flags == plr->inputflags) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-09-19 20:46:28 +02:00
|
|
|
plr->inputflags = flags;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-01-24 21:21:08 +01:00
|
|
|
static bool player_updateinputflags_moveonly(Player *plr, PlrInputFlag flags) {
|
2017-09-19 20:46:28 +02:00
|
|
|
return player_updateinputflags(plr, (flags & INFLAGS_MOVE) | (plr->inputflags & ~INFLAGS_MOVE));
|
|
|
|
}
|
|
|
|
|
2019-01-24 21:21:08 +01:00
|
|
|
static bool player_setinputflag(Player *plr, KeyIndex key, bool mode) {
|
2017-09-19 20:46:28 +02:00
|
|
|
PlrInputFlag newflags = plr->inputflags;
|
|
|
|
PlrInputFlag keyflag = key_to_inflag(key);
|
|
|
|
|
|
|
|
if(!keyflag) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-03-06 15:24:57 +01:00
|
|
|
|
2017-09-19 20:46:28 +02:00
|
|
|
if(mode) {
|
|
|
|
newflags |= keyflag;
|
2012-08-12 18:00:56 +02:00
|
|
|
} else {
|
2017-09-19 20:46:28 +02:00
|
|
|
newflags &= ~keyflag;
|
2012-08-12 18:00:56 +02:00
|
|
|
}
|
2017-09-19 20:46:28 +02:00
|
|
|
|
2017-10-06 12:19:17 +02:00
|
|
|
return player_updateinputflags(plr, newflags);
|
2012-07-14 16:37:52 +02:00
|
|
|
}
|
|
|
|
|
2017-09-19 20:46:28 +02:00
|
|
|
static bool player_set_axis(int *aptr, uint16_t value) {
|
|
|
|
int16_t new = (int16_t)value;
|
|
|
|
|
|
|
|
if(*aptr == new) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*aptr = new;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-10-31 10:48:30 +01:00
|
|
|
void player_event(Player *plr, uint8_t type, uint16_t value, bool *out_useful, bool *out_cheat) {
|
2017-09-19 20:46:28 +02:00
|
|
|
bool useful = true;
|
2017-10-29 23:45:24 +01:00
|
|
|
bool cheat = false;
|
2017-10-31 10:48:30 +01:00
|
|
|
bool is_replay = global.replaymode == REPLAY_PLAY;
|
2017-09-19 20:46:28 +02:00
|
|
|
|
2012-07-14 16:37:52 +02:00
|
|
|
switch(type) {
|
|
|
|
case EV_PRESS:
|
2019-07-03 19:50:43 +02:00
|
|
|
if(dialog_is_active(global.dialog) && (value == KEY_SHOT || value == KEY_BOMB)) {
|
2020-01-23 01:23:35 +01:00
|
|
|
useful = dialog_page(global.dialog);
|
2017-09-19 20:46:28 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(value) {
|
2012-07-14 16:37:52 +02:00
|
|
|
case KEY_BOMB:
|
2017-09-19 20:46:28 +02:00
|
|
|
useful = player_bomb(plr);
|
2012-07-14 16:37:52 +02:00
|
|
|
break;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
case KEY_SPECIAL:
|
2019-07-03 19:50:43 +02:00
|
|
|
if(dialog_is_active(global.dialog)) {
|
2019-02-28 11:00:35 +01:00
|
|
|
useful = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
useful = player_powersurge(plr);
|
|
|
|
|
|
|
|
if(!useful/* && plr->iddqd*/) {
|
|
|
|
player_cancel_powersurge(plr);
|
|
|
|
useful = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2017-02-11 02:24:47 +01:00
|
|
|
case KEY_IDDQD:
|
2017-02-11 02:56:34 +01:00
|
|
|
plr->iddqd = !plr->iddqd;
|
2017-10-29 23:45:24 +01:00
|
|
|
cheat = true;
|
2017-02-11 02:24:47 +01:00
|
|
|
break;
|
|
|
|
|
2017-03-06 13:02:46 +01:00
|
|
|
case KEY_POWERUP:
|
2019-02-22 00:56:03 +01:00
|
|
|
useful = player_add_power(plr, 100);
|
2017-10-29 23:45:24 +01:00
|
|
|
cheat = true;
|
2017-03-06 13:02:46 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case KEY_POWERDOWN:
|
2019-02-22 00:56:03 +01:00
|
|
|
useful = player_add_power(plr, -100);
|
2017-10-29 23:45:24 +01:00
|
|
|
cheat = true;
|
2017-03-06 13:02:46 +01:00
|
|
|
break;
|
|
|
|
|
2012-07-14 16:37:52 +02:00
|
|
|
default:
|
2017-09-19 20:46:28 +02:00
|
|
|
useful = player_setinputflag(plr, value, true);
|
2012-07-14 16:37:52 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2012-07-14 16:37:52 +02:00
|
|
|
case EV_RELEASE:
|
2017-10-23 12:10:40 +02:00
|
|
|
useful = player_setinputflag(plr, value, false);
|
2012-07-14 16:37:52 +02:00
|
|
|
break;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2012-08-15 16:36:39 +02:00
|
|
|
case EV_AXIS_LR:
|
2017-09-19 20:46:28 +02:00
|
|
|
useful = player_set_axis(&plr->axis_lr, value);
|
2012-08-15 16:36:39 +02:00
|
|
|
break;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2012-08-15 16:36:39 +02:00
|
|
|
case EV_AXIS_UD:
|
2017-09-19 20:46:28 +02:00
|
|
|
useful = player_set_axis(&plr->axis_ud, value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EV_INFLAGS:
|
|
|
|
useful = player_updateinputflags(plr, value);
|
2012-08-15 16:36:39 +02:00
|
|
|
break;
|
2017-03-21 03:28:35 +01:00
|
|
|
|
2017-10-29 23:45:24 +01:00
|
|
|
case EV_CONTINUE:
|
|
|
|
// continuing in the same frame will desync the replay,
|
|
|
|
// so schedule it for the next one
|
|
|
|
plr->continuetime = global.frames + 1;
|
|
|
|
useful = true;
|
|
|
|
break;
|
|
|
|
|
2017-03-21 03:28:35 +01:00
|
|
|
default:
|
2017-09-19 20:46:28 +02:00
|
|
|
log_warn("Can not handle event: [%i:%02x:%04x]", global.frames, type, value);
|
|
|
|
useful = false;
|
2017-03-21 03:28:35 +01:00
|
|
|
break;
|
2012-07-14 16:37:52 +02:00
|
|
|
}
|
2017-09-19 20:46:28 +02:00
|
|
|
|
2017-10-31 10:48:30 +01:00
|
|
|
if(is_replay) {
|
2017-10-29 23:45:24 +01:00
|
|
|
if(!useful) {
|
|
|
|
log_warn("Useless event in replay: [%i:%02x:%04x]", global.frames, type, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(cheat) {
|
|
|
|
log_warn("Cheat event in replay: [%i:%02x:%04x]", global.frames, type, value);
|
|
|
|
|
2018-01-12 19:26:07 +01:00
|
|
|
if( !(global.replay.flags & REPLAY_GFLAG_CHEATS) ||
|
|
|
|
!(global.replay_stage->flags & REPLAY_SFLAG_CHEATS)) {
|
2017-10-29 23:45:24 +01:00
|
|
|
log_warn("...but this replay was NOT properly cheat-flagged! Not cool, not cool at all");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(type == EV_CONTINUE && (
|
2018-01-12 19:26:07 +01:00
|
|
|
!(global.replay.flags & REPLAY_GFLAG_CONTINUES) ||
|
|
|
|
!(global.replay_stage->flags & REPLAY_SFLAG_CONTINUES))) {
|
2017-10-29 23:45:24 +01:00
|
|
|
log_warn("Continue event in replay: [%i:%02x:%04x], but this replay was not properly continue-flagged", global.frames, type, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(out_useful) {
|
|
|
|
*out_useful = useful;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(out_cheat) {
|
|
|
|
*out_cheat = cheat;
|
|
|
|
}
|
2017-09-19 20:46:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool player_event_with_replay(Player *plr, uint8_t type, uint16_t value) {
|
2017-10-29 23:45:24 +01:00
|
|
|
bool useful, cheat;
|
2017-09-19 20:46:28 +02:00
|
|
|
assert(global.replaymode == REPLAY_RECORD);
|
2017-10-31 10:48:30 +01:00
|
|
|
|
|
|
|
if(config_get_int(CONFIG_SHOT_INVERTED) && value == KEY_SHOT && (type == EV_PRESS || type == EV_RELEASE)) {
|
|
|
|
type = type == EV_PRESS ? EV_RELEASE : EV_PRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
player_event(plr, type, value, &useful, &cheat);
|
2017-09-19 20:46:28 +02:00
|
|
|
|
2017-10-29 23:45:24 +01:00
|
|
|
if(useful) {
|
2017-09-19 20:46:28 +02:00
|
|
|
replay_stage_event(global.replay_stage, global.frames, type, value);
|
2017-10-29 23:45:24 +01:00
|
|
|
|
|
|
|
if(type == EV_CONTINUE) {
|
|
|
|
global.replay.flags |= REPLAY_GFLAG_CONTINUES;
|
|
|
|
global.replay_stage->flags |= REPLAY_SFLAG_CONTINUES;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(cheat) {
|
|
|
|
global.replay.flags |= REPLAY_GFLAG_CHEATS;
|
|
|
|
global.replay_stage->flags |= REPLAY_SFLAG_CHEATS;
|
|
|
|
}
|
|
|
|
|
2017-09-19 20:46:28 +02:00
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
log_debug("Useless event discarded: [%i:%02x:%04x]", global.frames, type, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2012-07-14 16:37:52 +02:00
|
|
|
}
|
|
|
|
|
2012-08-15 16:36:39 +02:00
|
|
|
// free-axis movement
|
2019-01-24 21:21:08 +01:00
|
|
|
static bool player_applymovement_gamepad(Player *plr) {
|
2017-02-04 14:41:15 +01:00
|
|
|
if(!plr->axis_lr && !plr->axis_ud) {
|
|
|
|
if(plr->gamepadmove) {
|
2017-02-11 04:52:08 +01:00
|
|
|
plr->gamepadmove = false;
|
2017-03-06 15:24:57 +01:00
|
|
|
plr->inputflags &= ~INFLAGS_MOVE;
|
2017-02-04 14:41:15 +01:00
|
|
|
}
|
2017-02-11 04:52:08 +01:00
|
|
|
return false;
|
2017-02-04 14:41:15 +01:00
|
|
|
}
|
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
cmplx direction = (
|
2018-01-14 04:15:25 +01:00
|
|
|
gamepad_normalize_axis_value(plr->axis_lr) +
|
|
|
|
gamepad_normalize_axis_value(plr->axis_ud) * I
|
|
|
|
);
|
|
|
|
|
|
|
|
if(cabs(direction) > 1) {
|
2012-08-15 17:03:53 +02:00
|
|
|
direction /= cabs(direction);
|
2018-01-14 04:15:25 +01:00
|
|
|
}
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2017-03-06 16:32:51 +01:00
|
|
|
int sr = sign(creal(direction));
|
|
|
|
int si = sign(cimag(direction));
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2017-09-19 20:46:28 +02:00
|
|
|
player_updateinputflags_moveonly(plr,
|
|
|
|
(INFLAG_UP * (si == -1)) |
|
|
|
|
(INFLAG_DOWN * (si == 1)) |
|
|
|
|
(INFLAG_LEFT * (sr == -1)) |
|
|
|
|
(INFLAG_RIGHT * (sr == 1))
|
|
|
|
);
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2017-02-04 14:41:15 +01:00
|
|
|
if(direction) {
|
2017-02-11 04:52:08 +01:00
|
|
|
plr->gamepadmove = true;
|
2012-08-15 16:36:39 +02:00
|
|
|
player_move(&global.plr, direction);
|
2017-02-04 14:41:15 +01:00
|
|
|
}
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2017-02-11 04:52:08 +01:00
|
|
|
return true;
|
2012-08-15 16:36:39 +02:00
|
|
|
}
|
|
|
|
|
2018-05-05 11:23:56 +02:00
|
|
|
static const char *moveseqname(int dir) {
|
2018-04-12 16:08:48 +02:00
|
|
|
switch(dir) {
|
2018-05-05 11:23:56 +02:00
|
|
|
case -1: return "left";
|
|
|
|
case 0: return "main";
|
|
|
|
case 1: return "right";
|
2018-04-12 16:08:48 +02:00
|
|
|
default: log_fatal("Invalid player animation dir given");
|
|
|
|
}
|
2018-05-05 11:23:56 +02:00
|
|
|
}
|
2018-04-12 16:08:48 +02:00
|
|
|
|
2018-05-05 11:23:56 +02:00
|
|
|
static void player_ani_moving(Player *plr, int dir) {
|
2018-04-12 16:08:48 +02:00
|
|
|
if(plr->lastmovesequence == dir)
|
|
|
|
return;
|
2018-05-05 11:23:56 +02:00
|
|
|
const char *seqname = moveseqname(dir);
|
|
|
|
const char *lastseqname = moveseqname(plr->lastmovesequence);
|
|
|
|
|
|
|
|
char *transition = strjoin(lastseqname,"2",seqname,NULL);
|
|
|
|
|
|
|
|
aniplayer_hard_switch(&plr->ani,transition,1);
|
|
|
|
aniplayer_queue(&plr->ani,seqname,0);
|
2018-04-12 16:08:48 +02:00
|
|
|
plr->lastmovesequence = dir;
|
2018-05-05 11:23:56 +02:00
|
|
|
free(transition);
|
2017-10-03 17:25:38 +02:00
|
|
|
}
|
|
|
|
|
2012-08-17 20:58:23 +02:00
|
|
|
void player_applymovement(Player *plr) {
|
2018-05-13 15:08:58 +02:00
|
|
|
plr->velocity = 0;
|
|
|
|
|
2019-03-05 12:50:29 +01:00
|
|
|
if(!player_is_alive(plr))
|
2012-07-14 16:37:52 +02:00
|
|
|
return;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2017-02-11 04:52:08 +01:00
|
|
|
bool gamepad = player_applymovement_gamepad(plr);
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2018-04-12 16:08:48 +02:00
|
|
|
int up = plr->inputflags & INFLAG_UP;
|
|
|
|
int down = plr->inputflags & INFLAG_DOWN;
|
|
|
|
int left = plr->inputflags & INFLAG_LEFT;
|
|
|
|
int right = plr->inputflags & INFLAG_RIGHT;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2012-07-14 16:37:52 +02:00
|
|
|
if(left && !right) {
|
2018-04-12 16:08:48 +02:00
|
|
|
player_ani_moving(plr,-1);
|
2012-07-14 16:37:52 +02:00
|
|
|
} else if(right && !left) {
|
2018-04-12 16:08:48 +02:00
|
|
|
player_ani_moving(plr,1);
|
|
|
|
} else {
|
|
|
|
player_ani_moving(plr,0);
|
2012-08-15 16:36:39 +02:00
|
|
|
}
|
2018-05-13 15:08:58 +02:00
|
|
|
|
|
|
|
if(gamepad) {
|
2012-08-15 16:36:39 +02:00
|
|
|
return;
|
2018-05-13 15:08:58 +02:00
|
|
|
}
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
cmplx direction = 0;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2018-01-12 19:26:07 +01:00
|
|
|
if(up) direction -= 1.0*I;
|
|
|
|
if(down) direction += 1.0*I;
|
2018-05-18 20:15:11 +02:00
|
|
|
if(left) direction -= 1.0;
|
|
|
|
if(right) direction += 1.0;
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2012-07-20 16:11:24 +02:00
|
|
|
if(cabs(direction))
|
|
|
|
direction /= cabs(direction);
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2012-07-14 16:37:52 +02:00
|
|
|
if(direction)
|
2012-07-20 16:11:24 +02:00
|
|
|
player_move(&global.plr, direction);
|
2012-08-17 20:58:23 +02:00
|
|
|
}
|
|
|
|
|
2017-09-19 20:46:28 +02:00
|
|
|
void player_fix_input(Player *plr) {
|
|
|
|
// correct input state to account for any events we might have missed,
|
|
|
|
// usually because the pause menu ate them up
|
|
|
|
|
|
|
|
PlrInputFlag newflags = plr->inputflags;
|
2017-10-31 10:48:30 +01:00
|
|
|
bool invert_shot = config_get_int(CONFIG_SHOT_INVERTED);
|
2017-03-07 00:57:14 +01:00
|
|
|
|
2017-03-06 15:24:57 +01:00
|
|
|
for(KeyIndex key = KEYIDX_FIRST; key <= KEYIDX_LAST; ++key) {
|
|
|
|
int flag = key_to_inflag(key);
|
|
|
|
|
2017-09-19 20:46:28 +02:00
|
|
|
if(flag && !(plr->gamepadmove && (flag & INFLAGS_MOVE))) {
|
2017-03-06 15:24:57 +01:00
|
|
|
bool flagset = plr->inputflags & flag;
|
|
|
|
bool keyheld = gamekeypressed(key);
|
|
|
|
|
2017-10-31 10:48:30 +01:00
|
|
|
if(invert_shot && key == KEY_SHOT) {
|
|
|
|
keyheld = !keyheld;
|
|
|
|
}
|
|
|
|
|
2017-03-06 15:24:57 +01:00
|
|
|
if(flagset && !keyheld) {
|
2017-09-19 20:46:28 +02:00
|
|
|
newflags &= ~flag;
|
2017-03-06 15:24:57 +01:00
|
|
|
} else if(!flagset && keyheld) {
|
2017-09-19 20:46:28 +02:00
|
|
|
newflags |= flag;
|
2017-03-06 15:24:57 +01:00
|
|
|
}
|
2017-09-19 20:46:28 +02:00
|
|
|
}
|
|
|
|
}
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2017-09-19 20:46:28 +02:00
|
|
|
if(newflags != plr->inputflags) {
|
|
|
|
player_event_with_replay(plr, EV_INFLAGS, newflags);
|
|
|
|
}
|
|
|
|
|
2018-01-18 10:30:42 +01:00
|
|
|
int axis_lr = gamepad_player_axis_value(PLRAXIS_LR);
|
|
|
|
int axis_ud = gamepad_player_axis_value(PLRAXIS_UD);
|
2017-09-19 20:46:28 +02:00
|
|
|
|
2018-01-18 10:30:42 +01:00
|
|
|
if(plr->axis_lr != axis_lr) {
|
|
|
|
player_event_with_replay(plr, EV_AXIS_LR, axis_lr);
|
|
|
|
}
|
2017-09-19 20:46:28 +02:00
|
|
|
|
2018-01-18 10:30:42 +01:00
|
|
|
if(plr->axis_ud != axis_ud) {
|
|
|
|
player_event_with_replay(plr, EV_AXIS_UD, axis_ud);
|
2012-07-29 15:27:15 +02:00
|
|
|
}
|
2012-07-14 16:37:52 +02:00
|
|
|
}
|
2012-08-14 16:14:53 +02:00
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
void player_graze(Player *plr, cmplx pos, int pts, int effect_intensity, const Color *color) {
|
2019-02-22 00:56:03 +01:00
|
|
|
if(++plr->graze >= PLR_MAX_GRAZE) {
|
|
|
|
log_debug("Graze counter overflow");
|
|
|
|
plr->graze = PLR_MAX_GRAZE;
|
2018-01-09 19:49:30 +01:00
|
|
|
}
|
2017-03-25 19:18:24 +01:00
|
|
|
|
2019-01-04 23:59:39 +01:00
|
|
|
pos = (pos + plr->pos) * 0.5;
|
|
|
|
|
2019-04-07 00:55:13 +02:00
|
|
|
player_add_points(plr, pts, pos);
|
2012-08-14 16:14:53 +02:00
|
|
|
play_sound("graze");
|
2017-02-10 23:05:22 +01:00
|
|
|
|
2019-01-04 23:59:39 +01:00
|
|
|
Color *c = COLOR_COPY(color);
|
|
|
|
color_add(c, RGBA(1, 1, 1, 1));
|
|
|
|
color_mul_scalar(c, 0.5);
|
|
|
|
c->a = 0;
|
|
|
|
|
2018-01-06 10:44:27 +01:00
|
|
|
for(int i = 0; i < effect_intensity; ++i) {
|
2019-11-29 09:18:42 +01:00
|
|
|
RNG_ARRAY(R, 4);
|
2017-11-10 21:49:16 +01:00
|
|
|
PARTICLE(
|
2019-01-04 23:59:39 +01:00
|
|
|
.sprite = "graze",
|
|
|
|
.color = c,
|
2017-11-10 21:49:16 +01:00
|
|
|
.pos = pos,
|
2020-01-06 06:22:44 +01:00
|
|
|
.draw_rule = pdraw_timeout_scalefade_exp(1, 0, 1, 0, 2),
|
|
|
|
.move = move_asymptotic_simple(0.2 * vrng_range(R[0], 1, 6) * vrng_dir(R[1]), 16 * (1 + 0.5 * vrng_sreal(R[3]))),
|
2019-11-29 09:18:42 +01:00
|
|
|
.timeout = vrng_range(R[2], 4, 29),
|
2018-01-11 21:16:18 +01:00
|
|
|
.flags = PFLAG_NOREFLECT,
|
2019-01-04 23:59:39 +01:00
|
|
|
// .layer = LAYER_PARTICLE_LOW,
|
2017-11-10 21:49:16 +01:00
|
|
|
);
|
2019-01-04 23:59:39 +01:00
|
|
|
|
|
|
|
color_mul_scalar(c, 0.4);
|
2012-08-14 16:14:53 +02:00
|
|
|
}
|
2019-02-22 00:56:03 +01:00
|
|
|
|
2019-03-03 12:53:45 +01:00
|
|
|
spawn_items(pos, ITEM_POWER_MINI, 1);
|
2012-08-14 16:14:53 +02:00
|
|
|
}
|
2017-03-11 04:41:57 +01:00
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
static void player_add_fragments(Player *plr, int frags, int *pwhole, int *pfrags, int maxfrags, int maxwhole, const char *fragsnd, const char *upsnd, int score_per_excess) {
|
|
|
|
int total_frags = *pfrags + maxfrags * *pwhole;
|
|
|
|
int excess_frags = total_frags + frags - maxwhole * maxfrags;
|
|
|
|
|
|
|
|
if(excess_frags > 0) {
|
2019-04-07 00:55:13 +02:00
|
|
|
player_add_points(plr, excess_frags * score_per_excess, plr->pos);
|
2019-02-22 00:56:03 +01:00
|
|
|
frags -= excess_frags;
|
2017-03-21 11:09:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
*pfrags += frags;
|
|
|
|
int up = *pfrags / maxfrags;
|
|
|
|
|
|
|
|
*pwhole += up;
|
|
|
|
*pfrags %= maxfrags;
|
|
|
|
|
|
|
|
if(up) {
|
|
|
|
play_sound(upsnd);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(frags) {
|
|
|
|
// FIXME: when we have the extra life/bomb sounds,
|
|
|
|
// don't play this if upsnd was just played.
|
|
|
|
play_sound(fragsnd);
|
|
|
|
}
|
|
|
|
|
2017-12-01 14:09:30 +01:00
|
|
|
if(*pwhole >= maxwhole) {
|
2017-03-21 11:09:32 +01:00
|
|
|
*pwhole = maxwhole;
|
2017-12-01 14:09:30 +01:00
|
|
|
*pfrags = 0;
|
2017-03-21 11:09:32 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void player_add_life_fragments(Player *plr, int frags) {
|
2019-04-03 20:48:25 +02:00
|
|
|
if(global.stage->type == STAGE_SPELL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
player_add_fragments(
|
|
|
|
plr,
|
|
|
|
frags,
|
|
|
|
&plr->lives,
|
|
|
|
&plr->life_fragments,
|
|
|
|
PLR_MAX_LIFE_FRAGMENTS,
|
|
|
|
PLR_MAX_LIVES,
|
2017-03-21 11:09:32 +01:00
|
|
|
"item_generic", // FIXME: replacement needed
|
2019-02-22 00:56:03 +01:00
|
|
|
"extra_life",
|
|
|
|
(plr->point_item_value * 10) / PLR_MAX_LIFE_FRAGMENTS
|
2017-03-21 11:09:32 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void player_add_bomb_fragments(Player *plr, int frags) {
|
2019-04-03 20:48:25 +02:00
|
|
|
if(global.stage->type == STAGE_SPELL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
player_add_fragments(
|
|
|
|
plr,
|
|
|
|
frags,
|
|
|
|
&plr->bombs,
|
|
|
|
&plr->bomb_fragments,
|
|
|
|
PLR_MAX_BOMB_FRAGMENTS,
|
|
|
|
PLR_MAX_BOMBS,
|
2017-03-21 11:09:32 +01:00
|
|
|
"item_generic", // FIXME: replacement needed
|
2019-02-22 00:56:03 +01:00
|
|
|
"extra_bomb",
|
|
|
|
(plr->point_item_value * 5) / PLR_MAX_BOMB_FRAGMENTS
|
2017-03-21 11:09:32 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void player_add_lives(Player *plr, int lives) {
|
|
|
|
player_add_life_fragments(plr, PLR_MAX_LIFE_FRAGMENTS);
|
|
|
|
}
|
|
|
|
|
|
|
|
void player_add_bombs(Player *plr, int bombs) {
|
|
|
|
player_add_bomb_fragments(plr, PLR_MAX_BOMB_FRAGMENTS);
|
|
|
|
}
|
|
|
|
|
2019-04-11 11:23:24 +02:00
|
|
|
static void scoretext_update(StageText *txt, int t, float a) {
|
2019-11-29 09:18:42 +01:00
|
|
|
double r = bits_to_double((uintptr_t)txt->custom.data1);
|
2019-04-07 00:55:13 +02:00
|
|
|
txt->pos -= I * cexp(I*r) * a;
|
|
|
|
}
|
|
|
|
|
2019-04-14 06:16:10 +02:00
|
|
|
#define SCORETEXT_PIV_BIT ((uintptr_t)1 << ((sizeof(uintptr_t) * 8) - 1))
|
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
static StageText *find_scoretext_combination_candidate(cmplx pos, bool is_piv) {
|
2019-04-14 06:16:10 +02:00
|
|
|
for(StageText *stxt = stagetext_list_head(); stxt; stxt = stxt->next) {
|
|
|
|
if(
|
|
|
|
stxt->custom.update == scoretext_update &&
|
|
|
|
stxt->time.spawn > global.frames &&
|
|
|
|
(bool)((uintptr_t)(stxt->custom.data2) & SCORETEXT_PIV_BIT) == is_piv &&
|
|
|
|
cabs(pos - stxt->pos) < 32
|
|
|
|
) {
|
|
|
|
return stxt;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
static void add_score_text(Player *plr, cmplx location, uint points, bool is_piv) {
|
2019-11-29 09:18:42 +01:00
|
|
|
double rnd = rng_f64s();
|
2019-04-14 06:16:10 +02:00
|
|
|
|
|
|
|
StageText *stxt = find_scoretext_combination_candidate(location, is_piv);
|
|
|
|
|
|
|
|
if(stxt) {
|
|
|
|
points += ((uintptr_t)stxt->custom.data2 & ~SCORETEXT_PIV_BIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
float importance;
|
|
|
|
float a;
|
|
|
|
Color c;
|
|
|
|
|
|
|
|
struct {
|
|
|
|
int delay, lifetime, fadeintime, fadeouttime;
|
|
|
|
} timings;
|
|
|
|
|
|
|
|
timings.delay = 6;
|
|
|
|
timings.fadeintime = 10;
|
|
|
|
timings.fadeouttime = 20;
|
|
|
|
|
|
|
|
if(is_piv) {
|
|
|
|
importance = sqrt(min(points/500.0, 1));
|
|
|
|
a = lerp(0.4, 1.0, importance);
|
|
|
|
c = *color_lerp(RGB(0.5, 0.8, 1.0), RGB(1.0, 0.3, 1.0), importance);
|
|
|
|
timings.lifetime = 35 + 10 * importance;
|
|
|
|
} else {
|
|
|
|
importance = clamp(0.25 * (double)points / (double)plr->point_item_value, 0, 1);
|
|
|
|
a = clamp(0.5 + 0.5 * cbrtf(importance), 0, 1);
|
|
|
|
c = *color_lerp(RGB(1.0, 0.8, 0.4), RGB(0.4, 1.0, 0.3), importance);
|
|
|
|
timings.lifetime = 25 + 20 * importance;
|
|
|
|
}
|
|
|
|
|
|
|
|
a *= config_get_float(CONFIG_SCORETEXT_ALPHA);
|
|
|
|
color_mul_scalar(&c, a);
|
|
|
|
|
|
|
|
if(!stxt) {
|
|
|
|
if(c.a < 1e-4) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
stxt = stagetext_add(
|
|
|
|
NULL, location, ALIGN_CENTER, get_font("small"), &c,
|
|
|
|
timings.delay, timings.lifetime, timings.fadeintime, timings.fadeouttime
|
|
|
|
);
|
|
|
|
|
2019-11-29 09:18:42 +01:00
|
|
|
stxt->custom.data1 = (void*)(uintptr_t)double_to_bits(rnd);
|
2019-04-14 06:16:10 +02:00
|
|
|
stxt->custom.update = scoretext_update;
|
|
|
|
} else {
|
|
|
|
stxt->color = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
format_huge_num(0, points, sizeof(stxt->text), stxt->text);
|
|
|
|
stxt->custom.data2 = (void*)(uintptr_t)(points | (SCORETEXT_PIV_BIT * is_piv));
|
|
|
|
|
|
|
|
if(is_piv) {
|
|
|
|
strlcat(stxt->text, " PIV", sizeof(stxt->text));
|
|
|
|
}
|
2019-04-07 00:55:13 +02:00
|
|
|
}
|
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
void player_add_points(Player *plr, uint points, cmplx location) {
|
2019-02-22 00:56:03 +01:00
|
|
|
plr->points += points;
|
|
|
|
|
|
|
|
while(plr->points >= plr->extralife_threshold) {
|
|
|
|
plr->extralife_threshold = player_next_extralife_threshold(++plr->extralives_given);
|
2019-02-28 11:10:03 +01:00
|
|
|
player_add_lives(plr, 1);
|
2019-02-22 00:56:03 +01:00
|
|
|
}
|
2019-04-07 00:55:13 +02:00
|
|
|
|
2019-04-14 06:16:10 +02:00
|
|
|
add_score_text(plr, location, points, false);
|
2019-02-22 00:56:03 +01:00
|
|
|
}
|
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
void player_add_piv(Player *plr, uint piv, cmplx location) {
|
2019-02-22 00:56:03 +01:00
|
|
|
uint v = plr->point_item_value + piv;
|
2017-03-25 19:18:24 +01:00
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
if(v > PLR_MAX_PIV || v < plr->point_item_value) {
|
|
|
|
plr->point_item_value = PLR_MAX_PIV;
|
|
|
|
} else {
|
|
|
|
plr->point_item_value = v;
|
|
|
|
}
|
2019-04-07 00:55:13 +02:00
|
|
|
|
2019-04-14 06:16:10 +02:00
|
|
|
add_score_text(plr, location, piv, true);
|
2019-02-22 00:56:03 +01:00
|
|
|
}
|
2017-03-25 19:18:24 +01:00
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
void player_add_voltage(Player *plr, uint voltage) {
|
|
|
|
uint v = plr->voltage + voltage;
|
|
|
|
|
|
|
|
if(v > PLR_MAX_VOLTAGE || v < plr->voltage) {
|
|
|
|
plr->voltage = PLR_MAX_VOLTAGE;
|
|
|
|
} else {
|
|
|
|
plr->voltage = v;
|
2017-03-25 19:18:24 +01:00
|
|
|
}
|
2019-02-22 00:56:03 +01:00
|
|
|
|
|
|
|
player_add_bomb_fragments(plr, voltage);
|
2017-03-25 19:18:24 +01:00
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
bool player_drain_voltage(Player *plr, uint voltage) {
|
|
|
|
// TODO: animate (or maybe handle that at stagedraw level)
|
2017-03-25 19:18:24 +01:00
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
if(plr->voltage >= voltage) {
|
|
|
|
plr->voltage -= voltage;
|
|
|
|
return true;
|
2017-03-27 13:08:26 +02:00
|
|
|
}
|
2019-02-22 00:56:03 +01:00
|
|
|
|
|
|
|
plr->voltage = 0;
|
|
|
|
return false;
|
2017-03-25 19:18:24 +01:00
|
|
|
}
|
|
|
|
|
2018-07-30 09:04:09 +02:00
|
|
|
void player_register_damage(Player *plr, EntityInterface *target, const DamageInfo *damage) {
|
2019-02-22 00:56:03 +01:00
|
|
|
if(!DAMAGETYPE_IS_PLAYER(damage->type)) {
|
2018-07-30 09:04:09 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
cmplx pos = NAN;
|
2019-02-22 00:56:03 +01:00
|
|
|
|
2018-07-30 09:04:09 +02:00
|
|
|
if(target != NULL) {
|
|
|
|
switch(target->type) {
|
2020-04-17 09:18:53 +02:00
|
|
|
case ENT_TYPE_ID(Enemy): {
|
2019-02-22 00:56:03 +01:00
|
|
|
pos = ENT_CAST(target, Enemy)->pos;
|
2019-04-07 00:55:13 +02:00
|
|
|
player_add_points(&global.plr, damage->amount * 0.5, pos);
|
2018-07-30 09:04:09 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-04-17 09:18:53 +02:00
|
|
|
case ENT_TYPE_ID(Boss): {
|
2019-02-22 00:56:03 +01:00
|
|
|
pos = ENT_CAST(target, Boss)->pos;
|
2019-04-07 00:55:13 +02:00
|
|
|
player_add_points(&global.plr, damage->amount * 0.2, pos);
|
2018-07-30 09:04:09 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
if(!isnan(creal(pos)) && damage->type == DMG_PLAYER_DISCHARGE) {
|
2020-04-17 09:18:53 +02:00
|
|
|
double rate = target->type == ENT_TYPE_ID(Boss) ? 110 : 256;
|
2019-03-03 12:53:45 +01:00
|
|
|
spawn_and_collect_items(pos, 1, ITEM_VOLTAGE, (int)(damage->amount / rate));
|
2019-02-22 00:56:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if(player_is_powersurge_active(plr)) {
|
|
|
|
plr->powersurge.damage_done += damage->amount;
|
|
|
|
plr->powersurge.damage_accum += damage->amount;
|
|
|
|
|
|
|
|
if(!isnan(creal(pos))) {
|
|
|
|
double rate = 500;
|
|
|
|
|
|
|
|
while(plr->powersurge.damage_accum > rate) {
|
|
|
|
plr->powersurge.damage_accum -= rate;
|
|
|
|
spawn_item(pos, ITEM_SURGE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-07-30 09:04:09 +02:00
|
|
|
|
2018-05-03 19:51:10 +02:00
|
|
|
#ifdef PLR_DPS_STATS
|
|
|
|
while(global.frames > plr->dmglogframe) {
|
|
|
|
memmove(plr->dmglog + 1, plr->dmglog, sizeof(plr->dmglog) - sizeof(*plr->dmglog));
|
|
|
|
plr->dmglog[0] = 0;
|
|
|
|
plr->dmglogframe++;
|
|
|
|
}
|
|
|
|
|
2018-07-30 09:04:09 +02:00
|
|
|
plr->dmglog[0] += damage->amount;
|
2018-05-03 19:51:10 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2019-02-22 00:56:03 +01:00
|
|
|
uint64_t player_next_extralife_threshold(uint64_t step) {
|
|
|
|
static uint64_t base = 5000000;
|
|
|
|
return base * (step * step + step + 2) / 2;
|
|
|
|
}
|
|
|
|
|
2017-03-11 04:41:57 +01:00
|
|
|
void player_preload(void) {
|
|
|
|
const int flags = RESF_DEFAULT;
|
|
|
|
|
2018-05-18 20:15:11 +02:00
|
|
|
preload_resources(RES_SHADER_PROGRAM, flags,
|
2019-04-08 18:57:41 +02:00
|
|
|
"circle_distort",
|
2018-05-18 20:15:11 +02:00
|
|
|
"player_death",
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
preload_resources(RES_TEXTURE, flags,
|
|
|
|
"static",
|
|
|
|
NULL);
|
|
|
|
|
2018-02-06 07:19:25 +01:00
|
|
|
preload_resources(RES_SPRITE, flags,
|
2017-03-11 04:41:57 +01:00
|
|
|
"fairy_circle",
|
2019-02-22 00:56:03 +01:00
|
|
|
"focus",
|
|
|
|
"part/blast_huge_halo",
|
|
|
|
"part/powersurge_field",
|
2017-03-11 04:41:57 +01:00
|
|
|
NULL);
|
|
|
|
|
|
|
|
preload_resources(RES_SFX, flags | RESF_OPTIONAL,
|
|
|
|
"death",
|
2017-03-21 11:09:32 +01:00
|
|
|
"extra_bomb",
|
2019-02-22 00:56:03 +01:00
|
|
|
"extra_life",
|
|
|
|
"generic_shot",
|
|
|
|
"graze",
|
2018-08-14 02:56:16 +02:00
|
|
|
"hit0",
|
|
|
|
"hit1",
|
2019-02-22 00:56:03 +01:00
|
|
|
"powerup",
|
2019-04-25 01:43:50 +02:00
|
|
|
"powersurge_start",
|
|
|
|
"powersurge_end",
|
2017-03-11 04:41:57 +01:00
|
|
|
NULL);
|
|
|
|
}
|
2018-08-11 21:13:48 +02:00
|
|
|
|
|
|
|
// FIXME: where should this be?
|
|
|
|
|
2019-11-22 04:37:11 +01:00
|
|
|
cmplx plrutil_homing_target(cmplx org, cmplx fallback) {
|
2018-08-11 21:13:48 +02:00
|
|
|
double mindst = INFINITY;
|
2019-11-22 04:37:11 +01:00
|
|
|
cmplx target = fallback;
|
2018-08-11 21:13:48 +02:00
|
|
|
|
|
|
|
if(global.boss && boss_is_vulnerable(global.boss)) {
|
|
|
|
target = global.boss->pos;
|
|
|
|
mindst = cabs(target - org);
|
|
|
|
}
|
|
|
|
|
|
|
|
for(Enemy *e = global.enemies.first; e; e = e->next) {
|
|
|
|
if(e->hp == ENEMY_IMMUNE) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
double dst = cabs(e->pos - org);
|
|
|
|
|
|
|
|
if(dst < mindst) {
|
|
|
|
mindst = dst;
|
|
|
|
target = e->pos;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return target;
|
|
|
|
}
|
2020-04-06 03:29:11 +02:00
|
|
|
|
|
|
|
void plrutil_slave_retract(BoxedPlayer bplr, cmplx *pos, real retract_time) {
|
|
|
|
cmplx pos0 = *pos;
|
|
|
|
Player *plr;
|
|
|
|
|
|
|
|
for(int i = 1; i <= retract_time; ++i) {
|
|
|
|
YIELD;
|
|
|
|
plr = NOT_NULL(ENT_UNBOX(bplr));
|
|
|
|
*pos = clerp(pos0, plr->pos, i / retract_time);
|
|
|
|
}
|
|
|
|
}
|