2022-02-26 23:55:18 +00:00
|
|
|
#pragma option -G
|
|
|
|
|
|
|
|
#include "platform.h"
|
|
|
|
#include "pc98.h"
|
|
|
|
#include "master.hpp"
|
|
|
|
#include "th01/math/subpixel.hpp"
|
|
|
|
extern "C" {
|
|
|
|
#include "th04/math/vector.hpp"
|
|
|
|
#include "th04/snd/snd.h"
|
|
|
|
#include "th04/main/playfld.hpp"
|
2023-01-13 18:57:05 +00:00
|
|
|
#include "th04/main/bullet/clearzap.hpp"
|
2022-02-26 23:55:18 +00:00
|
|
|
#include "th05/main/playperf.hpp"
|
|
|
|
}
|
|
|
|
#include "th05/main/bullet/laser.hpp"
|
|
|
|
|
2023-01-13 18:57:05 +00:00
|
|
|
// Segment 1 (as allocated in the header)
|
|
|
|
// ---------
|
|
|
|
|
|
|
|
// ZUN bloat: Needed to circumvent 16-bit promotion in comparisons.
|
|
|
|
inline int8_t laser_width_min(void) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void near lasers_update(void)
|
|
|
|
{
|
|
|
|
#define shootout_update(laser) { \
|
|
|
|
if(laser->coords.starts_at_distance < to_sp(LASER_DISTANCE_MAX)) { \
|
|
|
|
laser->coords.ends_at_distance.v += laser->shootout_speed.v; \
|
|
|
|
} \
|
|
|
|
if(laser->age >= laser->active_at_age.moveout) { \
|
|
|
|
laser->coords.starts_at_distance.v += laser->shootout_speed.v; \
|
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
|
|
|
Laser near *laser;
|
|
|
|
int i;
|
|
|
|
for((laser = lasers, i = 0); i < LASER_COUNT; (i++, laser++)) {
|
|
|
|
if(laser->flag == LF_FREE) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(bullet_clear_time || bullet_zap.active) {
|
|
|
|
if(laser->flag == LF_SHOOTOUT) {
|
|
|
|
laser->flag = LF_SHOOTOUT_DECAY;
|
|
|
|
laser->shootout_speed.v /= 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(laser->flag) {
|
|
|
|
case LF_SHOOTOUT:
|
|
|
|
shootout_update(laser);
|
|
|
|
laser_hittest(*laser);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LF_FIXED_WAIT_TO_GROW:
|
|
|
|
if(laser->active_at_age.grow > 0) {
|
|
|
|
if(laser->age >= laser->active_at_age.grow) {
|
|
|
|
// laser->flag = LF_FIXED_GROW;
|
|
|
|
static_cast<uint8_t>(laser->flag)++;
|
|
|
|
snd_se_play(6);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LF_FIXED_GROW:
|
|
|
|
if(laser->age & 1) {
|
|
|
|
laser->coords.width.nonshrink += 2;
|
|
|
|
}
|
|
|
|
if(laser->coords.width.nonshrink >= laser->grow_to_width) {
|
|
|
|
// laser->flag = LF_FIXED_ACTIVE;
|
|
|
|
static_cast<uint8_t>(laser->flag)++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LF_FIXED_ACTIVE:
|
|
|
|
laser_hittest(*laser);
|
|
|
|
if(laser->shrink_at_age > 0) {
|
|
|
|
if(laser->age >= laser->shrink_at_age) {
|
|
|
|
// laser->flag = LF_FIXED_SHRINK;
|
|
|
|
static_cast<uint8_t>(laser->flag)++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LF_FIXED_SHRINK:
|
|
|
|
if(laser->age & 1) {
|
|
|
|
// MODDERS: The necessary clamp to 0 here is caught by the
|
|
|
|
// signed comparison below.
|
|
|
|
laser->coords.width.shrink -= 2;
|
|
|
|
}
|
|
|
|
if(laser->coords.width.shrink < laser_width_min()) {
|
|
|
|
laser->flag = LF_FREE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LF_FIXED_SHRINK_AND_WAIT_TO_GROW:
|
|
|
|
if(laser->age & 1) {
|
|
|
|
// MODDERS: See above… except that the signed comparison
|
|
|
|
// wouldn't even have been needed here, as the condition is
|
|
|
|
// `true` for both 1 and 0.
|
|
|
|
laser->coords.width.shrink -= 2;
|
|
|
|
}
|
|
|
|
if(laser->coords.width.shrink <= laser_width_min()) {
|
|
|
|
laser->flag = LF_FIXED_WAIT_TO_GROW;
|
|
|
|
laser->age = 0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LF_SHOOTOUT_DECAY:
|
|
|
|
shootout_update(laser);
|
|
|
|
laser->coords.width.nonshrink += 2;
|
|
|
|
if(
|
|
|
|
laser->coords.width.nonshrink >=
|
|
|
|
LASER_SHOOTOUT_DECAY_WIDTH_MAX
|
|
|
|
) {
|
|
|
|
laser->flag = LF_FREE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
laser->age++;
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef shootout_update
|
|
|
|
}
|
|
|
|
// ---------
|
|
|
|
|
2023-01-10 09:14:42 +00:00
|
|
|
// Segment 3 (as allocated in the header)
|
|
|
|
// ---------
|
|
|
|
|
2022-02-26 23:55:18 +00:00
|
|
|
void near lasers_shootout_add(void)
|
|
|
|
{
|
2022-07-22 17:01:30 +00:00
|
|
|
Laser near *laser;
|
2022-02-26 23:55:18 +00:00
|
|
|
int i;
|
|
|
|
subpixel_t speed = playperf_speedtune(laser_template.shootout_speed.v);
|
|
|
|
|
|
|
|
for((laser = lasers, i = 0); i < LASER_COUNT; (i++, laser++)) {
|
|
|
|
if(laser->flag != LF_FREE) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
laser->flag = LF_SHOOTOUT;
|
|
|
|
laser->coords.starts_at_distance.set(LASER_DISTANCE_MIN);
|
|
|
|
laser->coords.ends_at_distance.set(LASER_DISTANCE_MIN);
|
|
|
|
laser->age = 0;
|
|
|
|
laser->active_at_age = laser_template.active_at_age;
|
|
|
|
laser->col = laser_template.col;
|
|
|
|
laser->shootout_speed.v = speed;
|
|
|
|
laser->grow_to_width = 6;
|
2023-01-13 18:57:05 +00:00
|
|
|
laser->coords.width.nonshrink = 6;
|
2022-02-26 23:55:18 +00:00
|
|
|
laser->coords.angle = laser_template.coords.angle;
|
|
|
|
vector2_at(
|
|
|
|
laser->coords.origin,
|
|
|
|
laser_template.coords.origin.x.v,
|
|
|
|
laser_template.coords.origin.y.v,
|
|
|
|
to_sp(LASER_DISTANCE_MIN / 2.0f),
|
|
|
|
laser_template.coords.angle
|
|
|
|
);
|
|
|
|
snd_se_play(5);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void pascal near laser_fixed_spawn(int slot)
|
|
|
|
{
|
2022-07-22 17:01:30 +00:00
|
|
|
Laser near &laser = lasers[slot];
|
2022-02-26 23:55:18 +00:00
|
|
|
if(laser.flag != LF_FREE) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
laser.fixed_init(laser_template.coords.origin);
|
|
|
|
laser.active_at_age.grow = laser_template.active_at_age.grow;
|
|
|
|
laser.shrink_at_age = laser_template.shrink_at_age;
|
|
|
|
laser.col = laser_template.col;
|
2023-01-13 18:57:05 +00:00
|
|
|
laser.grow_to_width = laser_template.coords.width.nonshrink;
|
|
|
|
laser.coords.width.nonshrink = 1;
|
2022-02-26 23:55:18 +00:00
|
|
|
laser.coords.angle = laser_template.coords.angle;
|
|
|
|
snd_se_play(5);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pascal near laser_manual_fixed_spawn(int slot)
|
|
|
|
{
|
2022-07-22 17:01:30 +00:00
|
|
|
Laser near &laser = lasers[slot];
|
2022-02-26 23:55:18 +00:00
|
|
|
if(laser.flag != LF_FREE) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
laser.fixed_init(laser_template.coords.origin);
|
|
|
|
laser.active_at_age.grow = -1;
|
|
|
|
laser.shrink_at_age = -1;
|
|
|
|
laser.col = laser_template.col;
|
2023-01-13 18:57:05 +00:00
|
|
|
laser.coords.width.nonshrink = 1;
|
|
|
|
laser.grow_to_width = laser_template.coords.width.nonshrink;
|
2022-02-26 23:55:18 +00:00
|
|
|
laser.coords.angle = laser_template.coords.angle;
|
|
|
|
snd_se_play(5);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pascal near laser_manual_grow(int slot)
|
|
|
|
{
|
|
|
|
if(lasers[slot].flag != LF_FIXED_WAIT_TO_GROW) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
lasers[slot].flag = LF_FIXED_GROW;
|
|
|
|
snd_se_play(6);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pascal near laser_stop(int slot)
|
|
|
|
{
|
|
|
|
if(lasers[slot].flag == LF_FIXED_ACTIVE) {
|
|
|
|
lasers[slot].flag = LF_FIXED_SHRINK;
|
|
|
|
} else if (lasers[slot].flag == LF_SHOOTOUT) {
|
2023-01-13 18:57:05 +00:00
|
|
|
lasers[slot].flag = LF_SHOOTOUT_DECAY;
|
2022-02-26 23:55:18 +00:00
|
|
|
} else if (
|
|
|
|
(lasers[slot].flag != LF_FIXED_SHRINK) &&
|
2023-01-13 18:57:05 +00:00
|
|
|
(lasers[slot].flag != LF_SHOOTOUT_DECAY)
|
2022-02-26 23:55:18 +00:00
|
|
|
) {
|
|
|
|
lasers[slot].flag = LF_FREE;
|
|
|
|
}
|
|
|
|
}
|
2023-01-10 09:14:42 +00:00
|
|
|
// ---------
|