2002-08-26 16:16:16 +02:00
|
|
|
/* 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>
|
2003-07-29 11:48:03 +02:00
|
|
|
| (c) 1998 - 2003 | Henning Peters <faroul@beyond.kn-bremen.de>
|
2002-08-26 16:16:16 +02:00
|
|
|
| | 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>
|
2004-06-27 18:56:01 +02:00
|
|
|
#include "eressea.h"
|
2003-07-29 11:48:03 +02:00
|
|
|
#include "alliance.h"
|
2002-08-26 16:16:16 +02:00
|
|
|
|
2003-12-15 23:59:15 +01:00
|
|
|
#include <attributes/key.h>
|
|
|
|
|
2002-08-26 16:16:16 +02:00
|
|
|
/* kernel includes */
|
2003-12-15 23:59:15 +01:00
|
|
|
#include <kernel/building.h>
|
|
|
|
#include <kernel/faction.h>
|
|
|
|
#include <kernel/message.h>
|
2004-06-21 18:45:27 +02:00
|
|
|
#include <kernel/order.h>
|
2003-12-15 23:59:15 +01:00
|
|
|
#include <kernel/region.h>
|
|
|
|
#include <kernel/unit.h>
|
|
|
|
#include <kernel/item.h>
|
2002-08-26 16:16:16 +02:00
|
|
|
|
|
|
|
/* util includes */
|
2004-06-27 18:56:01 +02:00
|
|
|
#include <util/command.h>
|
|
|
|
#include <util/umlaut.h>
|
|
|
|
#include <util/base36.h>
|
2002-08-26 16:16:16 +02:00
|
|
|
|
|
|
|
/* libc includes */
|
2003-12-15 23:59:15 +01:00
|
|
|
#include <assert.h>
|
2002-08-26 16:16:16 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
alliance * alliances = NULL;
|
|
|
|
|
|
|
|
alliance *
|
|
|
|
makealliance(int id, const char * name)
|
|
|
|
{
|
|
|
|
alliance * al = calloc(1, sizeof(alliance));
|
|
|
|
al->id=id;
|
|
|
|
al->name=strdup(name);
|
|
|
|
al->next=alliances;
|
|
|
|
return alliances=al;
|
|
|
|
}
|
|
|
|
|
|
|
|
alliance *
|
|
|
|
findalliance(int id)
|
|
|
|
{
|
|
|
|
alliance * al;
|
|
|
|
for (al=alliances;al;al=al->next) {
|
|
|
|
if (al->id==id) return al;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void destroy_kick(struct attrib * a)
|
|
|
|
{
|
|
|
|
faction_list * flist = (faction_list*)a->data.v;
|
|
|
|
freelist(flist);
|
|
|
|
}
|
|
|
|
|
|
|
|
const attrib_type at_kick = { "kick",
|
|
|
|
DEFAULT_INIT, destroy_kick
|
|
|
|
};
|
|
|
|
|
|
|
|
static attrib *
|
|
|
|
make_kick(void)
|
|
|
|
{
|
|
|
|
return a_new(&at_kick);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
add_kick(attrib * a, const faction * f)
|
|
|
|
{
|
|
|
|
faction_list * flist = (faction_list*)a->data.v;
|
|
|
|
while (flist && flist->data!=f) flist = flist->next;
|
|
|
|
if (flist) return;
|
|
|
|
flist = calloc(1, sizeof(faction_list));
|
|
|
|
flist->data = (void*)f;
|
|
|
|
flist->next = (faction_list*)a->data.v;
|
|
|
|
a->data.v = flist;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2004-06-21 18:45:27 +02:00
|
|
|
alliance_kick(const tnode * tnext, const char * str, void * data, struct order * ord)
|
2002-08-26 16:16:16 +02:00
|
|
|
{
|
|
|
|
unit * u = (unit*)data;
|
|
|
|
faction * f = findfaction(atoi36(igetstrtoken(str)));
|
|
|
|
attrib * a;
|
2002-09-02 22:36:12 +02:00
|
|
|
unused(tnext);
|
2002-08-26 16:16:16 +02:00
|
|
|
|
|
|
|
if (f==NULL || f->alliance!=u->faction->alliance) {
|
|
|
|
/* does not belong to our alliance */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
a = a_find(f->attribs, &at_kick);
|
|
|
|
if (a==NULL) a = a_add(&f->attribs, make_kick());
|
|
|
|
add_kick(a, f);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2004-06-21 18:45:27 +02:00
|
|
|
alliance_join(const tnode * tnext, const char * str, void * data, struct order * ord)
|
2002-08-26 16:16:16 +02:00
|
|
|
{
|
|
|
|
unit * u = (unit*)data;
|
|
|
|
alliance * al = findalliance(atoi36(igetstrtoken(str)));
|
2002-09-02 22:36:12 +02:00
|
|
|
unused(tnext);
|
|
|
|
|
2002-08-26 16:16:16 +02:00
|
|
|
if (u->faction->alliance!=NULL || al==NULL) {
|
|
|
|
/* not found */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
u->faction->alliance = al;
|
|
|
|
/* inform the rest? */
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2002-09-02 22:36:12 +02:00
|
|
|
execute(const struct syntaxtree * syntax)
|
2002-08-26 16:16:16 +02:00
|
|
|
{
|
|
|
|
region ** rp = ®ions;
|
|
|
|
while (*rp) {
|
|
|
|
region * r = *rp;
|
|
|
|
unit **up = &r->units;
|
|
|
|
while (*up) {
|
|
|
|
unit * u = *up;
|
2002-09-02 22:36:12 +02:00
|
|
|
const struct locale * lang = u->faction->locale;
|
|
|
|
tnode * root = stree_find(syntax, lang);
|
2004-06-21 18:45:27 +02:00
|
|
|
order * ord;
|
|
|
|
for (ord = u->orders; ord; ord = ord->next) {
|
|
|
|
if (get_keyword(ord) == K_ALLIANCE) {
|
|
|
|
do_command(root, u, ord);
|
2002-08-26 16:16:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (u==*up) up = &u->next;
|
|
|
|
}
|
|
|
|
if (*rp==r) rp = &r->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2002-09-02 22:36:12 +02:00
|
|
|
alliancekick(void)
|
2002-08-26 16:16:16 +02:00
|
|
|
{
|
2002-09-02 22:36:12 +02:00
|
|
|
static syntaxtree * stree = NULL;
|
|
|
|
faction * f = factions;
|
|
|
|
if (stree==NULL) {
|
|
|
|
syntaxtree * slang = stree = stree_create();
|
|
|
|
while (slang) {
|
|
|
|
struct tnode * root = calloc(sizeof(tnode), 1);
|
|
|
|
struct tnode * leaf = calloc(sizeof(tnode), 1);
|
|
|
|
add_command(root, leaf, LOC(slang->lang, "alliance"), NULL);
|
|
|
|
add_command(leaf, NULL, LOC(slang->lang, "kick"), &alliance_kick);
|
|
|
|
slang = slang->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
execute(stree);
|
|
|
|
while (f) {
|
|
|
|
attrib * a = a_find(f->attribs, &at_kick);
|
|
|
|
if (a!=NULL) {
|
|
|
|
faction_list * flist = (faction_list*)a->data.v;
|
|
|
|
if (flist!=NULL) {
|
|
|
|
unsigned int votes = listlen(flist);
|
|
|
|
unsigned int size = listlen(f->alliance->members);
|
|
|
|
if (size<=votes*2) {
|
|
|
|
f->alliance = NULL;
|
|
|
|
/* tell him he's been kicked */
|
|
|
|
for (flist=f->alliance->members;flist;flist=flist->next) {
|
|
|
|
ADDMSG(&flist->data->msgs, msg_message("alliance::kickedout",
|
|
|
|
"member alliance votes", f, f->alliance, votes));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* warn that he's been attempted to kick */
|
|
|
|
for (flist=f->alliance->members;flist;flist=flist->next) {
|
|
|
|
ADDMSG(&flist->data->msgs, msg_message("alliance::kickattempt",
|
|
|
|
"member alliance votes", f, f->alliance, votes));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
f = f->next;
|
|
|
|
}
|
|
|
|
/* some may have been kicked, must remove f->alliance==NULL */
|
2002-08-26 16:16:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2002-09-02 22:36:12 +02:00
|
|
|
alliancejoin(void)
|
|
|
|
{
|
|
|
|
static syntaxtree * stree = NULL;
|
|
|
|
if (stree==NULL) {
|
|
|
|
syntaxtree * slang = stree = stree_create();
|
|
|
|
while (slang) {
|
|
|
|
struct tnode * root = calloc(sizeof(tnode), 1);
|
|
|
|
struct tnode * leaf = calloc(sizeof(tnode), 1);
|
|
|
|
add_command(root, leaf, LOC(slang->lang, "alliance"), NULL);
|
|
|
|
add_command(leaf, NULL, LOC(slang->lang, "join"), &alliance_join);
|
|
|
|
slang = slang->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
execute(stree);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
setalliance(struct faction * f, alliance * al)
|
2002-08-26 16:16:16 +02:00
|
|
|
{
|
2002-09-02 22:36:12 +02:00
|
|
|
if (f->alliance==al) return;
|
|
|
|
if (f->alliance!=NULL) {
|
|
|
|
faction_list ** flistp = &f->alliance->members;
|
|
|
|
while (*flistp) {
|
|
|
|
if ((*flistp)->data==f) {
|
|
|
|
*flistp = (*flistp)->next;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
flistp = &(*flistp)->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
f->alliance = al;
|
|
|
|
if (al!=NULL) {
|
|
|
|
faction_list * flist = calloc(sizeof(faction_list), 1);
|
|
|
|
flist->next = al->members;
|
|
|
|
flist->data = f;
|
|
|
|
al->members = flist;
|
|
|
|
}
|
2002-08-26 16:16:16 +02:00
|
|
|
}
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
const char *
|
|
|
|
alliancename(const alliance * al)
|
|
|
|
{
|
2007-02-25 19:49:28 +01:00
|
|
|
typedef char name[OBJECTIDSIZE + 1];
|
|
|
|
static name idbuf[8];
|
|
|
|
static int nextbuf = 0;
|
2002-09-02 22:36:12 +02:00
|
|
|
|
2007-02-25 19:49:28 +01:00
|
|
|
char *ibuf = idbuf[(++nextbuf) % 8];
|
2002-09-02 22:36:12 +02:00
|
|
|
|
2007-02-25 19:49:28 +01:00
|
|
|
if (al && al->name) {
|
|
|
|
snprintf(ibuf, sizeof(name), "%s (%s)", strcheck(al->name, NAMESIZE), itoa36(al->id));
|
2007-02-28 22:50:19 +01:00
|
|
|
ibuf[sizeof(name)-1] = 0;
|
2007-02-25 19:49:28 +01:00
|
|
|
} else {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return ibuf;
|
2002-09-02 22:36:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
alliancevictory(void)
|
|
|
|
{
|
|
|
|
const struct building_type * btype = bt_find("stronghold");
|
|
|
|
region * r = regions;
|
|
|
|
alliance * al = alliances;
|
2003-05-27 23:51:36 +02:00
|
|
|
if (btype==NULL) return;
|
2002-09-02 22:36:12 +02:00
|
|
|
while (r!=NULL) {
|
|
|
|
building * b = r->buildings;
|
|
|
|
while (b!=NULL) {
|
|
|
|
if (b->type==btype) {
|
|
|
|
unit * u = buildingowner(r, b);
|
|
|
|
if (u) {
|
|
|
|
fset(u->faction->alliance, FL_MARK);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
b = b->next;
|
|
|
|
}
|
|
|
|
r=r->next;
|
|
|
|
}
|
|
|
|
while (al!=NULL) {
|
|
|
|
if (!fval(al, FL_MARK)) {
|
|
|
|
faction_list * flist = al->members;
|
|
|
|
while (flist!=0) {
|
|
|
|
faction * f = flist->data;
|
|
|
|
if (f->alliance==al) {
|
|
|
|
ADDMSG(&f->msgs, msg_message("alliance::lost",
|
|
|
|
"alliance", al));
|
|
|
|
destroyfaction(f);
|
|
|
|
}
|
|
|
|
flist = flist->next;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
freset(al, FL_MARK);
|
|
|
|
}
|
2002-09-29 23:20:04 +02:00
|
|
|
al = al->next;
|
2002-09-02 22:36:12 +02:00
|
|
|
}
|
|
|
|
}
|
2003-12-15 23:59:15 +01:00
|
|
|
|
|
|
|
int
|
|
|
|
victorycondition(const alliance * al, const char * name)
|
|
|
|
{
|
|
|
|
const char * gems[] = { "opal", "diamond", "zaphire", "topaz", "beryl", "agate", "garnet", "emerald", NULL };
|
|
|
|
if (strcmp(name, "gems")==0) {
|
|
|
|
const char ** igem = gems;
|
|
|
|
|
|
|
|
for (;*igem;++igem) {
|
|
|
|
const struct item_type * itype = it_find(*igem);
|
|
|
|
faction_list * flist = al->members;
|
|
|
|
boolean found = false;
|
|
|
|
|
|
|
|
assert(itype!=NULL);
|
|
|
|
for (;flist && !found;flist=flist->next) {
|
|
|
|
unit * u = flist->data->units;
|
|
|
|
|
|
|
|
for (;u;u=u->nextF) {
|
|
|
|
if (i_get(u->items, itype)>0) {
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found) return 0;
|
|
|
|
}
|
|
|
|
return 1;
|
2006-05-07 23:11:49 +02:00
|
|
|
|
2003-12-15 23:59:15 +01:00
|
|
|
} else if (strcmp(name, "phoenix")==0) {
|
|
|
|
faction_list * flist = al->members;
|
|
|
|
for (;flist;flist=flist->next) {
|
|
|
|
faction * f = flist->data;
|
|
|
|
if (find_key(f->attribs, atoi36("phnx"))) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
2006-05-07 23:11:49 +02:00
|
|
|
|
2003-12-15 23:59:15 +01:00
|
|
|
} else if (strcmp(name, "pyramid")==0) {
|
2006-05-07 23:11:49 +02:00
|
|
|
|
|
|
|
/* Logik:
|
|
|
|
* - if (pyr > last_passed_size && pyr > all_others) {
|
|
|
|
* pyr->passed->counter++;
|
|
|
|
* for(all_other_pyrs) {
|
|
|
|
* pyr->passed->counter=0;
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* if(pyr->passed->counter >= 3) {
|
|
|
|
* set(pyr, passed);
|
|
|
|
* pyr->owner->set_attrib(pyra);
|
|
|
|
* }
|
|
|
|
* last_passed_size = pyr->size;
|
|
|
|
* }
|
|
|
|
*/
|
|
|
|
|
2003-12-15 23:59:15 +01:00
|
|
|
faction_list * flist = al->members;
|
|
|
|
for (;flist;flist=flist->next) {
|
|
|
|
faction * f = flist->data;
|
|
|
|
if (find_key(f->attribs, atoi36("pyra"))) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|