forked from github/server
d633c2a9fb
reduce number of count_all loops made, make f->num_units and f->num_people be correct.
241 lines
7.3 KiB
C
241 lines
7.3 KiB
C
/*
|
|
Copyright (c) 1998-2015, Enno Rehling <enno@eressea.de>
|
|
Katja Zedel <katze@felidae.kn-bremen.de
|
|
Christian Schlittchen <corwin@amber.kn-bremen.de>
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any
|
|
purpose with or without fee is hereby granted, provided that the above
|
|
copyright notice and this permission notice appear in all copies.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
**/
|
|
|
|
#include <platform.h>
|
|
#include <kernel/config.h>
|
|
#include "score.h"
|
|
|
|
/* kernel includes */
|
|
#include <kernel/alliance.h>
|
|
#include <kernel/building.h>
|
|
#include <kernel/build.h>
|
|
#include <kernel/faction.h>
|
|
#include <kernel/item.h>
|
|
#include <kernel/race.h>
|
|
#include <kernel/region.h>
|
|
#include <kernel/ship.h>
|
|
#include <kernel/unit.h>
|
|
#include <kernel/pool.h>
|
|
|
|
/* util includes */
|
|
#include <util/base36.h>
|
|
#include <util/bsdstring.h>
|
|
#include <util/language.h>
|
|
|
|
/* libc includes */
|
|
#include <assert.h>
|
|
#include <math.h>
|
|
#include <stdio.h>
|
|
|
|
score_t average_score_of_age(int age, int a)
|
|
{
|
|
faction *f;
|
|
score_t sum = 0, count = 0;
|
|
|
|
for (f = factions; f; f = f->next) {
|
|
if (!fval(f, FFL_NPC) && f->age <= age + a
|
|
&& f->age >= age - a && f->race != get_race(RC_TEMPLATE)) {
|
|
sum += f->score;
|
|
count++;
|
|
}
|
|
}
|
|
|
|
if (count == 0) {
|
|
return 0;
|
|
}
|
|
return sum / count;
|
|
}
|
|
|
|
void score(void)
|
|
{
|
|
FILE *scoreFP;
|
|
region *r;
|
|
faction *fc;
|
|
score_t allscores = 0;
|
|
char path[4096];
|
|
|
|
for (fc = factions; fc; fc = fc->next)
|
|
fc->score = 0;
|
|
|
|
for (r = regions; r; r = r->next) {
|
|
unit *u;
|
|
building *b;
|
|
ship *s;
|
|
|
|
if (rule_region_owners()) {
|
|
fc = region_get_owner(r);
|
|
if (fc)
|
|
fc->score += 10;
|
|
}
|
|
|
|
for (b = r->buildings; b; b = b->next) {
|
|
u = building_owner(b);
|
|
if (u != NULL) {
|
|
faction *fbo = u->faction;
|
|
|
|
if (fbo) {
|
|
fbo->score += b->size * 5;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (s = r->ships; s; s = s->next) {
|
|
unit *cap = ship_owner(s);
|
|
if (cap && cap->faction) {
|
|
cap->faction->score += s->size * 2;
|
|
}
|
|
}
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
item *itm;
|
|
int itemscore = 0;
|
|
int i;
|
|
faction *f = u->faction;
|
|
const race *rc = u_race(u);
|
|
|
|
if (f == NULL) {
|
|
continue;
|
|
}
|
|
else if (rc->recruitcost>0) {
|
|
assert(playerrace(rc));
|
|
f->score += (rc->recruitcost * u->number) / 50;
|
|
}
|
|
f->score += get_money(u) / 50;
|
|
for (itm = u->items; itm; itm = itm->next) {
|
|
itemscore += itm->number * itm->type->score;
|
|
}
|
|
f->score += itemscore / 10;
|
|
|
|
for (i = 0; i != u->skill_size; ++i) {
|
|
skill *sv = u->skills + i;
|
|
switch (sv->id) {
|
|
case SK_MAGIC:
|
|
f->score += (score_t)(u->number * pow(sv->level, 4));
|
|
break;
|
|
case SK_TACTICS:
|
|
f->score += (score_t)(u->number * pow(sv->level, 3));
|
|
break;
|
|
case SK_SPY:
|
|
case SK_ALCHEMY:
|
|
case SK_HERBALISM:
|
|
f->score += (score_t)(u->number * pow(sv->level, 2.5));
|
|
break;
|
|
default:
|
|
f->score += (score_t)(u->number * pow(sv->level, 2.5) / 10);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (fc = factions; fc; fc = fc->next) {
|
|
fc->score = fc->score / 5;
|
|
if (!fval(fc, FFL_NPC) && fc->race != get_race(RC_TEMPLATE)) {
|
|
allscores += fc->score;
|
|
}
|
|
}
|
|
if (allscores == 0) {
|
|
allscores = 1;
|
|
}
|
|
|
|
join_path(basepath(), "score", path, sizeof(path));
|
|
scoreFP = fopen(path, "w");
|
|
if (scoreFP) {
|
|
const unsigned char utf8_bom[4] = { 0xef, 0xbb, 0xbf, 0 };
|
|
faction *f;
|
|
fwrite(utf8_bom, 1, 3, scoreFP);
|
|
for (f = factions; f; f = f->next)
|
|
if (!fval(f, FFL_NPC) && f->num_people != 0) {
|
|
char score[32];
|
|
write_score(score, sizeof(score), f->score);
|
|
fprintf(scoreFP, "%s ", score);
|
|
write_score(score, sizeof(score), average_score_of_age(f->age, f->age / 24 + 1));
|
|
fprintf(scoreFP, "(%s) ", score);
|
|
fprintf(scoreFP, "%30.30s (%3.3s) %5s (%3d)\n",
|
|
f->name,
|
|
f->race->_name,
|
|
itoa36(f->no),
|
|
f->age);
|
|
}
|
|
fclose(scoreFP);
|
|
}
|
|
if (alliances != NULL) {
|
|
alliance *a;
|
|
const item_type *token = it_find("conquesttoken");
|
|
|
|
join_path(basepath(), "score.alliances", path, sizeof(path));
|
|
scoreFP = fopen(path, "w");
|
|
if (scoreFP) {
|
|
const unsigned char utf8_bom[4] = { 0xef, 0xbb, 0xbf, 0 };
|
|
fwrite(utf8_bom, 1, 3, scoreFP);
|
|
|
|
fprintf(scoreFP, "# alliance:factions:persons:score\n");
|
|
|
|
for (a = alliances; a; a = a->next) {
|
|
score_t alliance_score = 0;
|
|
int alliance_number = 0, alliance_factions = 0, grails = 0;
|
|
faction *f;
|
|
|
|
for (f = factions; f; f = f->next) {
|
|
if (f->alliance && f->alliance == a) {
|
|
alliance_factions++;
|
|
alliance_score += f->score;
|
|
alliance_number += f->num_people;
|
|
if (token != NULL) {
|
|
unit *u = f->units;
|
|
while (u != NULL) {
|
|
item **iitem = i_find(&u->items, token);
|
|
if (iitem != NULL && *iitem != NULL) {
|
|
grails += (*iitem)->number;
|
|
}
|
|
u = u->nextF;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fprintf(scoreFP, "%d:%d:%d:%lld", a->id, alliance_factions,
|
|
alliance_number, alliance_score);
|
|
if (token != NULL)
|
|
fprintf(scoreFP, ":%d", grails);
|
|
fputc('\n', scoreFP);
|
|
}
|
|
fclose(scoreFP);
|
|
}
|
|
}
|
|
}
|
|
|
|
int default_score(const item_type *itype) {
|
|
int result = 0;
|
|
if (itype->construction) {
|
|
requirement *req = itype->construction->materials;
|
|
while (req->number) {
|
|
int score = req->rtype->itype ? req->rtype->itype->score : 10;
|
|
result += score * req->number * 2;
|
|
++req;
|
|
}
|
|
}
|
|
else {
|
|
result = 10;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void write_score(char *buffer, size_t size, score_t score) {
|
|
slprintf(buffer, size, "%lld", score);
|
|
}
|