server/src/common/kernel/ugroup.c
Enno Rehling cfdbc32470 * möglichen Exploit beseitigt:
GIB xyz EINHEIT
  GIB 0 ALLES SILBER
  --> ALLE Befehle der übergebenen Einheit werden gelöscht.

* neue Funktion (convenience): ucansee(f, u1, u2)
  liefert u1, wenn cansee(f,u1), sonst u2

* neue mistakes eingefügt uns übersetzt

* message bugfix:
  u->htisorder kann gelöscht werden (z.b. NACH). Daher muss ein pointer auf
  einen befehl in einer message immer auf den u->order Eintrag zeigen, damit er
  zeit der auswertung noch existiert.
  findorder(u, u->thisorder) tut das. Ist an mehreren Stellen nicht benutzt
  worden. assert eingebaut, das das prüft.

* RESERVE_DONATIONS
  Gegenstände, die von einer anderen Partei übergeben wurden, werden nicht
  reserviert.

* TWOPASS_GIVE:
  GIB Befehle werden zuerst an fremde Einheiten, danach in einem zweiten
  Durchlauf an eigene Einheiten, ausgeführt.

* msg_message
  An einigen messages ausprobiert, ob man die gleiche Message mehreren
  Parteien einhängen kann - klappt, spart Speicher.
  Allerdings fehlt dazu ein ordentliches memory-management (refcounter)
2001-05-10 05:50:52 +00:00

200 lines
4 KiB
C

/* vi: set ts=2:
+-------------------+ Christian Schlittchen <corwin@amber.kn-bremen.de>
| | Enno Rehling <enno@eressea-pbem.de>
| Eressea PBEM host | Katja Zedel <katze@felidae.kn-bremen.de>
| (c) 1998 - 2001 | Henning Peters <faroul@beyond.kn-bremen.de>
| | Ingo Wilken <Ingo.Wilken@informatik.uni-oldenburg.de>
+-------------------+ Stefan Reich <reich@halbling.de>
This program may not be used, modified or distributed
without prior permission by the authors of Eressea.
*/
#include <config.h>
#include "eressea.h"
/* kernel includes */
#include "unit.h"
#include "region.h"
#include "faction.h"
#include "ugroup.h"
/* attributes includes */
#include <attributes/ugroup.h>
/* libc includes */
#include <stdlib.h>
/* TODO:
* - Anzeige (CR)
* - Speicherung
* - Evt. NACH (Betrete/Verlasse?) des Leaders vererben auf Members.
* - Routine zur automatisch Aufsplittung.
*/
/* Nur die erste Einheit in der Liste
(ugroup->unit_array[0] == u) kann NACH ausführen. */
#ifdef USE_UGROUPS
boolean
is_ugroupleader(const unit *u, const ugroup *ug)
{
if(ug->unit_array[0] == u) return true;
return false;
}
ugroup *
findugroupid(const faction *f, int id)
{
ugroup *ug;
for(ug=f->ugroups;ug; ug=ug->next) {
if(ug->id == id) return ug;
}
return NULL;
}
ugroup *
findugroup(const unit *u)
{
attrib *a = a_find(u->attribs, &at_ugroup);
if(!a) return NULL;
return findugroupid(u->faction, a->data.i);
}
static int
ugroupfreeid(const ugroup *ug)
{
const ugroup *ug2 = ug;
int id = 0;
while(ug2) {
if(ug2->id == id) {
id++;
ug2 = ug;
} else {
ug2 = ug2->next;
}
}
return id;
}
ugroup *
createugroup(unit *uleader, unit *umember)
{
ugroup *ug = calloc(1,sizeof(ugroup));
ug->id = ugroupfreeid(uleader->faction->ugroups);
ug->members = 2;
ug->unit_array = malloc(2 * sizeof(unit *));
ug->unit_array[0] = uleader;
a_add(&uleader->attribs, a_new(&at_ugroup))->data.i = ug->id;
ug->unit_array[1] = umember;
a_add(&umember->attribs, a_new(&at_ugroup))->data.i = ug->id;
ug->next = uleader->faction->ugroups;
uleader->faction->ugroups = ug;
return ug;
}
void
join_ugroup(unit *u, strlist *order)
{
unit *u2;
ugroup *ug;
u2 = getunit(u->region, u->faction);
if(!u2 || cansee(u->faction, u->region, u2, 0) == false) {
cmistake(u, order->s, 63, MSG_EVENT);
return;
}
if(u2 == u) {
cmistake(u, order->s, 292, MSG_EVENT);
return;
}
if(u2->faction != u->faction) {
cmistake(u, order->s, 293, MSG_EVENT);
return;
}
if(u2->building != u->building || u2->ship != u->ship) {
cmistake(u, order->s, 294, MSG_EVENT);
return;
}
if(a_find(u->attribs, &at_ugroup)) {
cmistake(u, order->s, 290, MSG_EVENT);
return;
}
ug = findugroup(u2);
if(ug) {
ug->members++;
ug->unit_array = realloc(ug->unit_array,
ug->members * sizeof(unit *));
ug->unit_array[ug->members-1] = u;
a_add(&u->attribs, a_new(&at_ugroup))->data.i = ug->id;
} else {
createugroup(u2, u);
}
}
void
leave_ugroup(unit *u, strlist *order)
{
attrib *a = a_find(u->attribs, &at_ugroup);
ugroup *ug;
boolean found = false;
int i;
if(!a) {
cmistake(u, order->s, 286, MSG_EVENT);
return;
}
ug = findugroupid(u->faction, a->data.i);
for(i=0; i<ug->members; i++) {
if(found == true) {
ug->unit_array[i-1] = ug->unit_array[i];
} else if(ug->unit_array[i] == u) {
found = true;
}
}
ug->members--;
a_remove(&u->attribs, a);
if(ug->members == 1) {
a_remove(&ug->unit_array[0]->attribs, a);
free(ug->unit_array);
removelist(&u->faction->ugroups, ug);
} else {
ug->unit_array = realloc(ug->unit_array,
ug->members * sizeof(unit *));
}
}
void
ugroups(void)
{
region *r;
unit *u;
strlist *o;
for(r=regions; r; r=r->next) {
for(u=r->units; u; u=u->next) {
for(o = u->orders; o; o=o->next) {
if(igetkeyword(o->s, u->faction->locale) == K_LEAVEUGROUP) {
leave_ugroup(u, o);
}
}
}
for(u=r->units; u; u=u->next) {
for(o = u->orders; o; o=o->next) {
if(igetkeyword(o->s, u->faction->locale) == K_JOINUGROUP) {
join_ugroup(u, o);
}
}
}
}
}
#endif /* USE_UGROUPS */