taisei/src/enemy.c

243 lines
5.1 KiB
C
Raw Normal View History

/*
* This software is licensed under the terms of the MIT-License
* See COPYING for further information.
* ---
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>.
*/
#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,
complex a1, complex a2, complex a3, complex a4) {
Enemy *e = (Enemy *)create_element((void **)enemies, sizeof(Enemy));
e->moving = false;
e->dir = 0;
e->birthtime = global.frames;
e->pos = pos;
e->pos0 = pos;
e->hp = hp;
2012-07-18 12:33:37 +02:00
e->alpha = 1.0;
e->unbombable = false;
e->logic_rule = logic_rule;
e->draw_rule = draw_rule;
e->args[0] = a1;
e->args[1] = a2;
e->args[2] = a3;
e->args[3] = a4;
e->logic_rule(e, EVENT_BIRTH);
2012-07-28 19:45:51 +02:00
return e;
}
void _delete_enemy(void **enemies, void* enemy) {
Enemy *e = (Enemy *)enemy;
2012-08-07 11:24:14 +02:00
if(e->hp <= 0 && e->hp > ENEMY_IMMUNE) {
int i;
2017-01-28 19:56:17 +01:00
play_sound("enemydeath");
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));
}
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);
}
2012-08-07 11:24:14 +02:00
e->logic_rule(enemy, EVENT_DEATH);
del_ref(enemy);
delete_element((void **)enemies, enemy);
}
void delete_enemy(Enemy **enemies, Enemy* enemy) {
_delete_enemy((void**) enemies, enemy);
}
void delete_enemies(Enemy **enemies) {
delete_all_elements((void **)enemies, _delete_enemy);
}
void draw_enemies(Enemy *enemies) {
2012-07-18 12:33:37 +02:00
Enemy *e;
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;
2012-07-18 12:33:37 +02:00
glColor4f(1,1,1,e->alpha);
reset = true;
2012-07-18 12:33:37 +02:00
}
e->draw_rule(e, global.frames - e->birthtime);
2012-07-18 12:33:37 +02:00
if(reset)
glColor4f(1,1,1,1);
}
}
}
2012-07-19 19:53:33 +02:00
void killall(Enemy *enemies) {
Enemy *e;
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;
}
2012-04-04 17:19:53 +02:00
p->pos += p->args[1];
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;
2012-04-04 17:19:53 +02:00
glPushMatrix();
float s = 2.0-t/p->args[0]*2;
if(e->pos + p->pos)
glTranslatef(creal(e->pos + p->pos), cimag(e->pos + p->pos), 0);
if(p->angle != M_PI*0.5)
glRotatef(p->angle*180/M_PI+90, 0, 0, 1);
2012-07-13 18:44:56 +02:00
if(s != 1)
glScalef(s, s, 1);
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);
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);
2012-04-04 17:19:53 +02:00
glPopMatrix();
if(p->clr)
glColor3f(1,1,1);
2012-04-04 17:19:53 +02:00
}
void BigFairy(Enemy *e, int t) {
if(!(t % 5)) {
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
}
2012-04-04 17:19:53 +02:00
glPushMatrix();
glTranslatef(creal(e->pos), cimag(e->pos), 0);
2012-04-04 17:19:53 +02:00
float s = sin((float)(global.frames-e->birthtime)/10.f)/6 + 0.8;
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();
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();
if(e->dir)
glCullFace(GL_BACK);
2012-04-04 17:19:53 +02:00
}
void Fairy(Enemy *e, int t) {
float s = sin((float)(global.frames-e->birthtime)/10.f)/6 + 0.8;
glPushMatrix();
glTranslatef(creal(e->pos),cimag(e->pos),0);
glPushMatrix();
glRotatef(global.frames*10,0,0,1);
glScalef(s, s, s);
draw_texture(0,0,"fairy_circle");
glPopMatrix();
glPushMatrix();
if(e->dir) {
glCullFace(GL_FRONT);
glScalef(-1,1,1);
}
draw_animation(0, 0, e->moving, "fairy");
glPopMatrix();
glPopMatrix();
if(e->dir) {
glCullFace(GL_BACK);
}
}
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();
}
void process_enemies(Enemy **enemies) {
Enemy *enemy = *enemies, *del = NULL;
while(enemy != NULL) {
2011-07-05 15:20:19 +02:00
int action = enemy->logic_rule(enemy, global.frames - enemy->birthtime);
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);
2011-07-05 15:20:19 +02:00
if((enemy->hp > ENEMY_IMMUNE
&& (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) {
del = enemy;
enemy = enemy->next;
delete_enemy(enemies, del);
} else {
enemy = enemy->next;
}
}
2012-07-18 12:33:37 +02: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);
}