2011-04-26 12:04:45 +02:00
|
|
|
/*
|
|
|
|
* This software is licensed under the terms of the MIT-License
|
2017-02-11 04:52:08 +01:00
|
|
|
* See COPYING for further information.
|
2011-04-26 12:04:45 +02:00
|
|
|
* ---
|
2017-09-12 03:28:15 +02:00
|
|
|
* Copyright (c) 2011-2017, Lukas Weber <laochailan@web.de>.
|
|
|
|
* Copyright (c) 2012-2017, Andrei Alexeyev <akari@alienslab.net>.
|
2011-04-26 12:04:45 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "enemy.h"
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "global.h"
|
|
|
|
#include "projectile.h"
|
|
|
|
#include "list.h"
|
|
|
|
|
2012-07-28 19:45:51 +02:00
|
|
|
Enemy *create_enemy_p(Enemy **enemies, complex pos, int hp, EnemyDrawRule draw_rule, EnemyLogicRule logic_rule,
|
2011-06-26 13:45:27 +02:00
|
|
|
complex a1, complex a2, complex a3, complex a4) {
|
2011-04-26 12:04:45 +02:00
|
|
|
Enemy *e = (Enemy *)create_element((void **)enemies, sizeof(Enemy));
|
2017-02-11 04:52:08 +01:00
|
|
|
e->moving = false;
|
2011-06-26 13:45:27 +02:00
|
|
|
e->dir = 0;
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-04-26 12:04:45 +02:00
|
|
|
e->birthtime = global.frames;
|
|
|
|
e->pos = pos;
|
|
|
|
e->pos0 = pos;
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-04-26 12:04:45 +02:00
|
|
|
e->hp = hp;
|
2012-07-18 12:33:37 +02:00
|
|
|
e->alpha = 1.0;
|
2017-02-11 04:52:08 +01:00
|
|
|
e->unbombable = false;
|
|
|
|
|
2011-04-26 12:04:45 +02:00
|
|
|
e->logic_rule = logic_rule;
|
|
|
|
e->draw_rule = draw_rule;
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-06-26 13:45:27 +02:00
|
|
|
e->args[0] = a1;
|
|
|
|
e->args[1] = a2;
|
|
|
|
e->args[2] = a3;
|
|
|
|
e->args[3] = a4;
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-04-26 12:04:45 +02:00
|
|
|
e->logic_rule(e, EVENT_BIRTH);
|
2012-07-28 19:45:51 +02:00
|
|
|
return e;
|
2011-04-26 12:04:45 +02:00
|
|
|
}
|
|
|
|
|
2011-04-26 22:39:50 +02:00
|
|
|
void _delete_enemy(void **enemies, void* enemy) {
|
2011-06-28 19:23:02 +02:00
|
|
|
Enemy *e = (Enemy *)enemy;
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-08-07 11:24:14 +02:00
|
|
|
if(e->hp <= 0 && e->hp > ENEMY_IMMUNE) {
|
2011-06-28 19:23:02 +02:00
|
|
|
int i;
|
2017-01-28 19:56:17 +01:00
|
|
|
play_sound("enemydeath");
|
2012-08-07 02:45:38 +02:00
|
|
|
for(i = 0; i < 10; i++) {
|
|
|
|
tsrand_fill(2);
|
2017-02-26 21:59:51 +01:00
|
|
|
create_particle2c("flare", e->pos, 0, Fade, timeout_linear, 10, (3+afrand(0)*10)*cexp(I*afrand(1)*2*M_PI));
|
2012-08-07 02:45:38 +02:00
|
|
|
}
|
2017-02-26 21:59:51 +01:00
|
|
|
create_particle1c("blast", e->pos, 0, Blast, timeout, 20);
|
|
|
|
create_particle1c("blast", e->pos, 0, Blast, timeout, 20);
|
|
|
|
create_particle2c("blast", e->pos, 0, GrowFade, timeout, 15,0);
|
2011-06-28 19:23:02 +02:00
|
|
|
}
|
2012-08-07 11:24:14 +02:00
|
|
|
e->logic_rule(enemy, EVENT_DEATH);
|
2011-05-13 19:03:02 +02:00
|
|
|
del_ref(enemy);
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-04-26 12:04:45 +02:00
|
|
|
delete_element((void **)enemies, enemy);
|
|
|
|
}
|
|
|
|
|
2011-04-26 22:39:50 +02:00
|
|
|
void delete_enemy(Enemy **enemies, Enemy* enemy) {
|
|
|
|
_delete_enemy((void**) enemies, enemy);
|
|
|
|
}
|
|
|
|
|
|
|
|
void delete_enemies(Enemy **enemies) {
|
|
|
|
delete_all_elements((void **)enemies, _delete_enemy);
|
2011-04-26 12:04:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void draw_enemies(Enemy *enemies) {
|
2012-07-18 12:33:37 +02:00
|
|
|
Enemy *e;
|
2017-02-11 04:52:08 +01:00
|
|
|
bool reset = false;
|
|
|
|
|
2012-07-18 12:33:37 +02:00
|
|
|
for(e = enemies; e; e = e->next) {
|
|
|
|
if(e->draw_rule) {
|
|
|
|
if(e->alpha < 1) {
|
|
|
|
e->alpha += 1 / 60.0;
|
|
|
|
if(e->alpha > 1)
|
|
|
|
e->alpha = 1;
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-07-18 12:33:37 +02:00
|
|
|
glColor4f(1,1,1,e->alpha);
|
2017-02-11 04:52:08 +01:00
|
|
|
reset = true;
|
2012-07-18 12:33:37 +02:00
|
|
|
}
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-11-04 18:54:53 +01:00
|
|
|
e->draw_rule(e, global.frames - e->birthtime);
|
2012-07-18 12:33:37 +02:00
|
|
|
if(reset)
|
|
|
|
glColor4f(1,1,1,1);
|
|
|
|
}
|
|
|
|
}
|
2011-04-26 12:04:45 +02:00
|
|
|
}
|
|
|
|
|
2012-07-19 19:53:33 +02:00
|
|
|
void killall(Enemy *enemies) {
|
2017-02-11 04:52:08 +01:00
|
|
|
Enemy *e;
|
|
|
|
|
2012-07-19 19:48:17 +02:00
|
|
|
for(e = enemies; e; e = e->next)
|
|
|
|
e->hp = 0;
|
|
|
|
}
|
2012-04-04 17:19:53 +02:00
|
|
|
|
|
|
|
int enemy_flare(Projectile *p, int t) { // a[0] timeout, a[1] velocity, a[2] ref to enemy
|
|
|
|
if(t >= creal(p->args[0]) || REF(p->args[2]) == NULL) {
|
|
|
|
return ACTION_DESTROY;
|
|
|
|
} if(t == EVENT_DEATH) {
|
|
|
|
free_ref(p->args[2]);
|
|
|
|
return 1;
|
|
|
|
} else if(t < 0) {
|
|
|
|
return 1;
|
|
|
|
}
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-04 17:19:53 +02:00
|
|
|
p->pos += p->args[1];
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-04 17:19:53 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EnemyFlareShrink(Projectile *p, int t) {
|
|
|
|
Enemy *e = (Enemy *)REF(p->args[2]);
|
|
|
|
if(e == NULL)
|
|
|
|
return;
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-04 17:19:53 +02:00
|
|
|
glPushMatrix();
|
|
|
|
float s = 2.0-t/p->args[0]*2;
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-05 14:24:55 +02:00
|
|
|
if(e->pos + p->pos)
|
|
|
|
glTranslatef(creal(e->pos + p->pos), cimag(e->pos + p->pos), 0);
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-05 14:24:55 +02:00
|
|
|
if(p->angle != M_PI*0.5)
|
|
|
|
glRotatef(p->angle*180/M_PI+90, 0, 0, 1);
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-07-13 18:44:56 +02:00
|
|
|
if(s != 1)
|
2012-04-05 14:24:55 +02:00
|
|
|
glScalef(s, s, 1);
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-04 17:19:53 +02:00
|
|
|
if(p->clr)
|
2017-02-26 21:59:51 +01:00
|
|
|
parse_color_call(p->clr, glColor4f);
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-04 17:19:53 +02:00
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE);
|
|
|
|
draw_texture_p(0, 0, p->tex);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-04 17:19:53 +02:00
|
|
|
glPopMatrix();
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-05 14:24:55 +02:00
|
|
|
if(p->clr)
|
|
|
|
glColor3f(1,1,1);
|
2012-04-04 17:19:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void BigFairy(Enemy *e, int t) {
|
|
|
|
if(!(t % 5)) {
|
2017-02-23 11:39:31 +01:00
|
|
|
complex offset = (frand()-0.5)*30 + (frand()-0.5)*20.0*I;
|
|
|
|
create_particle3c("lasercurve", offset, rgb(0,0.2,0.3), EnemyFlareShrink, enemy_flare, 50, (-50.0*I-offset)/50.0, add_ref(e));
|
2012-04-04 17:19:53 +02:00
|
|
|
}
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-04 17:19:53 +02:00
|
|
|
glPushMatrix();
|
|
|
|
glTranslatef(creal(e->pos), cimag(e->pos), 0);
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-04 17:19:53 +02:00
|
|
|
float s = sin((float)(global.frames-e->birthtime)/10.f)/6 + 0.8;
|
2017-02-11 04:52:08 +01:00
|
|
|
|
|
|
|
glPushMatrix();
|
2012-04-04 17:19:53 +02:00
|
|
|
glRotatef(global.frames*10,0,0,1);
|
|
|
|
glScalef(s, s, s);
|
|
|
|
draw_texture(0,0,"fairy_circle");
|
|
|
|
glPopMatrix();
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-04 17:19:53 +02:00
|
|
|
if(e->dir) {
|
|
|
|
glCullFace(GL_FRONT);
|
|
|
|
glScalef(-1,1,1);
|
|
|
|
}
|
|
|
|
draw_animation(0, 0, e->moving, "bigfairy");
|
|
|
|
glPopMatrix();
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-05 14:24:55 +02:00
|
|
|
if(e->dir)
|
|
|
|
glCullFace(GL_BACK);
|
2012-04-04 17:19:53 +02:00
|
|
|
}
|
|
|
|
|
2012-04-05 14:24:55 +02:00
|
|
|
void Fairy(Enemy *e, int t) {
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-04-26 12:04:45 +02:00
|
|
|
float s = sin((float)(global.frames-e->birthtime)/10.f)/6 + 0.8;
|
|
|
|
glPushMatrix();
|
|
|
|
glTranslatef(creal(e->pos),cimag(e->pos),0);
|
2017-02-11 04:52:08 +01:00
|
|
|
|
|
|
|
glPushMatrix();
|
2011-04-26 12:04:45 +02:00
|
|
|
glRotatef(global.frames*10,0,0,1);
|
|
|
|
glScalef(s, s, s);
|
|
|
|
draw_texture(0,0,"fairy_circle");
|
|
|
|
glPopMatrix();
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-04-26 12:04:45 +02:00
|
|
|
glPushMatrix();
|
|
|
|
if(e->dir) {
|
|
|
|
glCullFace(GL_FRONT);
|
|
|
|
glScalef(-1,1,1);
|
|
|
|
}
|
|
|
|
draw_animation(0, 0, e->moving, "fairy");
|
|
|
|
glPopMatrix();
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-04-05 14:24:55 +02:00
|
|
|
glPopMatrix();
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-06-26 13:45:27 +02:00
|
|
|
if(e->dir) {
|
|
|
|
glCullFace(GL_BACK);
|
|
|
|
}
|
2011-04-26 12:04:45 +02:00
|
|
|
}
|
|
|
|
|
2011-06-27 13:36:35 +02:00
|
|
|
void Swirl(Enemy *e, int t) {
|
|
|
|
glPushMatrix();
|
|
|
|
glTranslatef(creal(e->pos), cimag(e->pos),0);
|
|
|
|
glRotatef(t*15,0,0,1);
|
|
|
|
draw_texture(0,0, "swirl");
|
|
|
|
glPopMatrix();
|
|
|
|
}
|
|
|
|
|
2011-04-26 12:04:45 +02:00
|
|
|
void process_enemies(Enemy **enemies) {
|
|
|
|
Enemy *enemy = *enemies, *del = NULL;
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-04-26 12:04:45 +02:00
|
|
|
while(enemy != NULL) {
|
2011-07-05 15:20:19 +02:00
|
|
|
int action = enemy->logic_rule(enemy, global.frames - enemy->birthtime);
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2012-07-19 10:48:31 +02:00
|
|
|
if(enemy->hp > ENEMY_IMMUNE && cabs(enemy->pos - global.plr.pos) < 7)
|
2012-07-20 16:11:24 +02:00
|
|
|
player_death(&global.plr);
|
2017-02-11 04:52:08 +01:00
|
|
|
|
2011-07-05 15:20:19 +02:00
|
|
|
if((enemy->hp > ENEMY_IMMUNE
|
2011-04-26 12:04:45 +02:00
|
|
|
&& (creal(enemy->pos) < -20 || creal(enemy->pos) > VIEWPORT_W + 20
|
|
|
|
|| cimag(enemy->pos) < -20 || cimag(enemy->pos) > VIEWPORT_H + 20
|
2011-07-05 15:20:19 +02:00
|
|
|
|| enemy->hp <= 0)) || action == ACTION_DESTROY) {
|
2011-04-26 12:04:45 +02:00
|
|
|
del = enemy;
|
|
|
|
enemy = enemy->next;
|
|
|
|
delete_enemy(enemies, del);
|
|
|
|
} else {
|
|
|
|
enemy = enemy->next;
|
2017-02-11 04:52:08 +01:00
|
|
|
}
|
2011-04-26 12:04:45 +02:00
|
|
|
}
|
2012-07-18 12:33:37 +02:00
|
|
|
}
|
2017-03-11 04:41:57 +01:00
|
|
|
|
|
|
|
void enemies_preload(void) {
|
|
|
|
preload_resources(RES_ANIM, RESF_DEFAULT,
|
|
|
|
"fairy",
|
|
|
|
"bigfairy",
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
preload_resources(RES_TEXTURE, RESF_DEFAULT,
|
|
|
|
"swirl",
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
preload_resources(RES_SFX, RESF_OPTIONAL,
|
|
|
|
"enemydeath",
|
|
|
|
NULL);
|
|
|
|
}
|