/* +-------------------+ Christian Schlittchen | | Enno Rehling | Eressea PBEM host | Katja Zedel | (c) 1998 - 2003 | 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 "resources.h" /* kernel includes */ #include "build.h" #include "item.h" #include "region.h" #include "terrain.h" #include #include #include #include #include static double ResourceFactor(void) { return config_get_flt("resource.factor", 1.0); } void update_resources(region * r) { struct rawmaterial *res = r->resources; while (res) { if (res->type->update) res->type->update(res, r); res = res->next; } } extern int dice_rand(const char *s); static void update_resource(struct rawmaterial *res, double modifier) { double amount = (res->level - res->startlevel) / 100.0 * res->divisor + 1; amount = ResourceFactor() * res->base * amount * modifier; if (amount < 1.0) res->amount = 1; else res->amount = (int)amount; assert(res->amount > 0); } void add_resource(region * r, int level, int base, int divisor, const resource_type * rtype) { struct rawmaterial *rm = calloc(sizeof(struct rawmaterial), 1); rm->next = r->resources; r->resources = rm; rm->level = level; rm->startlevel = level; rm->base = base; rm->divisor = divisor; rm->flags = 0; rm->type = rmt_get(rtype); update_resource(rm, 1.0); rm->type->terraform(rm, r); } void terraform_resources(region * r) { int i; const terrain_type *terrain = r->terrain; bool terraform_all = config_get_int("rules.terraform.all", 0) != 0; if (terrain->production == NULL) return; for (i = 0; terrain->production[i].type; ++i) { rawmaterial *rm; const terrain_production *production = terrain->production + i; const resource_type *rtype = production->type; for (rm = r->resources; rm; rm = rm->next) { if (rm->type->rtype == rtype) break; } if (rm) { continue; } if (terraform_all || chance(production->chance)) { add_resource(r, dice_rand(production->startlevel), dice_rand(production->base), dice_rand(production->divisor), production->type); } } } static void terraform_default(struct rawmaterial *res, const region * r) { #define SHIFT 70 double modifier = 1.0 + ((rng_int() % (SHIFT * 2 + 1)) - SHIFT) * ((rng_int() % (SHIFT * 2 + 1)) - SHIFT) / 10000.0; res->amount = (int)(res->amount * modifier); /* random adjustment, +/- 91% */ if (res->amount < 1) res->amount = 1; UNUSED_ARG(r); } #ifdef RANDOM_CHANGE static void resource_random_change(int *pvalue, bool used) { int split = 5; int rnd = rng_int() % 100; if (pvalue == 0 || rnd % 10 >= 10) return; if (used) split = 4; /* if a resource was mined this round, there is a 6% probability * of a decline and a 4% probability of a raise. */ /* if it wasn't mined this round, there is an equal probability * of 5% for a decline or a raise. */ if (rnd < split) { (*pvalue)++; } else { (*pvalue)--; } if ((*pvalue) < 0) (*pvalue) = 0; } #endif static int visible_default(const rawmaterial * res, int skilllevel) /* resources are visible, if skill equals minimum skill to mine them * plus current level of difficulty */ { const struct item_type *itype = res->type->rtype->itype; if (res->level <= 1 && res->level + itype->construction->minskill <= skilllevel + 1) { assert(res->amount > 0); return res->amount; } else if (res->level + itype->construction->minskill <= skilllevel + 2) { assert(res->amount > 0); return res->amount; } return -1; } static void use_default(rawmaterial * res, const region * r, int amount) { assert(res->amount > 0 && amount >= 0 && amount <= res->amount); res->amount -= amount; while (res->amount == 0) { double modifier = 1.0 + ((rng_int() % (SHIFT * 2 + 1)) - SHIFT) * ((rng_int() % (SHIFT * 2 + 1)) - SHIFT) / 10000.0; int i; for (i = 0; r->terrain->production[i].type; ++i) { if (res->type->rtype == r->terrain->production[i].type) break; } ++res->level; update_resource(res, modifier); } } struct rawmaterial *rm_get(region * r, const struct resource_type *rtype) { struct rawmaterial *rm = r->resources; while (rm && rm->type->rtype != rtype) rm = rm->next; return rm; } struct rawmaterial_type *rmt_find(const char *str) { const char * replace[] = { "rm_tree", "log", NULL }; resource_type *rtype = rt_find(str); if (!rtype && strncmp(str, "rm_", 3) == 0) { int i; for (i = 0; replace[i]; i+=2) { if (strcmp(replace[i], str) == 0) { rtype = rt_find(replace[i+1]); break; } } if (!rtype) { rtype = rt_find(str+3); } } assert(rtype); return rtype ? rtype->raw : NULL; } struct rawmaterial_type *rmt_get(const struct resource_type *rtype) { return rtype->raw; } struct rawmaterial_type *rmt_create(struct resource_type *rtype) { rawmaterial_type *rmtype; assert(!rtype->raw); rmtype = rtype->raw = malloc(sizeof(rawmaterial_type)); rmtype->rtype = rtype; rmtype->terraform = terraform_default; rmtype->update = NULL; rmtype->use = use_default; rmtype->visible = visible_default; return rmtype; }