2002-03-17 09:24:52 +01: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-03-17 09:24:52 +01:00
|
|
|
| | Ingo Wilken <Ingo.Wilken@informatik.uni-oldenburg.de>
|
|
|
|
+-------------------+ Stefan Reich <reich@halbling.de>
|
|
|
|
|
2004-10-14 09:10:03 +02:00
|
|
|
This program may not be used, modified or distributed
|
2002-03-17 09:24:52 +01:00
|
|
|
without prior permission by the authors of Eressea.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
2005-03-24 00:39:26 +01:00
|
|
|
#include <kernel/eressea.h>
|
2002-03-17 09:24:52 +01:00
|
|
|
#include "autoseed.h"
|
|
|
|
|
|
|
|
/* kernel includes */
|
2005-03-24 00:39:26 +01:00
|
|
|
#include <kernel/alliance.h>
|
2005-04-11 23:49:30 +02:00
|
|
|
#include <kernel/item.h>
|
2005-03-24 00:39:26 +01:00
|
|
|
#include <kernel/region.h>
|
|
|
|
#include <kernel/plane.h>
|
|
|
|
#include <kernel/faction.h>
|
|
|
|
#include <kernel/race.h>
|
2005-10-25 14:38:01 +02:00
|
|
|
#include <kernel/terrain.h>
|
|
|
|
#include <kernel/terrainid.h>
|
2005-03-24 00:39:26 +01:00
|
|
|
#include <kernel/unit.h>
|
|
|
|
|
|
|
|
/* util includes */
|
|
|
|
#include <util/base36.h>
|
2005-04-25 19:44:19 +02:00
|
|
|
#include <util/sql.h>
|
2005-03-24 00:39:26 +01:00
|
|
|
#include <util/goodies.h>
|
2005-04-16 18:44:00 +02:00
|
|
|
#include <util/sql.h>
|
2002-03-17 09:24:52 +01:00
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
#include <limits.h>
|
|
|
|
#include <memory.h>
|
2002-04-11 19:37:06 +02:00
|
|
|
#include <string.h>
|
2002-03-17 12:42:23 +01:00
|
|
|
#include <stdlib.h>
|
2002-03-17 09:24:52 +01:00
|
|
|
|
2005-11-26 01:42:32 +01:00
|
|
|
const terrain_type *
|
|
|
|
random_terrain(unsigned int flags)
|
|
|
|
{
|
|
|
|
static int nterrains;
|
|
|
|
const terrain_type * terrain;
|
|
|
|
int n;
|
|
|
|
if (nterrains==0) {
|
|
|
|
for (terrain=terrains();terrain;terrain=terrain->next) {
|
|
|
|
if (fval(terrain, flags)==flags) {
|
|
|
|
++nterrains;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
n = rand() % nterrains;
|
2005-12-04 01:58:57 +01:00
|
|
|
for (terrain=terrains();terrain;terrain=terrain->next) {
|
2005-11-26 01:42:32 +01:00
|
|
|
if (fval(terrain, flags)==flags) {
|
2005-12-04 01:58:57 +01:00
|
|
|
if (n--==0) break;
|
2005-11-26 01:42:32 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return terrain;
|
|
|
|
}
|
|
|
|
|
2005-04-11 23:49:30 +02:00
|
|
|
|
|
|
|
static int
|
|
|
|
count_demand(const region *r)
|
|
|
|
{
|
|
|
|
struct demand *dmd;
|
|
|
|
int c = 0;
|
|
|
|
if (r->land) {
|
|
|
|
for (dmd=r->land->demands;dmd;dmd=dmd->next) c++;
|
|
|
|
}
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
recurse_regions(region *r, region_list **rlist, boolean(*fun)(const region *r))
|
|
|
|
{
|
|
|
|
if (!fun(r)) return 0;
|
|
|
|
else {
|
|
|
|
int len = 0;
|
|
|
|
direction_t d;
|
|
|
|
region_list * rl = calloc(sizeof(region_list), 1);
|
|
|
|
rl->next = *rlist;
|
|
|
|
rl->data = r;
|
|
|
|
(*rlist) = rl;
|
|
|
|
fset(r, FL_MARK);
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
region * nr = rconnect(r, d);
|
|
|
|
if (nr && !fval(nr, FL_MARK)) len += recurse_regions(nr, rlist, fun);
|
|
|
|
}
|
|
|
|
return len+1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static boolean
|
|
|
|
f_nolux(const region * r)
|
|
|
|
{
|
2005-11-26 02:16:55 +01:00
|
|
|
if (r->land && count_demand(r) != get_maxluxuries()) return true;
|
2005-04-11 23:49:30 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
fix_demand(region *r)
|
|
|
|
{
|
|
|
|
region_list *rl, *rlist = NULL;
|
|
|
|
static const struct luxury_type **mlux = 0, ** ltypes;
|
|
|
|
const luxury_type *sale = NULL;
|
|
|
|
int maxlux = 0;
|
2005-11-26 02:16:55 +01:00
|
|
|
int maxluxuries = get_maxluxuries();
|
2005-04-11 23:49:30 +02:00
|
|
|
|
2005-11-26 02:16:55 +01:00
|
|
|
if (maxluxuries==0) return 0;
|
2005-04-11 23:49:30 +02:00
|
|
|
recurse_regions(r, &rlist, f_nolux);
|
|
|
|
if (mlux==0) {
|
|
|
|
int i = 0;
|
|
|
|
mlux = (const luxury_type **)gc_add(calloc(maxluxuries, sizeof(const luxury_type *)));
|
|
|
|
ltypes = (const luxury_type **)gc_add(calloc(maxluxuries, sizeof(const luxury_type *)));
|
|
|
|
for (sale=luxurytypes;sale;sale=sale->next) {
|
|
|
|
ltypes[i++] = sale;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
int i;
|
|
|
|
for (i=0;i!=maxluxuries;++i) mlux[i] = 0;
|
|
|
|
}
|
|
|
|
for (rl=rlist;rl;rl=rl->next) {
|
|
|
|
region * r = rl->data;
|
|
|
|
direction_t d;
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
region * nr = rconnect(r, d);
|
|
|
|
if (nr && nr->land && nr->land->demands) {
|
|
|
|
struct demand * dmd;
|
|
|
|
for (dmd = nr->land->demands;dmd;dmd=dmd->next) {
|
|
|
|
if (dmd->value == 0) {
|
|
|
|
int i;
|
|
|
|
for (i=0;i!=maxluxuries;++i) {
|
|
|
|
if (mlux[i]==NULL) {
|
|
|
|
maxlux = i;
|
|
|
|
mlux[i] = dmd->type;
|
|
|
|
break;
|
|
|
|
} else if (mlux[i]==dmd->type) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
freset(r, FL_MARK); /* undo recursive marker */
|
|
|
|
}
|
|
|
|
if (maxlux<2) {
|
|
|
|
int i;
|
|
|
|
for (i=maxlux;i!=2;++i) {
|
|
|
|
int j;
|
|
|
|
do {
|
|
|
|
int k = rand() % maxluxuries;
|
|
|
|
mlux[i] = ltypes[k];
|
|
|
|
for (j=0;j!=i;++j) {
|
|
|
|
if (mlux[j]==mlux[i]) break;
|
|
|
|
}
|
|
|
|
} while (j!=i);
|
|
|
|
}
|
2005-11-26 02:16:55 +01:00
|
|
|
maxlux = 2;
|
2005-04-11 23:49:30 +02:00
|
|
|
}
|
|
|
|
for (rl=rlist;rl;rl=rl->next) {
|
|
|
|
region * r = rl->data;
|
|
|
|
if (!fval(r, RF_CHAOTIC)) {
|
|
|
|
log_warning(("fixing demand in %s\n", regionname(r, NULL)));
|
|
|
|
}
|
2005-11-26 02:16:55 +01:00
|
|
|
sale = mlux[rand() % maxlux];
|
|
|
|
if (sale) setluxuries(r, sale);
|
2005-04-11 23:49:30 +02:00
|
|
|
}
|
|
|
|
while (rlist) {
|
|
|
|
rl = rlist->next;
|
|
|
|
free(rlist);
|
|
|
|
rlist = rl;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-10-16 22:11:34 +02:00
|
|
|
/* nach 150 Runden ist Neustart erlaubt */
|
2005-10-10 06:59:54 +02:00
|
|
|
#define MINAGE_MULTI 150
|
2005-03-24 00:39:26 +01:00
|
|
|
newfaction *
|
|
|
|
read_newfactions(const char * filename)
|
2002-03-17 09:24:52 +01:00
|
|
|
{
|
2005-03-24 00:39:26 +01:00
|
|
|
newfaction * newfactions = NULL;
|
|
|
|
FILE * F = fopen(filename, "r");
|
|
|
|
if (F==NULL) return NULL;
|
|
|
|
for (;;) {
|
|
|
|
faction * f;
|
|
|
|
char race[20], email[64], lang[8], password[16];
|
|
|
|
newfaction *nf, **nfi;
|
2005-12-11 16:09:57 +01:00
|
|
|
int bonus = 0, subscription = 0;
|
2005-03-24 00:39:26 +01:00
|
|
|
int alliance = 0;
|
2002-03-17 09:24:52 +01:00
|
|
|
|
2005-12-11 16:09:57 +01:00
|
|
|
if (fgets(buf, sizeof(buf), F)==NULL) break;
|
|
|
|
|
|
|
|
email[0] = '\0';
|
|
|
|
password[0] = '\0';
|
|
|
|
|
|
|
|
sscanf(buf, "%s %s %s %d %d %s %d", email, race, lang, &bonus, &subscription, password, &alliance);
|
|
|
|
if (email[0]=='\0') break;
|
|
|
|
if (password[0]=='\0') {
|
|
|
|
strcpy(password, itoa36(rand()));
|
|
|
|
strcat(password, itoa36(rand()));
|
2005-03-24 00:39:26 +01:00
|
|
|
}
|
|
|
|
for (f=factions;f;f=f->next) {
|
2005-10-10 06:59:54 +02:00
|
|
|
if (strcmp(f->email, email)==0 && f->subscription && f->age<MINAGE_MULTI) break;
|
2005-03-24 00:39:26 +01:00
|
|
|
}
|
2005-04-03 10:40:13 +02:00
|
|
|
if (f && f->units) continue; /* skip the ones we've already got */
|
2005-03-24 00:39:26 +01:00
|
|
|
for (nf=newfactions;nf;nf=nf->next) {
|
|
|
|
if (strcmp(nf->email, email)==0) break;
|
|
|
|
}
|
|
|
|
if (nf) continue;
|
|
|
|
nf = calloc(sizeof(newfaction), 1);
|
|
|
|
if (set_email(&nf->email, email)!=0) {
|
2005-04-11 23:49:30 +02:00
|
|
|
log_error(("Invalid email address for subscription %s: %s\n",
|
|
|
|
itoa36(subscription), email));
|
2005-03-24 00:39:26 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
nf->password = strdup(password);
|
|
|
|
nf->race = rc_find(race);
|
|
|
|
nf->subscription = subscription;
|
|
|
|
if (alliances!=NULL) {
|
|
|
|
struct alliance * al = findalliance(alliance);
|
|
|
|
if (al==NULL) {
|
|
|
|
char zText[64];
|
|
|
|
sprintf(zText, "Allianz %d", alliance);
|
|
|
|
al = makealliance(alliance, zText);
|
|
|
|
}
|
|
|
|
nf->allies = al;
|
|
|
|
} else {
|
|
|
|
nf->allies = NULL;
|
|
|
|
}
|
|
|
|
if (nf->race==NULL) nf->race = findrace(race, default_locale);
|
|
|
|
nf->lang = find_locale(lang);
|
|
|
|
nf->bonus = bonus;
|
|
|
|
assert(nf->race && nf->email && nf->lang);
|
|
|
|
nfi = &newfactions;
|
|
|
|
while (*nfi) {
|
|
|
|
if ((*nfi)->race==nf->race) break;
|
|
|
|
nfi=&(*nfi)->next;
|
|
|
|
}
|
|
|
|
nf->next = *nfi;
|
|
|
|
*nfi = nf;
|
|
|
|
}
|
|
|
|
fclose(F);
|
|
|
|
return newfactions;
|
2002-03-17 09:24:52 +01:00
|
|
|
}
|
|
|
|
|
2005-10-25 14:46:25 +02:00
|
|
|
#ifdef SEED_QUALITY /* does this work? */
|
2002-03-17 09:24:52 +01:00
|
|
|
typedef struct seed_t {
|
2005-10-25 14:38:01 +02:00
|
|
|
struct region * region;
|
|
|
|
struct newfaction * player;
|
|
|
|
struct seed_t * next[MAXDIRECTIONS];
|
2002-03-17 09:24:52 +01:00
|
|
|
} seed_t;
|
|
|
|
|
2005-10-25 14:46:25 +02:00
|
|
|
static double
|
2002-03-17 09:24:52 +01:00
|
|
|
get_influence(struct seed_t * seed, struct seed_t * target)
|
|
|
|
{
|
|
|
|
double q = 0.0;
|
|
|
|
direction_t d;
|
|
|
|
if (target==0 || seed==0) return q;
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
seed_t * s = seed->next[d];
|
|
|
|
if (s==target) {
|
|
|
|
if (target->player) {
|
|
|
|
if (seed->player) {
|
|
|
|
/* neighbours bad. */
|
|
|
|
q -= 4.0;
|
|
|
|
if (seed->player->race==target->player->race) {
|
|
|
|
/* don't want similar people next to each other */
|
|
|
|
q -= 2.0;
|
|
|
|
} else if (seed->player->race==new_race[RC_ELF]) {
|
|
|
|
if (target->player->race==new_race[RC_TROLL]) {
|
|
|
|
/* elf sitting next to troll: poor troll */
|
|
|
|
q -= 5.0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (target->player) {
|
|
|
|
/* empty regions good */
|
|
|
|
q += 1.0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return q;
|
|
|
|
}
|
|
|
|
|
2005-10-25 14:46:25 +02:00
|
|
|
static double
|
2002-03-17 09:24:52 +01:00
|
|
|
get_quality(struct seed_t * seed)
|
|
|
|
{
|
|
|
|
double q = 0.0;
|
2005-10-25 14:38:01 +02:00
|
|
|
int nterrains[T_ASTRAL];
|
2002-03-17 09:24:52 +01:00
|
|
|
direction_t d;
|
|
|
|
|
|
|
|
memset(nterrains, 0, sizeof(nterrains));
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
seed_t * ns = seed->next[d];
|
2005-10-25 14:38:01 +02:00
|
|
|
if (ns) {
|
|
|
|
terrain_t t = rterrain(ns->region);
|
|
|
|
if (t<T_ASTRAL) ++nterrains[t];
|
|
|
|
}
|
2002-03-17 09:24:52 +01:00
|
|
|
}
|
|
|
|
++nterrains[rterrain(seed->region)];
|
|
|
|
|
|
|
|
if (nterrains[T_MOUNTAIN]) {
|
|
|
|
/* 5 points for the first, and 2 points for every other mountain */
|
|
|
|
q += (3+nterrains[T_MOUNTAIN]*2);
|
|
|
|
} else if (seed->player->race==new_race[RC_DWARF]) {
|
|
|
|
/* -10 for a dwarf who isn't close to a mountain */
|
|
|
|
q -= 10.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nterrains[T_PLAIN]) {
|
|
|
|
/* 5 points for the first, and 2 points for every other plain */
|
|
|
|
q += (3+nterrains[T_PLAIN]*2);
|
|
|
|
} else if (seed->player->race==new_race[RC_ELF]) {
|
|
|
|
/* -10 for an elf who isn't close to a plain */
|
|
|
|
q -= 10.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nterrains[T_DESERT]) {
|
2004-10-14 09:10:03 +02:00
|
|
|
/* +10 points for the first if we are insects, and -2 points for every
|
2002-03-17 09:24:52 +01:00
|
|
|
* other desert */
|
|
|
|
if (seed->player->race==new_race[RC_INSECT]) q += 12;
|
|
|
|
q -= (nterrains[T_DESERT]*2);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (rterrain(seed->region)) {
|
|
|
|
case T_DESERT:
|
|
|
|
q -=8.0;
|
|
|
|
break;
|
|
|
|
case T_VOLCANO:
|
|
|
|
q -=20.0;
|
|
|
|
break;
|
|
|
|
case T_GLACIER:
|
|
|
|
q -=10.0;
|
|
|
|
break;
|
|
|
|
case T_SWAMP:
|
|
|
|
q -= 1.0;
|
|
|
|
break;
|
|
|
|
case T_PLAIN:
|
|
|
|
q += 4.0;
|
|
|
|
break;
|
|
|
|
case T_MOUNTAIN:
|
|
|
|
q += 5.0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 2.0;
|
|
|
|
}
|
|
|
|
|
2005-10-25 14:38:01 +02:00
|
|
|
static double
|
2002-03-17 09:24:52 +01:00
|
|
|
recalc(seed_t * seeds, int nseeds, int nplayers)
|
|
|
|
{
|
|
|
|
int i, p = 0;
|
|
|
|
double quality = 0.0, q = 0.0, *qarr = malloc(sizeof(int) * nplayers);
|
|
|
|
for (i=0;i!=nseeds;++i) {
|
|
|
|
if (seeds[i].player) {
|
|
|
|
double quality = get_quality(seeds+i);
|
|
|
|
direction_t d;
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
quality += get_influence(seeds[i].next[d], seeds+i);
|
|
|
|
}
|
|
|
|
qarr[p++] = quality;
|
|
|
|
q += quality;
|
|
|
|
}
|
|
|
|
}
|
2005-03-24 00:39:26 +01:00
|
|
|
|
2002-03-17 09:24:52 +01:00
|
|
|
return quality + q;
|
|
|
|
}
|
2005-10-25 14:46:25 +02:00
|
|
|
#endif
|
2002-03-17 09:24:52 +01:00
|
|
|
|
2002-03-17 13:31:54 +01:00
|
|
|
extern int numnewbies;
|
|
|
|
|
2002-04-11 19:37:06 +02:00
|
|
|
static terrain_t
|
|
|
|
preferred_terrain(const struct race * rc)
|
|
|
|
{
|
2002-04-16 00:41:03 +02:00
|
|
|
if (rc==rc_find("dwarf")) return T_MOUNTAIN;
|
|
|
|
if (rc==rc_find("insect")) return T_DESERT;
|
|
|
|
if (rc==rc_find("halfling")) return T_SWAMP;
|
|
|
|
if (rc==rc_find("troll")) return T_MOUNTAIN;
|
2002-04-11 19:37:06 +02:00
|
|
|
return T_PLAIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define REGIONS_PER_FACTION 2
|
2005-04-03 10:32:16 +02:00
|
|
|
#define PLAYERS_PER_ISLAND 20
|
2005-09-10 23:57:02 +02:00
|
|
|
#define MAXISLANDSIZE 50
|
2002-04-16 00:41:03 +02:00
|
|
|
#define MINFACTIONS 1
|
2002-05-25 09:02:20 +02:00
|
|
|
#define MAXAGEDIFF 5
|
2004-08-16 07:46:38 +02:00
|
|
|
#define VOLCANO_CHANCE 100
|
2002-04-11 19:37:06 +02:00
|
|
|
|
2005-04-03 10:32:16 +02:00
|
|
|
static boolean
|
|
|
|
virgin_region(const region * r)
|
|
|
|
{
|
|
|
|
direction_t d;
|
|
|
|
if (r==NULL) return true;
|
|
|
|
if (r->age>MAXAGEDIFF) return false;
|
|
|
|
if (r->units) return false;
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
const region * rn = rconnect(r, d);
|
|
|
|
if (rn) {
|
|
|
|
if (rn->age>MAXAGEDIFF) return false;
|
|
|
|
if (rn->units) return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
get_island(region * root, region_list ** rlist)
|
|
|
|
{
|
|
|
|
region_list ** rnext = rlist;
|
|
|
|
while (*rnext) rnext=&(*rnext)->next;
|
|
|
|
|
|
|
|
fset(root, FL_MARK);
|
|
|
|
add_regionlist(rnext, root);
|
|
|
|
|
|
|
|
while (*rnext) {
|
|
|
|
direction_t dir;
|
|
|
|
|
|
|
|
region * rcurrent = (*rnext)->data;
|
|
|
|
rnext = &(*rnext)->next;
|
|
|
|
|
|
|
|
for (dir=0;dir!=MAXDIRECTIONS;++dir) {
|
|
|
|
region * r = rconnect(rcurrent, dir);
|
|
|
|
if (r!=NULL && r->land && !fval(r, FL_MARK)) {
|
|
|
|
fset(r, FL_MARK);
|
|
|
|
add_regionlist(rnext, r);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rnext=rlist;
|
|
|
|
while (*rnext) {
|
|
|
|
region_list * rptr = *rnext;
|
|
|
|
freset(rptr->data, FL_MARK);
|
|
|
|
rnext = &rptr->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-10-09 21:52:21 +02:00
|
|
|
static int
|
2005-09-10 23:57:02 +02:00
|
|
|
island_size(region * r)
|
|
|
|
{
|
|
|
|
int size = 0;
|
|
|
|
region_list * rlist = NULL;
|
|
|
|
region_list * island = NULL;
|
|
|
|
add_regionlist(&rlist, r);
|
|
|
|
island = rlist;
|
|
|
|
fset(r, FL_MARK);
|
|
|
|
while (rlist) {
|
|
|
|
direction_t d;
|
|
|
|
r = rlist->data;
|
|
|
|
++size;
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
region * rn = rconnect(r, d);
|
|
|
|
if (rn && !fval(rn, FL_MARK) && rn->land) {
|
2005-10-09 21:52:21 +02:00
|
|
|
region_list * rnew = malloc(sizeof(region_list));
|
|
|
|
rnew->data = rn;
|
|
|
|
rnew->next = rlist->next;
|
|
|
|
rlist->next = rnew;
|
|
|
|
fset(rn, FL_MARK);
|
2005-09-10 23:57:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
rlist = rlist->next;
|
|
|
|
}
|
|
|
|
for (rlist=island;rlist;rlist=rlist->next) {
|
|
|
|
freset(rlist->data, FL_MARK);
|
|
|
|
}
|
|
|
|
free_regionlist(island);
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2005-04-03 10:32:16 +02:00
|
|
|
/** create new island with up to nsize players
|
|
|
|
* returns the number of players placed on the new island.
|
|
|
|
*/
|
2002-05-25 08:58:54 +02:00
|
|
|
int
|
2005-08-07 15:48:18 +02:00
|
|
|
autoseed(newfaction ** players, int nsize, boolean new_island)
|
2002-04-11 19:37:06 +02:00
|
|
|
{
|
2005-08-07 15:48:18 +02:00
|
|
|
short x = 0, y = 0;
|
|
|
|
region * r = NULL;
|
|
|
|
region_list * rlist = NULL;
|
|
|
|
int rsize, tsize = 0;
|
2005-04-03 10:32:16 +02:00
|
|
|
int isize = REGIONS_PER_FACTION; /* target size for the island */
|
|
|
|
int psize = 0; /* players on this island */
|
2005-12-11 16:14:03 +01:00
|
|
|
const terrain_type * volcano_terrain = get_terrain("volcano");
|
2004-08-07 09:42:22 +02:00
|
|
|
|
2005-03-24 00:39:26 +01:00
|
|
|
if (listlen(*players)<MINFACTIONS) return 0;
|
2005-04-03 10:32:16 +02:00
|
|
|
|
2005-08-07 15:48:18 +02:00
|
|
|
if (!new_island) {
|
2005-04-03 10:32:16 +02:00
|
|
|
region * rmin = NULL;
|
|
|
|
/* find a spot that's adjacent to the previous island, but virgin.
|
|
|
|
* like the last land virgin ocean region adjacent to land.
|
|
|
|
*/
|
|
|
|
for (r=regions;r;r=r->next) {
|
|
|
|
struct plane * p = rplane(r);
|
2005-10-25 14:38:01 +02:00
|
|
|
if (rterrain(r)==T_OCEAN && p==NULL && virgin_region(r)) {
|
2005-04-03 10:32:16 +02:00
|
|
|
direction_t d;
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
region * rn = rconnect(r, d);
|
2005-04-08 21:47:10 +02:00
|
|
|
if (rn && rn->land) {
|
2005-09-10 23:57:02 +02:00
|
|
|
/* only expand islands that aren't single-islands and not too big already */
|
|
|
|
int n = island_size(rn);
|
|
|
|
if (n>2 && n<MAXISLANDSIZE) {
|
|
|
|
rmin = rn;
|
|
|
|
break;
|
|
|
|
}
|
2005-04-08 21:47:10 +02:00
|
|
|
}
|
2005-04-03 10:32:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (rmin!=NULL) {
|
|
|
|
region_list * rlist = NULL, * rptr;
|
|
|
|
faction * f;
|
2005-04-08 21:47:10 +02:00
|
|
|
get_island(rmin, &rlist);
|
2005-04-03 10:32:16 +02:00
|
|
|
for (rptr=rlist;rptr;rptr=rptr->next) {
|
|
|
|
region * r = rlist->data;
|
|
|
|
unit * u;
|
2005-04-03 10:40:13 +02:00
|
|
|
for (u=r->units;u;u=u->next) {
|
2005-04-03 10:32:16 +02:00
|
|
|
f = u->faction;
|
|
|
|
if (!fval(f, FL_MARK)) {
|
|
|
|
++psize;
|
|
|
|
fset(f, FL_MARK);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free_regionlist(rlist);
|
|
|
|
if (psize>0) for (f=factions;f;f=f->next) freset(f, FL_MARK);
|
|
|
|
if (psize<PLAYERS_PER_ISLAND) {
|
|
|
|
r = rmin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (r==NULL) {
|
|
|
|
region * rmin = NULL;
|
|
|
|
direction_t dmin = MAXDIRECTIONS;
|
|
|
|
/* find an empty spot.
|
|
|
|
* rmin = the youngest ocean region that has a missing neighbour
|
|
|
|
* dmin = direction in which it's empty
|
|
|
|
*/
|
|
|
|
for (r=regions;r;r=r->next) {
|
|
|
|
struct plane * p = rplane(r);
|
2005-10-25 14:38:01 +02:00
|
|
|
if (rterrain(r)==T_OCEAN && p==0 && (rmin==NULL || r->age<=MAXAGEDIFF)) {
|
2005-04-03 10:32:16 +02:00
|
|
|
direction_t d;
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
2005-10-16 22:11:34 +02:00
|
|
|
region * rn = rconnect(r, d);
|
|
|
|
if (rn==NULL) break;
|
2005-04-03 10:32:16 +02:00
|
|
|
}
|
|
|
|
if (d!=MAXDIRECTIONS) {
|
|
|
|
rmin=r;
|
|
|
|
dmin=d;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* create a new region where we found the empty spot, and make it the first
|
|
|
|
* in our island. island regions are kept in rlist, so only new regions can
|
|
|
|
* get populated, and old regions are not overwritten */
|
|
|
|
if (rmin!=NULL) {
|
|
|
|
assert(virgin_region(rconnect(rmin, dmin)));
|
|
|
|
x = rmin->x + delta_x[dmin];
|
|
|
|
y = rmin->y + delta_y[dmin];
|
|
|
|
}
|
|
|
|
r = new_region(x, y);
|
|
|
|
terraform(r, T_OCEAN); /* we change the terrain later */
|
|
|
|
}
|
|
|
|
|
|
|
|
add_regionlist(&rlist, r);
|
2005-05-01 19:46:33 +02:00
|
|
|
fset(r, FL_MARK);
|
2005-04-03 10:32:16 +02:00
|
|
|
rsize = 1;
|
|
|
|
|
|
|
|
while (rsize && (nsize || isize>=REGIONS_PER_FACTION)) {
|
2005-04-11 23:49:30 +02:00
|
|
|
int i = rand() % rsize;
|
|
|
|
region_list ** rnext = &rlist;
|
2005-05-01 19:46:33 +02:00
|
|
|
region_list * rfind;
|
2005-04-11 23:49:30 +02:00
|
|
|
direction_t d;
|
|
|
|
while (i--) rnext=&(*rnext)->next;
|
2005-05-01 19:46:33 +02:00
|
|
|
rfind = *rnext;
|
|
|
|
r = rfind->data;
|
|
|
|
freset(r, FL_MARK);
|
|
|
|
*rnext = rfind->next;
|
|
|
|
free(rfind);
|
2005-04-11 23:49:30 +02:00
|
|
|
--rsize;
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
region * rn = rconnect(r, d);
|
2005-05-01 23:29:55 +02:00
|
|
|
if (rn && fval(rn, FL_MARK)) continue;
|
2005-04-11 23:49:30 +02:00
|
|
|
if (virgin_region(rn)) {
|
2005-04-03 10:32:16 +02:00
|
|
|
if (rn==NULL) {
|
|
|
|
rn = new_region(r->x + delta_x[d], r->y + delta_y[d]);
|
|
|
|
terraform(rn, T_OCEAN);
|
|
|
|
}
|
2005-04-11 23:49:30 +02:00
|
|
|
add_regionlist(&rlist, rn);
|
2005-05-01 19:46:33 +02:00
|
|
|
fset(rn, FL_MARK);
|
2005-04-11 23:49:30 +02:00
|
|
|
++rsize;
|
|
|
|
}
|
|
|
|
}
|
2005-12-11 16:14:03 +01:00
|
|
|
if (volcano_terrain!=NULL && (rand() % VOLCANO_CHANCE == 0)) {
|
|
|
|
terraform_region(r, volcano_terrain);
|
2005-04-03 10:32:16 +02:00
|
|
|
} else if (nsize && (rand() % isize == 0 || rsize==0)) {
|
2005-04-11 23:49:30 +02:00
|
|
|
newfaction ** nfp, * nextf = *players;
|
2005-04-16 15:03:35 +02:00
|
|
|
faction * f;
|
2005-04-11 23:49:30 +02:00
|
|
|
unit * u;
|
2005-09-10 23:09:51 +02:00
|
|
|
int n;
|
2005-04-03 10:32:16 +02:00
|
|
|
|
|
|
|
isize += REGIONS_PER_FACTION;
|
2005-04-11 23:49:30 +02:00
|
|
|
terraform(r, preferred_terrain(nextf->race));
|
2005-09-10 23:09:51 +02:00
|
|
|
n = rhorses(r);
|
2005-10-25 14:38:01 +02:00
|
|
|
if (n<r->terrain->size/200) {
|
|
|
|
n = r->terrain->size/200;
|
2005-09-10 23:09:51 +02:00
|
|
|
rsethorses(r, n);
|
|
|
|
}
|
|
|
|
n = rtrees(r, 2);
|
2005-10-25 14:38:01 +02:00
|
|
|
if (n<r->terrain->size/100) {
|
|
|
|
n = r->terrain->size/100;
|
2005-09-10 23:09:51 +02:00
|
|
|
rsettrees(r, 2, n);
|
|
|
|
rsettrees(r, 2, n/4);
|
|
|
|
rsettrees(r, 2, n/2);
|
|
|
|
}
|
|
|
|
n = rhorses(r);
|
2005-04-11 23:49:30 +02:00
|
|
|
++tsize;
|
2005-04-16 10:52:36 +02:00
|
|
|
assert(r->land && r->units==0);
|
2005-04-11 23:49:30 +02:00
|
|
|
u = addplayer(r, addfaction(nextf->email, nextf->password, nextf->race,
|
|
|
|
nextf->lang, nextf->subscription));
|
2005-04-16 15:03:35 +02:00
|
|
|
f = u->faction;
|
2005-04-16 18:44:00 +02:00
|
|
|
f->alliance = nextf->allies;
|
|
|
|
log_printf("New faction (%s), %s at %s\n", itoa36(f->no),
|
|
|
|
f->email, regionname(r, NULL));
|
2005-04-16 15:03:35 +02:00
|
|
|
if (f->subscription) {
|
2005-04-16 18:44:00 +02:00
|
|
|
sql_print(("UPDATE subscriptions SET status='ACTIVE', faction='%s', lastturn=%d, password='%s' WHERE id=%u;\n",
|
|
|
|
factionid(f), f->lastorders, f->override, f->subscription));
|
2005-04-16 15:03:35 +02:00
|
|
|
}
|
2005-04-11 23:49:30 +02:00
|
|
|
|
|
|
|
/* remove duplicate email addresses */
|
|
|
|
nfp = players;
|
|
|
|
while (*nfp) {
|
|
|
|
newfaction * nf = *nfp;
|
|
|
|
if (strcmp(nextf->email, nf->email)==0) {
|
|
|
|
*nfp = nf->next;
|
|
|
|
if (nextf!=nf) free(nf);
|
|
|
|
}
|
|
|
|
else nfp = &nf->next;
|
|
|
|
}
|
2005-04-03 10:32:16 +02:00
|
|
|
++psize;
|
2005-04-11 23:49:30 +02:00
|
|
|
--nsize;
|
2005-04-03 10:32:16 +02:00
|
|
|
--isize;
|
|
|
|
if (psize>=PLAYERS_PER_ISLAND) break;
|
2005-04-11 23:49:30 +02:00
|
|
|
} else {
|
2005-12-31 14:01:11 +01:00
|
|
|
terraform_region(r, random_terrain(AUTO_TERRAIN));
|
2005-04-03 10:32:16 +02:00
|
|
|
--isize;
|
2005-04-11 23:49:30 +02:00
|
|
|
}
|
|
|
|
}
|
2005-04-03 10:32:16 +02:00
|
|
|
|
2005-04-11 23:49:30 +02:00
|
|
|
if (r!=NULL) {
|
|
|
|
/* reicht das? */
|
|
|
|
fix_demand(r);
|
|
|
|
}
|
|
|
|
|
2005-04-03 10:32:16 +02:00
|
|
|
if (nsize!=0) {
|
|
|
|
log_error(("Could not place all factions on the same island as requested\n"));
|
|
|
|
}
|
2005-04-11 23:49:30 +02:00
|
|
|
|
2005-04-03 10:32:16 +02:00
|
|
|
|
|
|
|
if (rlist) {
|
2002-05-25 08:58:54 +02:00
|
|
|
#define MINOCEANDIST 3
|
2002-06-02 08:23:29 +02:00
|
|
|
#define MAXFILLDIST 10
|
2004-09-05 15:02:09 +02:00
|
|
|
#define SPECIALCHANCE 80
|
2005-04-11 23:49:30 +02:00
|
|
|
region_list ** rbegin = &rlist;
|
|
|
|
int i;
|
2004-09-05 15:02:09 +02:00
|
|
|
int special = 1;
|
2005-04-11 23:49:30 +02:00
|
|
|
|
2004-09-05 15:02:09 +02:00
|
|
|
for (i=0;i!=MINOCEANDIST;++i) {
|
2005-04-11 23:49:30 +02:00
|
|
|
region_list ** rend = rbegin;
|
|
|
|
while (*rend) rend=&(*rend)->next;
|
|
|
|
while (rbegin!=rend) {
|
|
|
|
direction_t d;
|
|
|
|
region * r = (*rbegin)->data;
|
|
|
|
rbegin=&(*rbegin)->next;
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) {
|
|
|
|
region * rn = rconnect(r, d);
|
|
|
|
if (rn==NULL) {
|
2005-11-26 01:42:32 +01:00
|
|
|
const struct terrain_type * terrain = newterrain(T_OCEAN);
|
2005-04-11 23:49:30 +02:00
|
|
|
rn = new_region(r->x + delta_x[d], r->y + delta_y[d]);
|
2004-09-05 15:02:09 +02:00
|
|
|
if (rand() % SPECIALCHANCE < special) {
|
2005-12-31 14:01:11 +01:00
|
|
|
terrain = random_terrain(AUTO_TERRAIN);
|
2004-09-05 15:02:09 +02:00
|
|
|
special = SPECIALCHANCE / 3; /* 33% chance auf noch eines */
|
|
|
|
} else {
|
|
|
|
special = 1;
|
|
|
|
}
|
2005-11-26 01:42:32 +01:00
|
|
|
terraform_region(rn, terrain);
|
2004-10-14 09:10:03 +02:00
|
|
|
/* the new region has an extra 15% chance to have laen */
|
|
|
|
if (rand() % 100 < 15) rsetlaen(r, 5 + rand() % 5);
|
|
|
|
/* the new region has an extra 20% chance to have mallorn */
|
|
|
|
if (rand() % 100 < 20) fset(r, RF_MALLORN);
|
2005-04-11 23:49:30 +02:00
|
|
|
add_regionlist(rend, rn);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
while (*rbegin) {
|
|
|
|
region * r = (*rbegin)->data;
|
|
|
|
direction_t d;
|
|
|
|
rbegin=&(*rbegin)->next;
|
|
|
|
for (d=0;d!=MAXDIRECTIONS;++d) if (rconnect(r, d)==NULL) {
|
2005-06-10 00:10:35 +02:00
|
|
|
short i;
|
2005-04-11 23:49:30 +02:00
|
|
|
for (i=1;i!=MAXFILLDIST;++i) {
|
|
|
|
if (findregion(r->x + i*delta_x[d], r->y + i*delta_y[d]))
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
if (i!=MAXFILLDIST) {
|
|
|
|
while (--i) {
|
|
|
|
region * rn = new_region(r->x + i*delta_x[d], r->y + i*delta_y[d]);
|
|
|
|
terraform(rn, T_OCEAN);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-07-26 18:37:07 +02:00
|
|
|
while (rlist) {
|
|
|
|
region_list * self = rlist;
|
|
|
|
rlist = rlist->next;
|
|
|
|
freset(self->data, FL_MARK);
|
|
|
|
free(self);
|
|
|
|
}
|
2005-04-11 23:49:30 +02:00
|
|
|
}
|
|
|
|
return tsize;
|
2002-05-25 08:58:54 +02:00
|
|
|
}
|