/* vi: set ts=2: +-------------------+ Christian Schlittchen | | Enno Rehling | Eressea PBEM host | Katja Zedel | (c) 1998 - 2001 | Henning Peters | | Ingo Wilken +-------------------+ Stefan Reich This program may not be used, modified or distributed without prior permission by the authors of Eressea. */ #include #include #include "autoseed.h" /* kernel includes */ #include #include #include /* libc includes */ #include #include #include newfaction * newfactions = NULL; static int regions_per_faction = 3; void get_island(regionlist ** rlist) { regionlist *rcurrent = *rlist; regionlist **rnext = &rcurrent->next; assert(rcurrent && *rnext==NULL); fset(rcurrent->region, FL_MARK); while (rcurrent!=NULL) { direction_t dir; for (dir=0;dir!=MAXDIRECTIONS;++dir) { region * r = rconnect(rcurrent->region, dir); if (r!=NULL && r->land && !fval(r, FL_MARK)) { fset(r, FL_MARK); add_regionlist(rnext, r); } } rcurrent = *rnext; rnext = &rcurrent->next; } rnext=rlist; while (*rnext) { rcurrent = *rnext; freset(rcurrent->region, FL_MARK); rnext = &rcurrent->next; } } typedef struct seed_t { struct region * region; struct newfaction * player; struct seed_t * next[MAXDIRECTIONS]; } seed_t; double 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; } double get_quality(struct seed_t * seed) { double q = 0.0; int nterrains[MAXTERRAINS]; direction_t d; memset(nterrains, 0, sizeof(nterrains)); for (d=0;d!=MAXDIRECTIONS;++d) { seed_t * ns = seed->next[d]; if (ns) ++nterrains[rterrain(ns->region)]; } ++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]) { /* +10 points for the first if we are insects, and -2 points for every * 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; } double 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; } } /* q = q / nplayers; for (i=0;i!=nplayers;++i) { quality -= (qarr[i]-q)*(qarr[i]-q); } */ return quality + q; } void autoseed(struct regionlist * rlist) { int nregions = listlen(rlist); int nseeds = nregions; double lastq; seed_t * seeds = calloc(sizeof(seed_t), nseeds); int i, nfactions = 0; newfaction * nf = newfactions; while (nf) { if (nf->bonus==0) ++nfactions; nf = nf->next; } nfactions = min(nfactions, nregions/regions_per_faction); nf = newfactions; for (i=0;i!=nseeds;++i) { seeds[i].region = rlist->region; rlist = rlist->next; if (ibonus!=0) nf=nf->next; seeds[i].player = nf; nf = nf->next; } } for (i=0;i!=nseeds;++i) { direction_t d; for (d=0;d!=MAXDIRECTIONS;++d) { region * r = rconnect(seeds[i].region, d); if (r && r->land) { int k; for (k=i+1;k!=nseeds;++k) if (seeds[k].region==r) { seeds[k].next[back[d]] = seeds+i; seeds[i].next[d] = seeds+k; } } } } lastq = recalc(seeds, nseeds, nfactions); for (;;) { double quality = lastq; int i; for (i=0;i!=nseeds;++i) { int k; for (k=i+1;k!=nseeds;++k) { double q; newfaction * player = seeds[k].player; if (player==NULL && seeds[i].player==NULL) continue; seeds[k].player = seeds[i].player; seeds[i].player = player; q = recalc(seeds, nseeds, nfactions); if (q>quality) { quality = q; } else { /* undo */ seeds[i].player = seeds[k].player; seeds[k].player = player; } } } if (lastq==quality) break; else lastq = quality; } for (i=0;i!=nseeds;++i) { newfaction * nf = seeds[i].player; if (nf) { newfaction ** nfp = &newfactions; while (*nfp!=nf) nfp=&(*nfp)->next; addplayer(seeds[i].region, nf->email, nf->race, nf->lang); *nfp = nf->next; free(nf); } } }