server/src/common/kernel/group.c

227 lines
4 KiB
C
Raw Normal View History

2001-01-25 10:37:55 +01:00
/* vi: set ts=2:
*
*
* Eressea PB(E)M host Copyright (C) 1998-2003
2001-01-25 10:37:55 +01:00
* Christian Schlittchen (corwin@amber.kn-bremen.de)
* Katja Zedel (katze@felidae.kn-bremen.de)
* Henning Peters (faroul@beyond.kn-bremen.de)
* Enno Rehling (enno@eressea.de)
2001-01-25 10:37:55 +01:00
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
*
* This program may not be used, modified or distributed without
* prior permission by the authors of Eressea.
*/
#include <config.h>
#include <kernel/eressea.h>
2001-01-25 10:37:55 +01:00
#include "group.h"
/* kernel includes */
#include "unit.h"
#include "faction.h"
#include "save.h"
2007-06-20 02:34:02 +02:00
#include "version.h"
2001-01-25 10:37:55 +01:00
/* attrib includes */
#include <attributes/raceprefix.h>
2001-01-25 10:37:55 +01:00
/* util includes */
2007-06-20 02:34:02 +02:00
#include <util/attrib.h>
#include <util/base36.h>
#include <util/resolve.h>
#include <util/unicode.h>
2001-01-25 10:37:55 +01:00
/* libc includes */
#include <stdlib.h>
#include <string.h>
#include <wctype.h>
2001-01-25 10:37:55 +01:00
#define GMAXHASH 2039
2001-01-25 10:37:55 +01:00
static group * ghash[GMAXHASH];
static int maxgid;
static group *
new_group(faction * f, const char * name, int gid)
2001-01-25 10:37:55 +01:00
{
group ** gp = &f->groups;
int index = gid % GMAXHASH;
group * g = calloc(sizeof(group), 1);
while (*gp) gp = &(*gp)->next;
*gp = g;
maxgid = max(gid, maxgid);
g->name = strdup(name);
2001-01-25 10:37:55 +01:00
g->gid = gid;
g->nexthash = ghash[index];
return ghash[index] = g;
}
static void
init_group(faction * f, group * g)
{
ally * a, ** an;
an = &g->allies;
for (a=f->allies;a;a=a->next) if (a->faction) {
ally * ga = calloc(sizeof(ally), 1);
*ga = *a;
*an = ga;
an = &ga->next;
}
}
static group *
find_groupbyname(group * g, const char * name)
2001-01-25 10:37:55 +01:00
{
while (g && unicode_utf8_strcasecmp(name, g->name)!=0) g = g->next;
2001-01-25 10:37:55 +01:00
return g;
}
static group *
find_group(int gid)
{
int index = gid % GMAXHASH;
group * g = ghash[index];
while (g && g->gid!=gid) g = g->nexthash;
return g;
}
static int
read_group(attrib * a, FILE * f)
{
int gid;
group * g;
fscanf(f, "%d ", &gid);
a->data.v = g = find_group(gid);
if (g!=0) {
g->members++;
return AT_READ_OK;
}
return AT_READ_FAIL;
2001-01-25 10:37:55 +01:00
}
static void
write_group(const attrib * a, FILE * f)
{
group * g = (group*)a->data.v;
fprintf(f, "%d ", g->gid);
}
attrib_type
at_group = { /* attribute for units assigned to a group */
"grp",
DEFAULT_INIT,
DEFAULT_FINALIZE,
DEFAULT_AGE,
write_group,
read_group,
ATF_UNIQUE
};
void
free_group(group * g)
{
while (g->allies) {
ally * a = g->allies;
g->allies = a->next;
free(a);
}
free(g->name);
free(g);
}
void
set_group(struct unit * u, struct group * g)
2001-01-25 10:37:55 +01:00
{
attrib * a = NULL;
if (fval(u, UFL_GROUP)) {
a = a_find(u->attribs, &at_group);
}
2001-01-25 10:37:55 +01:00
if (a) {
group * og = (group *)a->data.v;
if (og==g) return;
--og->members;
}
if (g) {
if (!a) {
a = a_add(&u->attribs, a_new(&at_group));
fset(u, UFL_GROUP);
}
a->data.v = g;
g->members++;
} else if (a) {
a_remove(&u->attribs, a);
freset(u, UFL_GROUP);
}
}
boolean
join_group(unit * u, const char * name)
{
2008-02-03 16:48:38 +01:00
group * g = NULL;
if (name && name[0]) {
g = find_groupbyname(u->faction->groups, name);
if (g==NULL) {
g = new_group(u->faction, name, ++maxgid);
init_group(u->faction, g);
}
}
set_group(u, g);
2008-02-03 16:48:38 +01:00
return true;
2001-01-25 10:37:55 +01:00
}
void
write_groups(FILE * F, group * g)
{
while (g) {
ally * a;
fprintf(F, "%d \"%s\" ", g->gid, g->name);
for (a=g->allies;a;a=a->next) if (a->faction)
fprintf(F, "%s %d ", factionid(a->faction), a->status);
fputs("0 ", F);
a_write(F, g->attribs);
fputs("\n", F);
2001-01-25 10:37:55 +01:00
g=g->next;
}
fputs("0\n", F);
2001-01-25 10:37:55 +01:00
}
void
read_groups(FILE * F, faction * f)
{
for(;;) {
ally ** pa;
group * g;
int gid;
char buf[1024];
fscanf(F, "%d ", &gid);
2001-01-25 10:37:55 +01:00
if (!gid) break;
freadstr(F, enc_gamedata, buf, sizeof(buf));
g = new_group(f, buf, gid);
2001-01-25 10:37:55 +01:00
pa = &g->allies;
for (;;) {
ally * a;
variant aid;
2002-09-02 22:36:12 +02:00
fscanf(F, "%s ", buf);
aid.i = atoi36(buf);
if (aid.i==0) break;
a = malloc(sizeof(ally));
2001-01-25 10:37:55 +01:00
*pa = a;
pa = &a->next;
fscanf(F, "%d ", &a->status);
a->faction = findfaction(aid.i);
if (!a->faction) ur_add(aid, (void**)&a->faction, resolve_faction);
2001-01-25 10:37:55 +01:00
}
*pa = 0;
a_read(F, &g->attribs);
2001-01-25 10:37:55 +01:00
}
}