2015-01-30 20:37:14 +01:00
|
|
|
/*
|
2010-08-08 10:06:34 +02:00
|
|
|
+-------------------+ Christian Schlittchen <corwin@amber.kn-bremen.de>
|
|
|
|
| | Enno Rehling <enno@eressea.de>
|
|
|
|
| Eressea PBEM host | Katja Zedel <katze@felidae.kn-bremen.de>
|
|
|
|
| (c) 1998 - 2003 | Henning Peters <faroul@beyond.kn-bremen.de>
|
|
|
|
| | Ingo Wilken <Ingo.Wilken@informatik.uni-oldenburg.de>
|
|
|
|
+-------------------+ Stefan Reich <reich@halbling.de>
|
|
|
|
|
|
|
|
This program may not be used, modified or distributed
|
|
|
|
without prior permission by the authors of Eressea.
|
2015-01-30 20:37:14 +01:00
|
|
|
*/
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
#include <platform.h>
|
|
|
|
#include <kernel/config.h>
|
|
|
|
#include "resources.h"
|
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
#include "build.h"
|
|
|
|
#include "item.h"
|
|
|
|
#include "region.h"
|
|
|
|
#include "terrain.h"
|
|
|
|
|
|
|
|
#include <util/rand.h>
|
|
|
|
#include <util/rng.h>
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static double ResourceFactor(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-11-22 17:32:35 +01:00
|
|
|
return config_get_flt("resource.factor", 1.0);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
void update_resources(region * r)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
struct rawmaterial *res = r->resources;
|
|
|
|
while (res) {
|
|
|
|
if (res->type->update)
|
|
|
|
res->type->update(res, r);
|
|
|
|
res = res->next;
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
extern int dice_rand(const char *s);
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static void update_resource(struct rawmaterial *res, double modifier)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2016-11-20 19:08:39 +01:00
|
|
|
double amount = (res->level - res->startlevel) / 100.0 * res->divisor + 1;
|
2015-01-30 20:37:14 +01:00
|
|
|
amount = ResourceFactor() * res->base * amount * modifier;
|
|
|
|
if (amount < 1.0)
|
|
|
|
res->amount = 1;
|
|
|
|
else
|
|
|
|
res->amount = (int)amount;
|
|
|
|
assert(res->amount > 0);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2011-03-07 08:02:35 +01:00
|
|
|
add_resource(region * r, int level, int base, int divisor,
|
2015-01-30 20:37:14 +01:00
|
|
|
const resource_type * rtype)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
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);
|
2017-02-26 13:19:47 +01:00
|
|
|
assert(rm->type);
|
2015-01-30 20:37:14 +01:00
|
|
|
update_resource(rm, 1.0);
|
|
|
|
rm->type->terraform(rm, r);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
void terraform_resources(region * r)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
int i;
|
|
|
|
const terrain_type *terrain = r->terrain;
|
2015-11-22 16:14:27 +01:00
|
|
|
bool terraform_all = config_get_int("rules.terraform.all", 0) != 0;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static void terraform_default(struct rawmaterial *res, const region * r)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
|
|
|
#define SHIFT 70
|
2015-01-30 20:37:14 +01:00
|
|
|
double modifier =
|
|
|
|
1.0 + ((rng_int() % (SHIFT * 2 + 1)) - SHIFT) * ((rng_int() % (SHIFT * 2 +
|
2011-03-07 08:02:35 +01:00
|
|
|
1)) - SHIFT) / 10000.0;
|
2015-01-30 20:37:14 +01:00
|
|
|
res->amount = (int)(res->amount * modifier); /* random adjustment, +/- 91% */
|
|
|
|
if (res->amount < 1)
|
|
|
|
res->amount = 1;
|
2017-01-10 16:31:05 +01:00
|
|
|
UNUSED_ARG(r);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef RANDOM_CHANGE
|
2012-06-24 07:41:07 +02:00
|
|
|
static void resource_random_change(int *pvalue, bool used)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
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;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int visible_default(const rawmaterial * res, int skilllevel)
|
2010-08-08 10:06:34 +02:00
|
|
|
/* resources are visible, if skill equals minimum skill to mine them
|
|
|
|
* plus current level of difficulty */
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
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;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static void use_default(rawmaterial * res, const region * r, int amount)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
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);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
struct rawmaterial *rm_get(region * r, const struct resource_type *rtype)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
struct rawmaterial *rm = r->resources;
|
|
|
|
while (rm && rm->type->rtype != rtype)
|
|
|
|
rm = rm->next;
|
|
|
|
return rm;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
struct rawmaterial_type *rmt_find(const char *str)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2017-02-23 07:53:28 +01:00
|
|
|
resource_type *rtype = rt_find(str);
|
|
|
|
if (!rtype && strncmp(str, "rm_", 3) == 0) {
|
2017-02-23 08:07:47 +01:00
|
|
|
rtype = rt_find(str+3);
|
2017-02-23 07:53:28 +01:00
|
|
|
}
|
2017-02-23 08:02:24 +01:00
|
|
|
assert(rtype);
|
2017-02-23 07:53:28 +01:00
|
|
|
return rtype ? rtype->raw : NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
struct rawmaterial_type *rmt_get(const struct resource_type *rtype)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2017-02-22 21:10:22 +01:00
|
|
|
return rtype->raw;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2017-02-23 08:02:24 +01:00
|
|
|
struct rawmaterial_type *rmt_create(struct resource_type *rtype)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2017-02-22 21:10:22 +01:00
|
|
|
rawmaterial_type *rmtype;
|
|
|
|
|
|
|
|
assert(!rtype->raw);
|
2017-03-05 17:18:17 +01:00
|
|
|
assert(!rtype->itype || rtype->itype->construction);
|
2017-02-22 21:10:22 +01:00
|
|
|
rmtype = rtype->raw = malloc(sizeof(rawmaterial_type));
|
2015-01-30 20:37:14 +01:00
|
|
|
rmtype->rtype = rtype;
|
|
|
|
rmtype->terraform = terraform_default;
|
|
|
|
rmtype->update = NULL;
|
|
|
|
rmtype->use = use_default;
|
|
|
|
rmtype->visible = visible_default;
|
|
|
|
return rmtype;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2017-02-26 13:19:47 +01:00
|
|
|
|
2017-02-27 04:22:28 +01:00
|
|
|
int(*item_use_fun)(struct unit *u, const struct item_type *itype, int amount,
|
|
|
|
struct order *ord);
|
2017-02-26 13:19:47 +01:00
|
|
|
int(*res_limit_fun)(const struct region *, const struct resource_type *);
|
|
|
|
void(*res_produce_fun)(struct region *, const struct resource_type *, int);
|
|
|
|
|
|
|
|
int limit_resource(const struct region *r, const resource_type *rtype)
|
|
|
|
{
|
|
|
|
assert(!rtype->raw);
|
|
|
|
if (res_limit_fun) {
|
|
|
|
return res_limit_fun(r, rtype);
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void produce_resource(struct region *r, const struct resource_type *rtype, int amount)
|
|
|
|
{
|
|
|
|
assert(!rtype->raw);
|
|
|
|
if (res_produce_fun) {
|
|
|
|
res_produce_fun(r, rtype, amount);
|
|
|
|
}
|
|
|
|
}
|