2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
2003-09-21 10:47:28 +02:00
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* Eressea PB(E)M host Copyright (C) 1998-2003
|
|
|
|
|
* Christian Schlittchen (corwin@amber.kn-bremen.de)
|
|
|
|
|
* Katja Zedel (katze@felidae.kn-bremen.de)
|
|
|
|
|
* Henning Peters (faroul@beyond.kn-bremen.de)
|
2007-09-02 20:11:17 +02:00
|
|
|
|
* Enno Rehling (enno@eressea.de)
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
|
|
|
|
|
*
|
|
|
|
|
* based on:
|
|
|
|
|
*
|
|
|
|
|
* Atlantis v1.0 13 September 1993 Copyright 1993 by Russell Wallace
|
|
|
|
|
* Atlantis v1.7 Copyright 1996 by Alex Schr<EFBFBD>der
|
|
|
|
|
*
|
|
|
|
|
* This program may not be used, modified or distributed without
|
|
|
|
|
* prior permission by the authors of Eressea.
|
|
|
|
|
* This program may not be sold or used commercially without prior written
|
|
|
|
|
* permission from the authors.
|
|
|
|
|
*/
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include "eressea.h"
|
|
|
|
|
#include "economy.h"
|
|
|
|
|
|
2001-02-10 11:40:12 +01:00
|
|
|
|
/* gamecode includes */
|
2007-07-17 17:33:48 +02:00
|
|
|
|
#include "archetype.h"
|
2007-05-30 03:21:09 +02:00
|
|
|
|
#include "give.h"
|
2001-02-10 11:40:12 +01:00
|
|
|
|
#include "laws.h"
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include "randenc.h"
|
2007-07-17 17:33:48 +02:00
|
|
|
|
#include "spy.h"
|
2004-06-21 18:45:27 +02:00
|
|
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
|
#include <kernel/alchemy.h>
|
|
|
|
|
#include <kernel/building.h>
|
2005-06-04 15:22:31 +02:00
|
|
|
|
#include <kernel/calendar.h>
|
2007-05-28 18:03:48 +02:00
|
|
|
|
#include <kernel/equipment.h>
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include <kernel/faction.h>
|
|
|
|
|
#include <kernel/item.h>
|
|
|
|
|
#include <kernel/karma.h>
|
|
|
|
|
#include <kernel/magic.h>
|
|
|
|
|
#include <kernel/message.h>
|
2007-08-05 14:26:46 +02:00
|
|
|
|
#include <kernel/move.h>
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include <kernel/order.h>
|
|
|
|
|
#include <kernel/plane.h>
|
|
|
|
|
#include <kernel/pool.h>
|
|
|
|
|
#include <kernel/race.h>
|
|
|
|
|
#include <kernel/region.h>
|
|
|
|
|
#include <kernel/reports.h>
|
|
|
|
|
#include <kernel/resources.h>
|
|
|
|
|
#include <kernel/ship.h>
|
|
|
|
|
#include <kernel/skill.h>
|
2005-10-25 14:38:01 +02:00
|
|
|
|
#include <kernel/terrain.h>
|
|
|
|
|
#include <kernel/terrainid.h>
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include <kernel/unit.h>
|
2001-02-10 11:40:12 +01:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* util includes */
|
2004-06-21 18:45:27 +02:00
|
|
|
|
#include <util/attrib.h>
|
|
|
|
|
#include <util/base36.h>
|
|
|
|
|
#include <util/event.h>
|
2004-09-30 00:17:49 +02:00
|
|
|
|
#include <util/goodies.h>
|
2007-06-20 02:34:02 +02:00
|
|
|
|
#include <util/lists.h>
|
2001-05-20 23:47:56 +02:00
|
|
|
|
#include <util/message.h>
|
2007-06-22 00:31:28 +02:00
|
|
|
|
#include <util/parser.h>
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <util/rng.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
#include <attributes/reduceproduction.h>
|
|
|
|
|
#include <attributes/racename.h>
|
|
|
|
|
#include <attributes/orcification.h>
|
|
|
|
|
|
|
|
|
|
#include <items/seed.h>
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* libs includes */
|
|
|
|
|
#include <math.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <limits.h>
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2004-03-28 22:53:47 +02:00
|
|
|
|
typedef struct request {
|
|
|
|
|
struct request * next;
|
2006-10-29 14:11:41 +01:00
|
|
|
|
struct unit * unit;
|
|
|
|
|
struct order * ord;
|
2004-03-28 22:53:47 +02:00
|
|
|
|
int qty;
|
|
|
|
|
int no;
|
|
|
|
|
union {
|
|
|
|
|
boolean goblin; /* stealing */
|
|
|
|
|
const struct luxury_type * ltype; /* trading */
|
|
|
|
|
} type;
|
|
|
|
|
} request;
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
static request workers[1024];
|
|
|
|
|
static request *nextworker;
|
|
|
|
|
static int working;
|
|
|
|
|
|
|
|
|
|
static request entertainers[1024];
|
|
|
|
|
static request *nextentertainer;
|
|
|
|
|
static int entertaining;
|
|
|
|
|
|
|
|
|
|
static int norders;
|
|
|
|
|
static request *oa;
|
|
|
|
|
|
- Neue Messages fertig
Messages werden jetzt in einem anderen Meta-Format (message* of
message_type*) gespeichert, das man in beliebige Formate (CR oder NR)
rendern kann. crmessage.c und nrmessage.c sind die render-engines dafür.
Die Messagetypen werden in res/{de,en}/messages.xml gesammelt, ultimativ
kann das aber durchaus eine einzelne Datei sein. Die ist derzeit nicht
wirklich xml (Umlaute drin, keine Definitionsdatei), aber gut lesbar.
- make_message
Diese Funktion ersetzt new_message, und ist etwas einfacher in der Syntax:
make_message("dumb_mistake", "unit region command", u, r, cmd) erzeugt
eine neue Nachricht, die dann einfach mit add_message wie bisher an die
Nachrichtenliste gehängt werden kann.
TODO: Messages könnte man durchaus reference-counten, und in mehrere Listen
einfügen, solang sie a) mehrfachverwendet (Kampf!) und b) vom Betrachter
unabhängig sind. Das spart einigen Speicher.
- CR Version erhöht.
Weil die MESSAGETYPES Blocks anders sind als früher
- OFFENSIVE_DELAY
Verbietet Einheiten, deren Partei eine Reigon niht bewachen, den
Angriff in der Region, wenn sie sich in der Runde zuvor bewegt haben.
Status der letzten Runde wird in neuem Attribut at_moved gespeichert.
- SHORT_ATTACKS
ein define, das angibt ob Kämpfen grundsätzlich keine lange Aktion ist.
- XML Parser
xml.[hc] enthält einen XML-Parser, dem man ein plugin mit callbacks
übergibt, die nach dem Parsen eines tokens aufgerufen werden.
2001-04-12 19:21:57 +02:00
|
|
|
|
int
|
2001-01-25 10:37:55 +01:00
|
|
|
|
income(const unit * u)
|
|
|
|
|
{
|
2001-12-10 01:13:39 +01:00
|
|
|
|
switch(old_race(u->race)) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
case RC_FIREDRAGON:
|
|
|
|
|
return 150 * u->number;
|
|
|
|
|
case RC_DRAGON:
|
|
|
|
|
return 1000 * u->number;
|
|
|
|
|
case RC_WYRM:
|
|
|
|
|
return 5000 * u->number;
|
|
|
|
|
}
|
|
|
|
|
return 20 * u->number;
|
|
|
|
|
}
|
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
scramble(void *data, int n, size_t width)
|
|
|
|
|
{
|
2004-01-04 19:14:20 +01:00
|
|
|
|
int j;
|
|
|
|
|
char temp[64];
|
|
|
|
|
assert(width<=sizeof(temp));
|
|
|
|
|
for (j=0;j!=n;++j) {
|
2006-02-19 23:43:56 +01:00
|
|
|
|
int k = rng_int() % n;
|
2004-02-12 15:59:05 +01:00
|
|
|
|
if (k==j) continue;
|
2004-01-04 19:14:20 +01:00
|
|
|
|
memcpy(temp, (char*)data+j*width, width);
|
|
|
|
|
memcpy((char*)data+j*width, (char*)data+k*width, width);
|
|
|
|
|
memcpy((char*)data+k*width, temp, width);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
expandorders(region * r, request * requests)
|
|
|
|
|
{
|
|
|
|
|
unit *u;
|
|
|
|
|
request *o;
|
|
|
|
|
|
|
|
|
|
/* Alle Units ohne request haben ein -1, alle units mit orders haben ein
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* 0 hier stehen */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
for (u = r->units; u; u = u->next)
|
|
|
|
|
u->n = -1;
|
|
|
|
|
|
|
|
|
|
norders = 0;
|
|
|
|
|
|
2004-07-14 09:12:36 +02:00
|
|
|
|
for (o = requests; o; o = o->next) {
|
|
|
|
|
if (o->qty > 0) {
|
|
|
|
|
norders += o->qty;
|
|
|
|
|
}
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-07-14 09:12:36 +02:00
|
|
|
|
if (norders > 0) {
|
|
|
|
|
int i = 0;
|
|
|
|
|
oa = (request *) calloc(norders, sizeof(request));
|
|
|
|
|
for (o = requests; o; o = o->next) {
|
|
|
|
|
if (o->qty > 0) {
|
|
|
|
|
int j;
|
|
|
|
|
for (j = o->qty; j; j--) {
|
|
|
|
|
oa[i] = *o;
|
|
|
|
|
oa[i].unit->n = 0;
|
|
|
|
|
i++;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2003-09-21 10:47:28 +02:00
|
|
|
|
}
|
2004-07-14 09:12:36 +02:00
|
|
|
|
scramble(oa, norders, sizeof(request));
|
|
|
|
|
} else {
|
|
|
|
|
oa = NULL;
|
|
|
|
|
}
|
|
|
|
|
while (requests) {
|
|
|
|
|
request * o = requests->next;
|
2006-10-29 14:11:41 +01:00
|
|
|
|
free_order(requests->ord);
|
2004-07-14 09:12:36 +02:00
|
|
|
|
free(requests);
|
|
|
|
|
requests = o;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2002-02-16 11:37:18 +01:00
|
|
|
|
static void
|
|
|
|
|
change_level(unit * u, skill_t sk, int bylevel)
|
|
|
|
|
{
|
|
|
|
|
skill * sv = get_skill(u, sk);
|
|
|
|
|
assert(bylevel>0);
|
2002-02-23 17:18:26 +01:00
|
|
|
|
if (sv==0) sv = add_skill(u, sk);
|
|
|
|
|
sk_set(sv, sv->level+bylevel);
|
2002-02-16 11:37:18 +01:00
|
|
|
|
}
|
|
|
|
|
|
2005-06-04 03:29:45 +02:00
|
|
|
|
typedef struct recruitment {
|
|
|
|
|
struct recruitment * next;
|
|
|
|
|
faction * f;
|
|
|
|
|
request * requests;
|
|
|
|
|
int total, assigned;
|
|
|
|
|
} recruitment;
|
|
|
|
|
|
|
|
|
|
static recruitment *
|
|
|
|
|
select_recruitment(request ** rop, int (*quantify)(const struct race*, int), int * total)
|
|
|
|
|
{
|
|
|
|
|
recruitment * recruits = NULL;
|
|
|
|
|
|
|
|
|
|
while (*rop) {
|
|
|
|
|
recruitment * rec = recruits;
|
|
|
|
|
request * ro = *rop;
|
|
|
|
|
unit * u = ro->unit;
|
|
|
|
|
const race * rc = u->number?u->race:u->faction->race;
|
|
|
|
|
int qty = quantify(rc, ro->qty);
|
|
|
|
|
|
|
|
|
|
if (qty<0) {
|
|
|
|
|
rop = &ro->next; /* skip this one */
|
|
|
|
|
} else {
|
|
|
|
|
*rop = ro->next; /* remove this one */
|
|
|
|
|
while (rec && rec->f!=u->faction) rec = rec->next;
|
|
|
|
|
if (rec==NULL) {
|
|
|
|
|
rec = malloc(sizeof(recruitment));
|
|
|
|
|
rec->f = u->faction;
|
|
|
|
|
rec->total = 0;
|
|
|
|
|
rec->assigned = 0;
|
|
|
|
|
rec->requests = NULL;
|
|
|
|
|
rec->next = recruits;
|
|
|
|
|
recruits = rec;
|
|
|
|
|
}
|
|
|
|
|
*total += qty;
|
|
|
|
|
rec->total += qty;
|
|
|
|
|
ro->next = rec->requests;
|
|
|
|
|
rec->requests = ro;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return recruits;
|
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
static void
|
2005-06-04 03:29:45 +02:00
|
|
|
|
add_recruits(unit * u, int number, int wanted)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2005-06-05 14:21:37 +02:00
|
|
|
|
assert(number<=wanted);
|
2005-06-04 03:29:45 +02:00
|
|
|
|
if (number > 0) {
|
|
|
|
|
unit * unew;
|
2006-02-17 23:59:03 +01:00
|
|
|
|
#ifdef KARMA_MODULE
|
2005-06-04 03:29:45 +02:00
|
|
|
|
int i = fspecial(u->faction, FS_MILITIA);
|
2006-02-17 23:59:03 +01:00
|
|
|
|
#endif /* KARMA_MODULE */
|
2005-06-04 03:29:45 +02:00
|
|
|
|
|
|
|
|
|
if (u->number==0) {
|
|
|
|
|
set_number(u, number);
|
|
|
|
|
u->hp = number * unit_max_hp(u);
|
|
|
|
|
unew = u;
|
|
|
|
|
} else {
|
|
|
|
|
unew = create_unit(u->region, u->faction, number, u->race, 0, NULL, u);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-06-04 03:29:45 +02:00
|
|
|
|
if (unew->race == new_race[RC_URUK]) {
|
2005-10-02 19:35:25 +02:00
|
|
|
|
change_level(unew, SK_MELEE, 1);
|
2005-06-04 03:29:45 +02:00
|
|
|
|
change_level(unew, SK_SPEAR, 1);
|
|
|
|
|
}
|
|
|
|
|
if (unew->race->ec_flags & ECF_REC_HORSES) {
|
|
|
|
|
change_level(unew, SK_RIDING, 1);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-02-17 23:59:03 +01:00
|
|
|
|
#ifdef KARMA_MODULE
|
2005-06-04 03:29:45 +02:00
|
|
|
|
if (i > 0) {
|
|
|
|
|
if (unew->race->bonus[SK_SPEAR] >= 0)
|
|
|
|
|
change_level(unew, SK_SPEAR, i);
|
2005-10-02 19:35:25 +02:00
|
|
|
|
if (unew->race->bonus[SK_MELEE] >= 0)
|
|
|
|
|
change_level(unew, SK_MELEE, i);
|
2005-06-04 03:29:45 +02:00
|
|
|
|
if (unew->race->bonus[SK_LONGBOW] >= 0)
|
|
|
|
|
change_level(unew, SK_LONGBOW, i);
|
|
|
|
|
if (unew->race->bonus[SK_CROSSBOW] >= 0)
|
|
|
|
|
change_level(unew, SK_CROSSBOW, i);
|
|
|
|
|
if (unew->race->bonus[SK_RIDING] >= 0)
|
|
|
|
|
change_level(unew, SK_RIDING, i);
|
|
|
|
|
if (unew->race->bonus[SK_AUSDAUER] >= 0)
|
|
|
|
|
change_level(unew, SK_AUSDAUER, i);
|
|
|
|
|
}
|
2006-02-17 23:59:03 +01:00
|
|
|
|
#endif /* KARMA_MODULE */
|
2005-06-04 03:29:45 +02:00
|
|
|
|
if (unew!=u) {
|
|
|
|
|
transfermen(unew, u, unew->number);
|
|
|
|
|
destroy_unit(unew);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (number < wanted) {
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("recruit",
|
|
|
|
|
"unit region amount want", u, u->region, number, wanted));
|
|
|
|
|
}
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-06-04 03:29:45 +02:00
|
|
|
|
static int
|
|
|
|
|
any_recruiters(const struct race * rc, int qty)
|
|
|
|
|
{
|
|
|
|
|
if (rc==new_race[RC_URUK]) return qty;
|
|
|
|
|
return qty * 2;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-06-04 03:29:45 +02:00
|
|
|
|
static int
|
|
|
|
|
peasant_recruiters(const struct race * rc, int qty)
|
|
|
|
|
{
|
|
|
|
|
if (rc->ec_flags & ECF_REC_UNLIMITED) return -1;
|
|
|
|
|
if (rc->ec_flags & ECF_REC_HORSES) return -1;
|
|
|
|
|
if (rc==new_race[RC_URUK]) return qty;
|
|
|
|
|
return qty * 2;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-06-04 03:29:45 +02:00
|
|
|
|
static int
|
|
|
|
|
horse_recruiters(const struct race * rc, int qty)
|
|
|
|
|
{
|
|
|
|
|
if (rc->ec_flags & ECF_REC_UNLIMITED) return -1;
|
|
|
|
|
if (rc->ec_flags & ECF_REC_HORSES) return qty * 2;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
2002-02-18 18:57:27 +01:00
|
|
|
|
|
2005-06-04 03:29:45 +02:00
|
|
|
|
static int
|
|
|
|
|
do_recruiting(recruitment * recruits, int available)
|
|
|
|
|
{
|
|
|
|
|
recruitment * rec;
|
|
|
|
|
int recruited = 0;
|
|
|
|
|
|
|
|
|
|
while (available>0) {
|
|
|
|
|
int n = 0;
|
|
|
|
|
int rest, mintotal = INT_MAX;
|
|
|
|
|
|
|
|
|
|
for (rec=recruits;rec!=NULL;rec=rec->next) {
|
|
|
|
|
int want = rec->total - rec->assigned;
|
|
|
|
|
if (want>0) {
|
|
|
|
|
if (mintotal>want) mintotal = want;
|
|
|
|
|
++n;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (n==0) break;
|
|
|
|
|
if (mintotal*n>available) {
|
|
|
|
|
mintotal = available/n;
|
|
|
|
|
}
|
|
|
|
|
rest = available - mintotal*n;
|
|
|
|
|
|
|
|
|
|
for (rec=recruits;rec!=NULL;rec=rec->next) {
|
|
|
|
|
int want = rec->total - rec->assigned;
|
|
|
|
|
|
|
|
|
|
if (want>0) {
|
|
|
|
|
int get = mintotal;
|
2006-02-19 23:43:56 +01:00
|
|
|
|
if (want>mintotal && rest<n && (rng_int() % n) < rest) {
|
2005-06-04 03:29:45 +02:00
|
|
|
|
--rest;
|
|
|
|
|
++get;
|
|
|
|
|
}
|
|
|
|
|
assert(get<=want);
|
|
|
|
|
available -= get;
|
|
|
|
|
rec->assigned += get;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (rec=recruits;rec!=NULL;rec=rec->next) {
|
|
|
|
|
request * req;
|
|
|
|
|
int get = rec->assigned;
|
|
|
|
|
|
|
|
|
|
for (req=rec->requests;req;req=req->next) {
|
|
|
|
|
unit * u = req->unit;
|
|
|
|
|
const race * rc = u->faction->race;
|
|
|
|
|
int number, multi = 2;
|
|
|
|
|
|
|
|
|
|
if (rc==new_race[RC_URUK]) multi = 1;
|
|
|
|
|
|
2005-06-05 14:21:37 +02:00
|
|
|
|
number = min(req->qty, get / multi);
|
2005-06-04 03:29:45 +02:00
|
|
|
|
if (rc->recruitcost) {
|
2006-02-19 23:43:56 +01:00
|
|
|
|
int afford = get_pooled(u, oldresourcetype[R_SILVER], GET_DEFAULT, number*rc->recruitcost) / rc->recruitcost;
|
2005-06-04 03:29:45 +02:00
|
|
|
|
number = min(number, afford);
|
2005-10-30 16:42:15 +01:00
|
|
|
|
use_pooled(u, oldresourcetype[R_SILVER], GET_DEFAULT, rc->recruitcost*number);
|
2005-06-04 03:29:45 +02:00
|
|
|
|
}
|
2006-10-29 14:04:18 +01:00
|
|
|
|
if (u->number+number>UNIT_MAXSIZE) {
|
2006-10-29 14:11:41 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, req->ord, "error_unit_size", "maxsize", UNIT_MAXSIZE));
|
2006-10-29 14:04:18 +01:00
|
|
|
|
number = UNIT_MAXSIZE-u->number;
|
|
|
|
|
assert(number>=0);
|
|
|
|
|
}
|
2005-06-04 03:29:45 +02:00
|
|
|
|
add_recruits(u, number, req->qty);
|
|
|
|
|
if ((rc->ec_flags & ECF_REC_ETHEREAL)==0) {
|
|
|
|
|
recruited += number * multi;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
get -= number * multi;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return recruited;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
free_recruitments(recruitment * recruits)
|
|
|
|
|
{
|
|
|
|
|
while (recruits) {
|
|
|
|
|
recruitment * rec = recruits;
|
|
|
|
|
recruits = rec->next;
|
2005-06-11 10:09:55 +02:00
|
|
|
|
while (rec->requests) {
|
|
|
|
|
request * req = rec->requests;
|
|
|
|
|
rec->requests = req->next;
|
|
|
|
|
free(req);
|
|
|
|
|
}
|
2005-06-04 03:29:45 +02:00
|
|
|
|
free(rec);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Rekrutierung */
|
|
|
|
|
static void
|
|
|
|
|
expandrecruit(region * r, request * recruitorders)
|
|
|
|
|
{
|
|
|
|
|
recruitment * recruits = NULL;
|
|
|
|
|
|
2005-06-05 14:21:37 +02:00
|
|
|
|
int orc_total = 0;
|
2005-06-04 03:29:45 +02:00
|
|
|
|
|
|
|
|
|
/* centaurs: */
|
2005-06-05 14:21:37 +02:00
|
|
|
|
recruits = select_recruitment(&recruitorders, horse_recruiters, &orc_total);
|
2005-06-04 03:29:45 +02:00
|
|
|
|
if (recruits) {
|
2005-06-04 03:43:38 +02:00
|
|
|
|
int recruited, horses = rhorses(r) * 2;
|
2005-06-05 14:21:37 +02:00
|
|
|
|
if (orc_total<horses) horses = orc_total;
|
2005-06-04 03:43:38 +02:00
|
|
|
|
recruited = do_recruiting(recruits, horses);
|
|
|
|
|
rsethorses(r, (horses - recruited) / 2);
|
2005-06-04 03:29:45 +02:00
|
|
|
|
free_recruitments(recruits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* peasant limited: */
|
2005-06-05 14:21:37 +02:00
|
|
|
|
recruits = select_recruitment(&recruitorders, peasant_recruiters, &orc_total);
|
2005-06-04 03:29:45 +02:00
|
|
|
|
if (recruits) {
|
2005-06-05 14:21:37 +02:00
|
|
|
|
int orc_recruited, orc_peasants = rpeasants(r) * 2;
|
|
|
|
|
int orc_frac = orc_peasants / RECRUITFRACTION; /* anzahl orks. 2 ork = 1 bauer */
|
|
|
|
|
if (orc_total<orc_frac) orc_frac = orc_total;
|
|
|
|
|
orc_recruited = do_recruiting(recruits, orc_frac);
|
|
|
|
|
assert(orc_recruited<=orc_frac);
|
|
|
|
|
rsetpeasants(r, (orc_peasants - orc_recruited)/2);
|
2005-06-04 03:29:45 +02:00
|
|
|
|
free_recruitments(recruits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* no limit: */
|
2005-06-05 14:21:37 +02:00
|
|
|
|
recruits = select_recruitment(&recruitorders, any_recruiters, &orc_total);
|
2005-06-04 03:29:45 +02:00
|
|
|
|
if (recruits) {
|
|
|
|
|
do_recruiting(recruits, INT_MAX);
|
|
|
|
|
free_recruitments(recruits);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assert(recruitorders==NULL);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2004-06-21 18:45:27 +02:00
|
|
|
|
recruit(unit * u, struct order * ord, request ** recruitorders)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
int n;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region * r = u->region;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
plane * pl;
|
|
|
|
|
request *o;
|
|
|
|
|
int recruitcost;
|
2005-02-12 20:54:32 +01:00
|
|
|
|
const faction * f = u->faction;
|
|
|
|
|
const struct race * rc = f->race;
|
|
|
|
|
|
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
2007-06-26 11:32:28 +02:00
|
|
|
|
n = getuint();
|
2005-02-12 20:54:32 +01:00
|
|
|
|
if (f->no==MONSTER_FACTION) {
|
|
|
|
|
/* Monster d<>rfen REKRUTIERE 15 dracoid machen */
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char * str = getstrtoken();
|
2007-06-27 22:39:49 +02:00
|
|
|
|
rc = findrace(str, f->locale);
|
2005-02-12 20:54:32 +01:00
|
|
|
|
if (rc==NULL) rc = f->race;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2002-04-21 09:31:27 +02:00
|
|
|
|
#if GUARD_DISABLES_RECRUIT == 1
|
2002-04-21 19:22:48 +02:00
|
|
|
|
/* this is a very special case because the recruiting unit may be empty
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* at this point and we have to look at the creating unit instead. This
|
|
|
|
|
* is done in cansee, which is called indirectly by is_guarded(). */
|
2005-02-12 20:54:32 +01:00
|
|
|
|
if (is_guarded(r, u, GUARD_RECRUIT)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 70, MSG_EVENT);
|
2002-04-21 09:31:27 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2004-09-05 14:29:46 +02:00
|
|
|
|
if (rc == new_race[RC_INSECT]) {
|
2005-06-05 15:34:08 +02:00
|
|
|
|
if (get_gamedate(turn, 0)->season == 0 && rterrain(r) != T_DESERT) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#ifdef INSECT_POTION
|
|
|
|
|
boolean usepotion = false;
|
|
|
|
|
unit *u2;
|
|
|
|
|
|
|
|
|
|
for (u2 = r->units; u2; u2 = u2->next)
|
2003-07-29 11:48:03 +02:00
|
|
|
|
if (fval(u2, UFL_WARMTH)) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
usepotion = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2003-09-21 10:47:28 +02:00
|
|
|
|
if (!usepotion)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#endif
|
2003-09-21 10:47:28 +02:00
|
|
|
|
{
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 98, MSG_EVENT);
|
2003-09-21 10:47:28 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
/* in Gletschern, Eisbergen gar nicht rekrutieren */
|
|
|
|
|
if (r_insectstalled(r)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 97, MSG_EVENT);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (is_cursed(r->attribs, C_RIOT, 0)) {
|
|
|
|
|
/* Die Region befindet sich in Aufruhr */
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 237, MSG_EVENT);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2004-09-05 14:29:46 +02:00
|
|
|
|
if (!(rc->ec_flags & ECF_REC_HORSES) && fval(r, RF_ORCIFIED)) {
|
|
|
|
|
if (rc != new_race[RC_URUK])
|
2004-07-14 09:12:36 +02:00
|
|
|
|
{
|
|
|
|
|
cmistake(u, ord, 238, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-02-19 23:43:56 +01:00
|
|
|
|
|
|
|
|
|
recruitcost = rc->recruitcost;
|
|
|
|
|
if (recruitcost) {
|
|
|
|
|
pl = getplane(r);
|
|
|
|
|
if (pl && fval(pl, PFL_NORECRUITS)) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "error_pflnorecruit", ""));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (get_pooled(u, oldresourcetype[R_SILVER], GET_DEFAULT, recruitcost) < recruitcost) {
|
2005-10-29 14:08:40 +02:00
|
|
|
|
cmistake(u, ord, 142, MSG_EVENT);
|
2006-02-19 23:43:56 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!playerrace(u->race) || idle(u->faction)) {
|
|
|
|
|
cmistake(u, ord, 139, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
/* snotlinge sollten hiermit bereits abgefangen werden, die
|
|
|
|
|
* parteirasse ist uruk oder ork*/
|
|
|
|
|
if (u->race != rc) {
|
|
|
|
|
if (u->number != 0) {
|
|
|
|
|
cmistake(u, ord, 139, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
else u->race = rc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (has_skill(u, SK_MAGIC)) {
|
|
|
|
|
/* error158;de;{unit} in {region}: '{command}' - Magier arbeiten
|
|
|
|
|
* grunds<EFBFBD>tzlich nur alleine! */
|
|
|
|
|
cmistake(u, ord, 158, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (has_skill(u, SK_ALCHEMY)
|
|
|
|
|
&& count_skill(u->faction, SK_ALCHEMY) + n >
|
|
|
|
|
max_skill(u->faction, SK_ALCHEMY))
|
|
|
|
|
{
|
|
|
|
|
cmistake(u, ord, 156, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (recruitcost>0) {
|
|
|
|
|
int pooled = get_pooled(u, oldresourcetype[R_SILVER], GET_DEFAULT, recruitcost * n);
|
|
|
|
|
n = min(n, pooled / recruitcost);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u->wants = n;
|
|
|
|
|
|
|
|
|
|
if (!n) {
|
|
|
|
|
cmistake(u, ord, 142, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
o = (request *) calloc(1, sizeof(request));
|
|
|
|
|
o->qty = n;
|
|
|
|
|
o->unit = u;
|
2006-10-29 14:11:41 +01:00
|
|
|
|
o->ord = copy_order(ord);
|
2006-02-19 23:43:56 +01:00
|
|
|
|
addlist(recruitorders, o);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2004-06-21 18:45:27 +02:00
|
|
|
|
give_cmd(unit * u, order * ord)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region * r = u->region;
|
2005-12-02 21:23:54 +01:00
|
|
|
|
unit *u2;
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char *s;
|
2005-12-02 21:23:54 +01:00
|
|
|
|
int i, n;
|
|
|
|
|
const item_type * itype;
|
2006-10-03 14:02:27 +02:00
|
|
|
|
param_t p;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
2005-12-02 21:23:54 +01:00
|
|
|
|
u2 = getunit(r, u->faction);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
if (!u2 && !getunitpeasants) {
|
2007-02-10 22:47:51 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "feedback_unit_not_found", ""));
|
2005-12-02 21:23:54 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
/* Damit Tarner nicht durch die Fehlermeldung enttarnt werden k<>nnen */
|
|
|
|
|
if (u2 && !alliedunit(u2, u->faction, HELP_GIVE) && !cansee(u->faction,r,u2,0) && !ucontact(u2, u) && !fval(u2, UFL_TAKEALL)) {
|
2007-02-10 22:47:51 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "feedback_unit_not_found", ""));
|
2005-12-02 21:23:54 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (u == u2) {
|
|
|
|
|
cmistake(u, ord, 8, MSG_COMMERCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
/* UFL_TAKEALL ist ein grober Hack. Generalisierung tut not, ist aber nicht
|
|
|
|
|
* wirklich einfach. */
|
|
|
|
|
if (r->planep && fval(r->planep, PFL_NOGIVE) && (!u2 || !fval(u2, UFL_TAKEALL))) {
|
|
|
|
|
cmistake(u, ord, 268, MSG_COMMERCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
s = getstrtoken();
|
2006-10-03 14:02:27 +02:00
|
|
|
|
p = findparam(s, u->faction->locale);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-10-03 14:02:27 +02:00
|
|
|
|
if (p == P_CONTROL) {
|
2007-06-26 11:32:28 +02:00
|
|
|
|
message * msg;
|
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
if (!u2) {
|
2007-02-10 22:47:51 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "feedback_unit_not_found", ""));
|
2005-12-02 21:23:54 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!u->building && !u->ship) {
|
|
|
|
|
cmistake(u, ord, 140, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (u->building && u2->building != u->building) {
|
|
|
|
|
cmistake(u, ord, 33, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (u->ship && u2->ship != u->ship) {
|
|
|
|
|
cmistake(u, ord, 32, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!fval(u, UFL_OWNER)) {
|
|
|
|
|
cmistake(u, ord, 49, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!alliedunit(u2, u->faction, HELP_GIVE) && !ucontact(u2, u)) {
|
|
|
|
|
cmistake(u, ord, 40, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
freset(u, UFL_OWNER);
|
|
|
|
|
fset(u2, UFL_OWNER);
|
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
msg = msg_message("givecommand", "unit receipient", u, u2);
|
|
|
|
|
add_message(&u->faction->msgs, msg);
|
2005-12-02 21:23:54 +01:00
|
|
|
|
if (u->faction != u2->faction) {
|
2007-06-26 11:32:28 +02:00
|
|
|
|
add_message(&u2->faction->msgs, msg);
|
2005-12-02 21:23:54 +01:00
|
|
|
|
}
|
2007-06-26 11:32:28 +02:00
|
|
|
|
msg_release(msg);
|
2005-12-02 21:23:54 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2006-10-03 14:02:27 +02:00
|
|
|
|
|
|
|
|
|
else if (u2 && u2->race == new_race[RC_SPELL]) {
|
2007-02-10 22:47:51 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "feedback_unit_not_found", ""));
|
2005-12-02 21:23:54 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
/* if ((u->race->ec_flags & NOGIVE) || fval(u,UFL_LOCKED)) {*/
|
2006-10-03 14:02:27 +02:00
|
|
|
|
else if (u->race->ec_flags & NOGIVE && u2!=NULL) {
|
2005-12-02 21:23:54 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_nogive", "race", u->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
/* sperrt hier auch personen<65>bergaben!
|
2006-10-03 14:02:27 +02:00
|
|
|
|
else if (u2 && !(u2->race->ec_flags & GETITEM)) {
|
2005-12-02 21:23:54 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_notake", "race", u2->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* <20>bergabe aller Kr<4B>uter */
|
2006-10-03 14:02:27 +02:00
|
|
|
|
else if (p == P_HERBS) {
|
2005-12-02 21:23:54 +01:00
|
|
|
|
boolean given = false;
|
|
|
|
|
if (!(u->race->ec_flags & GIVEITEM) && u2!=NULL) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_nogive", "race", u->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (u2 && !(u2->race->ec_flags & GETITEM)) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_notake", "race", u2->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!u2) {
|
|
|
|
|
if (!getunitpeasants) {
|
2007-02-10 22:47:51 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "feedback_unit_not_found", ""));
|
2005-12-02 21:23:54 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (u->items) {
|
|
|
|
|
item **itmp=&u->items;
|
|
|
|
|
while (*itmp) {
|
2007-10-11 23:48:31 +02:00
|
|
|
|
item * itm = *itmp;
|
|
|
|
|
const item_type * itype = itm->type;
|
|
|
|
|
if (fval(itype, ITF_HERB) && itm->number>0) {
|
2005-12-02 21:23:54 +01:00
|
|
|
|
/* give_item <20>ndert im fall,das man alles <20>bergibt, die
|
|
|
|
|
* item-liste der unit, darum continue vor pointerumsetzten */
|
2007-10-11 23:48:31 +02:00
|
|
|
|
if (give_item(itm->number, itm->type, u, u2, ord)==0) {
|
2005-12-02 21:23:54 +01:00
|
|
|
|
given = true;
|
2007-10-11 23:48:31 +02:00
|
|
|
|
if (*itmp!=itm) continue;
|
2005-12-02 21:23:54 +01:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-10-11 23:48:31 +02:00
|
|
|
|
itmp = &itm->next;
|
2005-12-02 21:23:54 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!given) cmistake(u, ord, 38, MSG_COMMERCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2006-10-03 14:02:27 +02:00
|
|
|
|
|
|
|
|
|
else if (p == P_ZAUBER) {
|
2005-12-02 21:23:54 +01:00
|
|
|
|
cmistake(u, ord, 7, MSG_COMMERCE);
|
|
|
|
|
/* geht nimmer */
|
|
|
|
|
return;
|
|
|
|
|
}
|
2006-10-03 14:02:27 +02:00
|
|
|
|
|
|
|
|
|
else if (p == P_UNIT) { /* Einheiten uebergeben */
|
2005-12-02 21:23:54 +01:00
|
|
|
|
if (!(u->race->ec_flags & GIVEUNIT)) {
|
|
|
|
|
cmistake(u, ord, 167, MSG_COMMERCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
if (u2 && !alliedunit(u2, u->faction, HELP_GIVE) && !ucontact(u2, u)) {
|
|
|
|
|
cmistake(u, ord, 40, MSG_COMMERCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
give_unit(u, u2, ord);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2006-10-03 14:02:27 +02:00
|
|
|
|
|
|
|
|
|
else if (p==P_ANY) {
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char * s = getstrtoken();
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2006-10-03 14:02:27 +02:00
|
|
|
|
if (*s == 0) { /* Alle Gegenst<73>nde <20>bergeben */
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2006-10-03 14:02:27 +02:00
|
|
|
|
if (u2 && !alliedunit(u2, u->faction, HELP_GIVE) && !ucontact(u2, u)) {
|
|
|
|
|
cmistake(u, ord, 40, MSG_COMMERCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* do these checks once, not for each item we have: */
|
2005-12-02 21:23:54 +01:00
|
|
|
|
if (!(u->race->ec_flags & GIVEITEM) && u2!=NULL) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_nogive", "race", u->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (u2 && !(u2->race->ec_flags & GETITEM)) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_notake", "race", u2->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
2002-09-19 14:34:17 +02:00
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
/* f<>r alle items einmal pr<70>fen, ob wir mehr als von diesem Typ
|
2006-10-03 14:02:27 +02:00
|
|
|
|
* reserviert ist besitzen und diesen Teil dann <EFBFBD>bergeben */
|
2005-12-02 21:23:54 +01:00
|
|
|
|
if (u->items) {
|
|
|
|
|
item **itmp=&u->items;
|
|
|
|
|
while (*itmp) {
|
2006-08-07 23:46:40 +02:00
|
|
|
|
item * itm = *itmp;
|
|
|
|
|
const item_type * itype = itm->type;
|
|
|
|
|
if (itm->number > 0 && itm->number - get_reservation(u, itype->rtype) > 0) {
|
|
|
|
|
n = itm->number - get_reservation(u, itype->rtype);
|
2007-10-11 23:48:31 +02:00
|
|
|
|
if (give_item(n, itype, u, u2, ord)==0) {
|
|
|
|
|
if (*itmp!=itm) continue;
|
|
|
|
|
}
|
2006-08-07 23:46:40 +02:00
|
|
|
|
}
|
|
|
|
|
itmp = &itm->next;
|
2005-12-02 21:23:54 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
2006-10-03 14:02:27 +02:00
|
|
|
|
else {
|
|
|
|
|
param_t p2 = findparam(s, u->faction->locale);
|
|
|
|
|
|
|
|
|
|
if (p2 == P_PERSON) {
|
|
|
|
|
if (!(u->race->ec_flags & GIVEPERSON)) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_noregroup", "race", u->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
n = u->number;
|
|
|
|
|
give_men(n, u, u2, ord);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(u->race->ec_flags & GIVEITEM) && u2!=NULL) {
|
2005-12-02 21:23:54 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2006-10-03 14:02:27 +02:00
|
|
|
|
msg_feedback(u, ord, "race_nogive", "race", u->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (u2 && !(u2->race->ec_flags & GETITEM)) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_notake", "race", u2->race));
|
2005-12-02 21:23:54 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-10-03 14:02:27 +02:00
|
|
|
|
itype = finditemtype(s, u->faction->locale);
|
|
|
|
|
if (itype!=NULL) {
|
|
|
|
|
item * i = *i_find(&u->items, itype);
|
|
|
|
|
if (i!=NULL) {
|
|
|
|
|
n = i->number - get_reservation(u, itype->rtype);
|
|
|
|
|
give_item(n, itype, u, u2, ord);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-12-02 21:23:54 +01:00
|
|
|
|
}
|
2006-10-03 14:02:27 +02:00
|
|
|
|
} else if (p==P_EACH) {
|
|
|
|
|
if (u2==NULL) {
|
2005-12-02 21:23:54 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2006-10-03 14:02:27 +02:00
|
|
|
|
msg_feedback(u, ord, "peasants_give_invalid", ""));
|
2005-12-02 21:23:54 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2006-10-03 14:02:27 +02:00
|
|
|
|
s = getstrtoken(); /* skip one ahead to get the amount. */
|
2005-12-02 21:23:54 +01:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
n = atoip((const char *)s); /* n: anzahl */
|
2006-10-03 14:02:27 +02:00
|
|
|
|
if (p==P_EACH) {
|
|
|
|
|
n *= u2->number;
|
|
|
|
|
}
|
2005-12-02 21:23:54 +01:00
|
|
|
|
s = getstrtoken();
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
if (s == NULL) {
|
|
|
|
|
cmistake(u, ord, 113, MSG_COMMERCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-02-05 19:11:23 +01:00
|
|
|
|
if (u2 && !alliedunit(u2, u->faction, HELP_GIVE) && !ucontact(u2, u)) {
|
2005-12-02 21:23:54 +01:00
|
|
|
|
const resource_type * rtype = findresourcetype(s, u->faction->locale);
|
|
|
|
|
if (rtype==NULL || !fval(rtype, RTF_SNEAK))
|
|
|
|
|
{
|
|
|
|
|
cmistake(u, ord, 40, MSG_COMMERCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
i = findparam(s, u->faction->locale);
|
|
|
|
|
if (i == P_PERSON) {
|
|
|
|
|
if (!(u->race->ec_flags & GIVEPERSON)) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_noregroup", "race", u->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
give_men(n, u, u2, ord);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
if (u2!=NULL) {
|
|
|
|
|
if (!(u->race->ec_flags & GIVEITEM) && u2!=NULL) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_nogive", "race", u->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!(u2->race->ec_flags & GETITEM)) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, ord, "race_notake", "race", u2->race));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2002-09-19 14:34:17 +02:00
|
|
|
|
|
2005-12-02 21:23:54 +01:00
|
|
|
|
itype = finditemtype(s, u->faction->locale);
|
|
|
|
|
if (itype!=NULL) {
|
|
|
|
|
give_item(n, itype, u, u2, ord);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
cmistake(u, ord, 123, MSG_COMMERCE);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2004-06-21 18:45:27 +02:00
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
forget_cmd(unit * u, order * ord)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2007-06-20 02:34:02 +02:00
|
|
|
|
skill_t sk;
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char *s;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
|
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
|
|
|
|
s = getstrtoken();
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-06-20 02:34:02 +02:00
|
|
|
|
if ((sk = findskill(s, u->faction->locale)) != NOSKILL) {
|
2003-09-21 10:47:28 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2007-06-20 02:34:02 +02:00
|
|
|
|
msg_message("forget", "unit skill", u, sk));
|
|
|
|
|
set_level(u, sk, 0);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2004-06-21 18:45:27 +02:00
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
void
|
2005-06-04 16:47:02 +02:00
|
|
|
|
add_spende(faction * f1, faction * f2, int amount, region * r)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2005-06-04 16:47:02 +02:00
|
|
|
|
donation *sp;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-06-04 16:47:02 +02:00
|
|
|
|
sp = r->donations;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
while (sp) {
|
2005-06-04 16:47:02 +02:00
|
|
|
|
if (sp->f1 == f1 && sp->f2 == f2) {
|
|
|
|
|
sp->amount += amount;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
sp = sp->next;
|
|
|
|
|
}
|
|
|
|
|
|
2005-06-04 16:47:02 +02:00
|
|
|
|
sp = calloc(1, sizeof(donation));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
sp->f1 = f1;
|
|
|
|
|
sp->f2 = f2;
|
2005-06-04 16:47:02 +02:00
|
|
|
|
sp->amount = amount;
|
|
|
|
|
sp->next = r->donations;
|
|
|
|
|
r->donations = sp;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static boolean
|
2001-12-10 01:13:39 +01:00
|
|
|
|
maintain(building * b, boolean first)
|
2003-09-21 10:47:28 +02:00
|
|
|
|
/* first==false -> take money from wherever you can */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
int c;
|
|
|
|
|
region * r = b->region;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
boolean paid = true, work = first;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
unit * u;
|
2005-01-19 21:33:13 +01:00
|
|
|
|
if (fval(b, BLD_MAINTAINED) || b->type==NULL || b->type->maintenance==NULL || is_cursed(b->attribs, C_NOCOST, 0)) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fset(b, BLD_MAINTAINED);
|
2001-02-03 14:45:35 +01:00
|
|
|
|
fset(b, BLD_WORKING);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return true;
|
|
|
|
|
}
|
2006-02-19 23:43:56 +01:00
|
|
|
|
u = buildingowner(r, b);
|
|
|
|
|
if (u==NULL) return false;
|
|
|
|
|
for (c=0;b->type->maintenance[c].number;++c) {
|
|
|
|
|
const maintenance * m = b->type->maintenance+c;
|
|
|
|
|
int need = m->number;
|
|
|
|
|
|
|
|
|
|
if (fval(m, MTF_VARIABLE)) need = need * b->size;
|
|
|
|
|
if (u) {
|
|
|
|
|
/* first ist im ersten versuch true, im zweiten aber false! Das
|
|
|
|
|
* bedeutet, das in der Runde in die Region geschafften Resourcen
|
|
|
|
|
* nicht genutzt werden k<EFBFBD>nnen, weil die reserviert sind! */
|
|
|
|
|
if (!first) need -= get_pooled(u, m->rtype, GET_ALL, need);
|
|
|
|
|
else need -= get_pooled(u, m->rtype, GET_DEFAULT, need);
|
|
|
|
|
if (!first && need > 0) {
|
|
|
|
|
unit * ua;
|
2007-06-20 02:34:02 +02:00
|
|
|
|
for (ua=r->units;ua;ua=ua->next) freset(ua->faction, FFL_SELECT);
|
|
|
|
|
fset(u->faction, FFL_SELECT); /* hat schon */
|
2006-02-19 23:43:56 +01:00
|
|
|
|
for (ua=r->units;ua;ua=ua->next) {
|
2007-06-20 02:34:02 +02:00
|
|
|
|
if (!fval(ua->faction, FFL_SELECT) && (ua->faction == u->faction || alliedunit(ua, u->faction, HELP_MONEY))) {
|
2006-02-19 23:43:56 +01:00
|
|
|
|
need -= get_pooled(ua, m->rtype, GET_ALL, need);
|
2007-06-20 02:34:02 +02:00
|
|
|
|
fset(ua->faction, FFL_SELECT);
|
2006-02-19 23:43:56 +01:00
|
|
|
|
if (need<=0) break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (need > 0) {
|
|
|
|
|
if (!fval(m, MTF_VITAL)) work = false;
|
|
|
|
|
else {
|
|
|
|
|
paid = false;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (paid && c>0) {
|
|
|
|
|
/* TODO: wieviel von was wurde bezahlt */
|
|
|
|
|
if (first) {
|
2005-07-10 23:22:54 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("maintenance", "unit building", u, b));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
} else {
|
2005-07-10 23:22:54 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("maintenance_late", "building", b));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
fset(b, BLD_MAINTAINED);
|
2005-07-10 23:22:54 +02:00
|
|
|
|
if (work) {
|
|
|
|
|
fset(b, BLD_WORKING);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
for (c=0;b->type->maintenance[c].number;++c) {
|
|
|
|
|
const maintenance * m = b->type->maintenance+c;
|
|
|
|
|
int cost = m->number;
|
|
|
|
|
|
|
|
|
|
if (!fval(m, MTF_VITAL) && !work) continue;
|
|
|
|
|
if (fval(m, MTF_VARIABLE)) cost = cost * b->size;
|
|
|
|
|
|
2005-10-30 16:42:15 +01:00
|
|
|
|
if (!first) cost -= use_pooled(u, m->rtype, GET_ALL, cost);
|
|
|
|
|
else cost -= use_pooled(u, m->rtype, GET_SLACK|GET_RESERVE|GET_POOLED_SLACK, cost);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (!first && cost > 0) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
unit * ua;
|
2007-06-20 02:34:02 +02:00
|
|
|
|
for (ua=r->units;ua;ua=ua->next) freset(ua->faction, FFL_SELECT);
|
|
|
|
|
fset(u->faction, FFL_SELECT); /* hat schon */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
for (ua=r->units;ua;ua=ua->next) {
|
2007-06-20 02:34:02 +02:00
|
|
|
|
if (!fval(ua->faction, FFL_SELECT) && alliedunit(ua, u->faction, HELP_MONEY)) {
|
2005-10-30 16:42:15 +01:00
|
|
|
|
int give = use_pooled(ua, m->rtype, GET_ALL, cost);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (!give) continue;
|
|
|
|
|
cost -= give;
|
2007-06-20 02:34:02 +02:00
|
|
|
|
fset(ua->faction, FFL_SELECT);
|
2004-03-28 22:53:47 +02:00
|
|
|
|
if (m->rtype==r_silver) add_spende(ua->faction, u->faction, give, r);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (cost<=0) break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
assert(cost==0);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2006-01-29 02:55:36 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2003-09-21 10:47:28 +02:00
|
|
|
|
msg_message("maintenancefail", "unit building", u, b));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
gebaeude_stuerzt_ein(region * r, building * b)
|
|
|
|
|
{
|
|
|
|
|
unit *u;
|
|
|
|
|
int n, i;
|
|
|
|
|
int opfer = 0;
|
* möglichen Exploit beseitigt:
GIB xyz EINHEIT
GIB 0 ALLES SILBER
--> ALLE Befehle der übergebenen Einheit werden gelöscht.
* neue Funktion (convenience): ucansee(f, u1, u2)
liefert u1, wenn cansee(f,u1), sonst u2
* neue mistakes eingefügt uns übersetzt
* message bugfix:
u->htisorder kann gelöscht werden (z.b. NACH). Daher muss ein pointer auf
einen befehl in einer message immer auf den u->order Eintrag zeigen, damit er
zeit der auswertung noch existiert.
findorder(u, u->thisorder) tut das. Ist an mehreren Stellen nicht benutzt
worden. assert eingebaut, das das prüft.
* RESERVE_DONATIONS
Gegenstände, die von einer anderen Partei übergeben wurden, werden nicht
reserviert.
* TWOPASS_GIVE:
GIB Befehle werden zuerst an fremde Einheiten, danach in einem zweiten
Durchlauf an eigene Einheiten, ausgeführt.
* msg_message
An einigen messages ausprobiert, ob man die gleiche Message mehreren
Parteien einhängen kann - klappt, spart Speicher.
Allerdings fehlt dazu ein ordentliches memory-management (refcounter)
2001-05-10 07:50:52 +02:00
|
|
|
|
int road = 0;
|
|
|
|
|
struct message * msg;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if (u->building == b) {
|
|
|
|
|
int loss = 0;
|
|
|
|
|
|
2007-06-20 02:34:02 +02:00
|
|
|
|
fset(u->faction, FFL_MARK);
|
2003-07-29 11:48:03 +02:00
|
|
|
|
freset(u, UFL_OWNER);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
leave(r,u);
|
|
|
|
|
n = u->number;
|
|
|
|
|
for (i = 0; i < n; i++) {
|
2007-06-20 02:34:02 +02:00
|
|
|
|
if (rng_double() >= COLLAPSE_SURVIVAL) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
++loss;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
scale_number(u, u->number - loss);
|
|
|
|
|
opfer += loss;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
* möglichen Exploit beseitigt:
GIB xyz EINHEIT
GIB 0 ALLES SILBER
--> ALLE Befehle der übergebenen Einheit werden gelöscht.
* neue Funktion (convenience): ucansee(f, u1, u2)
liefert u1, wenn cansee(f,u1), sonst u2
* neue mistakes eingefügt uns übersetzt
* message bugfix:
u->htisorder kann gelöscht werden (z.b. NACH). Daher muss ein pointer auf
einen befehl in einer message immer auf den u->order Eintrag zeigen, damit er
zeit der auswertung noch existiert.
findorder(u, u->thisorder) tut das. Ist an mehreren Stellen nicht benutzt
worden. assert eingebaut, das das prüft.
* RESERVE_DONATIONS
Gegenstände, die von einer anderen Partei übergeben wurden, werden nicht
reserviert.
* TWOPASS_GIVE:
GIB Befehle werden zuerst an fremde Einheiten, danach in einem zweiten
Durchlauf an eigene Einheiten, ausgeführt.
* msg_message
An einigen messages ausprobiert, ob man die gleiche Message mehreren
Parteien einhängen kann - klappt, spart Speicher.
Allerdings fehlt dazu ein ordentliches memory-management (refcounter)
2001-05-10 07:50:52 +02:00
|
|
|
|
msg = msg_message("buildingcrash", "region building opfer road", r, b, opfer, road);
|
|
|
|
|
add_message(&r->msgs, msg);
|
2001-02-10 20:24:05 +01:00
|
|
|
|
for (u=r->units; u; u=u->next) {
|
|
|
|
|
faction * f = u->faction;
|
2007-06-20 02:34:02 +02:00
|
|
|
|
if (fval(f, FFL_MARK)) {
|
|
|
|
|
freset(u->faction, FFL_MARK);
|
* möglichen Exploit beseitigt:
GIB xyz EINHEIT
GIB 0 ALLES SILBER
--> ALLE Befehle der übergebenen Einheit werden gelöscht.
* neue Funktion (convenience): ucansee(f, u1, u2)
liefert u1, wenn cansee(f,u1), sonst u2
* neue mistakes eingefügt uns übersetzt
* message bugfix:
u->htisorder kann gelöscht werden (z.b. NACH). Daher muss ein pointer auf
einen befehl in einer message immer auf den u->order Eintrag zeigen, damit er
zeit der auswertung noch existiert.
findorder(u, u->thisorder) tut das. Ist an mehreren Stellen nicht benutzt
worden. assert eingebaut, das das prüft.
* RESERVE_DONATIONS
Gegenstände, die von einer anderen Partei übergeben wurden, werden nicht
reserviert.
* TWOPASS_GIVE:
GIB Befehle werden zuerst an fremde Einheiten, danach in einem zweiten
Durchlauf an eigene Einheiten, ausgeführt.
* msg_message
An einigen messages ausprobiert, ob man die gleiche Message mehreren
Parteien einhängen kann - klappt, spart Speicher.
Allerdings fehlt dazu ein ordentliches memory-management (refcounter)
2001-05-10 07:50:52 +02:00
|
|
|
|
add_message(&f->msgs, msg);
|
2001-02-10 20:24:05 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2001-05-20 23:47:56 +02:00
|
|
|
|
msg_release(msg);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
destroy_building(b);
|
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
void
|
2006-08-12 21:15:16 +02:00
|
|
|
|
maintain_buildings(region * r, boolean crash)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2007-02-23 22:18:35 +01:00
|
|
|
|
building **bp = &r->buildings;
|
|
|
|
|
while (*bp) {
|
|
|
|
|
building * b = *bp;
|
2006-08-12 21:15:16 +02:00
|
|
|
|
boolean maintained = maintain(b, !crash);
|
|
|
|
|
|
|
|
|
|
/* the second time, send a message */
|
|
|
|
|
if (crash) {
|
2007-06-20 02:34:02 +02:00
|
|
|
|
if (!maintained && (rng_double() < COLLAPSE_CHANCE)) {
|
2007-02-23 22:18:35 +01:00
|
|
|
|
gebaeude_stuerzt_ein(r, b);
|
|
|
|
|
continue;
|
2006-08-12 21:15:16 +02:00
|
|
|
|
} else if (!fval(b, BLD_WORKING)) {
|
|
|
|
|
unit * u = buildingowner(r, b);
|
|
|
|
|
const char * msgtype = maintained?"maintenance_nowork":"maintenance_none";
|
2007-02-23 22:18:35 +01:00
|
|
|
|
struct message * msg = msg_message(msgtype, "building", b);
|
2006-08-12 21:15:16 +02:00
|
|
|
|
|
|
|
|
|
if (u) {
|
2007-02-23 22:18:35 +01:00
|
|
|
|
add_message(&u->faction->msgs, msg);
|
|
|
|
|
r_addmessage(r, u->faction, msg);
|
2006-08-12 21:15:16 +02:00
|
|
|
|
} else {
|
|
|
|
|
add_message(&r->msgs, msg);
|
2005-07-10 23:22:54 +02:00
|
|
|
|
}
|
2006-08-12 21:15:16 +02:00
|
|
|
|
msg_release(msg);
|
2005-07-10 23:22:54 +02:00
|
|
|
|
}
|
2006-08-12 21:15:16 +02:00
|
|
|
|
}
|
2007-02-23 22:18:35 +01:00
|
|
|
|
bp=&b->next;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2007-05-28 18:03:48 +02:00
|
|
|
|
static int
|
|
|
|
|
recruit_archetype(unit * u, order * ord)
|
|
|
|
|
{
|
2007-05-28 23:10:59 +02:00
|
|
|
|
int want;
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char * s;
|
2007-05-28 18:03:48 +02:00
|
|
|
|
|
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
2007-06-26 11:32:28 +02:00
|
|
|
|
want = getuint();
|
2007-05-28 18:03:48 +02:00
|
|
|
|
s = getstrtoken();
|
2007-05-28 23:10:59 +02:00
|
|
|
|
if (want>0 && s && s[0]) {
|
|
|
|
|
int n = want;
|
2007-05-28 18:03:48 +02:00
|
|
|
|
const archetype * arch = find_archetype(s, u->faction->locale);
|
2007-05-28 19:31:40 +02:00
|
|
|
|
attrib * a = NULL;
|
2007-05-28 18:03:48 +02:00
|
|
|
|
|
|
|
|
|
if (u->number>0) {
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "unit_must_be_new", ""));
|
|
|
|
|
/* TODO: error message */
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (arch==NULL) {
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "unknown_archetype", "name", s));
|
|
|
|
|
/* TODO: error message */
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2007-05-30 03:21:09 +02:00
|
|
|
|
if (arch->rules) {
|
|
|
|
|
/* Simple allow/deny style restrictions for archetypes (let only humans
|
|
|
|
|
* recruit gamedesigners, etc). These need to be more powerful to be
|
|
|
|
|
* useful, and the current way they are implemented is not, but the
|
|
|
|
|
* general idea strikes me as good. Also, feedback should be configurable
|
|
|
|
|
* for each failed rule.
|
|
|
|
|
*/
|
|
|
|
|
int k;
|
|
|
|
|
for (k=0;arch->rules[k].property;++k) {
|
|
|
|
|
boolean match = false;
|
|
|
|
|
if (arch->rules[k].value[0]=='*') match = true;
|
|
|
|
|
else if (strcmp(arch->rules[k].property, "race")==0) {
|
|
|
|
|
const race * rc = rc_find(arch->rules[k].value);
|
|
|
|
|
assert(rc);
|
|
|
|
|
if (rc==u->race) match = true;
|
|
|
|
|
} else if (strcmp(arch->rules[k].property, "building")==0) {
|
|
|
|
|
const building_type * btype = bt_find(arch->rules[k].value);
|
|
|
|
|
assert(btype);
|
|
|
|
|
if (u->building && u->building->type==btype) match = true;
|
|
|
|
|
}
|
|
|
|
|
if (match) {
|
|
|
|
|
if (arch->rules[k].allow) break;
|
|
|
|
|
else {
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "recruit_rule_fail",
|
|
|
|
|
"property value", arch->rules[k].property, arch->rules[k].value));
|
|
|
|
|
/* TODO: error message */
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-05-28 19:31:40 +02:00
|
|
|
|
if (arch->btype) {
|
|
|
|
|
if (u->building==NULL || u->building->type!=arch->btype) {
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "unit_must_be_in_building", "type", arch->btype));
|
|
|
|
|
/* TODO: error message */
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
if (arch->size) {
|
|
|
|
|
int maxsize = u->building->size;
|
|
|
|
|
attrib * a = a_find(u->building->attribs, &at_recruit);
|
|
|
|
|
if (a!=NULL) {
|
|
|
|
|
maxsize -= a->data.i;
|
|
|
|
|
}
|
2007-05-28 19:32:15 +02:00
|
|
|
|
n = min(maxsize/arch->size, n);
|
2007-05-28 19:31:40 +02:00
|
|
|
|
if (n<=0) {
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "recruit_capacity_exhausted", "building", u->building));
|
|
|
|
|
/* TODO: error message */
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-05-28 18:03:48 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
n = build(u, arch->ctype, 0, n);
|
|
|
|
|
if (n>0) {
|
2007-05-28 21:45:08 +02:00
|
|
|
|
set_number(u, n);
|
2007-05-28 18:03:48 +02:00
|
|
|
|
equip_unit(u, arch->equip);
|
2007-05-28 21:45:08 +02:00
|
|
|
|
u->hp = n * unit_max_hp(u);
|
2007-05-28 19:31:40 +02:00
|
|
|
|
if (arch->size) {
|
|
|
|
|
if (a==NULL) a = a_add(&u->building->attribs, a_new(&at_recruit));
|
|
|
|
|
a->data.i += n*arch->size;
|
|
|
|
|
}
|
2007-05-28 22:04:42 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("recruit_archetype",
|
2007-05-28 23:10:59 +02:00
|
|
|
|
"unit amount archetype", u, n, arch->name[n==1]));
|
2007-05-28 18:03:48 +02:00
|
|
|
|
return n;
|
|
|
|
|
} else switch(n) {
|
|
|
|
|
case ENOMATERIALS:
|
2007-05-28 23:10:59 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_materials_required(u, ord, arch->ctype, want));
|
2007-05-28 19:52:24 +02:00
|
|
|
|
break;
|
2007-05-28 18:03:48 +02:00
|
|
|
|
case ELOWSKILL:
|
|
|
|
|
case ENEEDSKILL:
|
|
|
|
|
/* no skill, or not enough skill points to build */
|
|
|
|
|
cmistake(u, ord, 50, MSG_PRODUCE);
|
2007-05-28 19:52:24 +02:00
|
|
|
|
break;
|
2007-05-28 18:03:48 +02:00
|
|
|
|
default:
|
|
|
|
|
assert(!"unhandled return value from build() in recruit_archetype");
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2007-05-30 03:21:09 +02:00
|
|
|
|
int
|
2007-05-28 18:03:48 +02:00
|
|
|
|
recruit_classic(void)
|
|
|
|
|
{
|
|
|
|
|
static int value = -1;
|
|
|
|
|
if (value<0) {
|
|
|
|
|
const char * str = get_param(global.parameters, "recruit.classic");
|
|
|
|
|
value = str?atoi(str):1;
|
|
|
|
|
}
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
|
2007-05-30 03:21:09 +02:00
|
|
|
|
int
|
2007-05-28 18:03:48 +02:00
|
|
|
|
recruit_archetypes(void)
|
|
|
|
|
{
|
|
|
|
|
static int value = -1;
|
|
|
|
|
if (value<0) {
|
|
|
|
|
const char * str = get_param(global.parameters, "recruit.archetypes");
|
|
|
|
|
value = str?atoi(str):0;
|
|
|
|
|
}
|
|
|
|
|
return value;
|
|
|
|
|
}
|
2004-06-21 18:45:27 +02:00
|
|
|
|
|
2002-08-26 16:16:16 +02:00
|
|
|
|
void
|
2006-08-12 21:15:16 +02:00
|
|
|
|
economics(region *r)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-05-22 00:10:18 +02:00
|
|
|
|
unit *u;
|
2006-08-12 21:15:16 +02:00
|
|
|
|
request *recruitorders = NULL;
|
2004-05-22 00:10:18 +02:00
|
|
|
|
|
|
|
|
|
/* Geben vor Selbstmord (doquit)! Hier alle unmittelbaren Befehle.
|
2006-08-12 21:15:16 +02:00
|
|
|
|
* Rekrutieren vor allen Einnahmequellen. Bewachen JA vor Steuern
|
|
|
|
|
* eintreiben. */
|
|
|
|
|
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
order * ord;
|
|
|
|
|
boolean destroyed = false;
|
2007-02-11 12:10:53 +01:00
|
|
|
|
if (u->number>0) {
|
|
|
|
|
for (ord = u->orders; ord; ord = ord->next) {
|
|
|
|
|
switch (get_keyword(ord)) {
|
2006-08-12 21:15:16 +02:00
|
|
|
|
case K_DESTROY:
|
|
|
|
|
if (!destroyed) {
|
|
|
|
|
if (destroy_cmd(u, ord)!=0) ord = NULL;
|
|
|
|
|
destroyed = true;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-08-12 21:15:16 +02:00
|
|
|
|
case K_GIVE:
|
|
|
|
|
case K_LIEFERE:
|
|
|
|
|
give_cmd(u, ord);
|
|
|
|
|
break;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-08-12 21:15:16 +02:00
|
|
|
|
case K_FORGET:
|
|
|
|
|
forget_cmd(u, ord);
|
2004-05-22 00:10:18 +02:00
|
|
|
|
break;
|
2006-08-12 21:15:16 +02:00
|
|
|
|
|
2007-02-11 12:10:53 +01:00
|
|
|
|
}
|
|
|
|
|
if (u->orders==NULL) break;
|
2004-05-22 00:10:18 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-08-12 21:15:16 +02:00
|
|
|
|
/* RECRUIT orders */
|
|
|
|
|
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
2007-10-11 23:48:31 +02:00
|
|
|
|
order * ord;
|
2007-05-28 18:03:48 +02:00
|
|
|
|
if (!recruit_classic()) {
|
|
|
|
|
if (u->number>0) continue;
|
|
|
|
|
}
|
2007-10-11 23:48:31 +02:00
|
|
|
|
for (ord = u->orders; ord; ord = ord->next) {
|
2006-08-12 21:15:16 +02:00
|
|
|
|
if (get_keyword(ord) == K_RECRUIT) {
|
2007-05-28 18:03:48 +02:00
|
|
|
|
if (recruit_archetypes()) {
|
|
|
|
|
if (recruit_archetype(u, ord)>=0) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (recruit_classic()) {
|
|
|
|
|
recruit(u, ord, &recruitorders);
|
|
|
|
|
}
|
2006-08-12 21:15:16 +02:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (recruitorders) expandrecruit(r, recruitorders);
|
|
|
|
|
remove_empty_units_in_region(r);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
|
2002-08-26 16:16:16 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
manufacture(unit * u, const item_type * itype, int want)
|
|
|
|
|
{
|
2004-06-12 13:19:17 +02:00
|
|
|
|
int n;
|
|
|
|
|
int skill;
|
|
|
|
|
int minskill = itype->construction->minskill;
|
|
|
|
|
skill_t sk = itype->construction->skill;
|
|
|
|
|
|
|
|
|
|
skill = effskill(u, sk);
|
|
|
|
|
skill = skillmod(itype->rtype->attribs, u, u->region, sk, skill, SMF_PRODUCTION);
|
|
|
|
|
|
|
|
|
|
if (skill < 0) {
|
|
|
|
|
/* an error occured */
|
|
|
|
|
int err = -skill;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, u->thisorder, err, MSG_PRODUCE);
|
2004-06-12 13:19:17 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-05-28 23:10:59 +02:00
|
|
|
|
if (want==0) {
|
|
|
|
|
want = maxbuild(u, itype->construction);
|
|
|
|
|
}
|
2004-06-12 13:19:17 +02:00
|
|
|
|
n = build(u, itype->construction, 0, want);
|
|
|
|
|
switch (n) {
|
|
|
|
|
case ENEEDSKILL:
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2004-06-21 18:45:27 +02:00
|
|
|
|
msg_feedback(u, u->thisorder, "skill_needed", "skill", sk));
|
2004-06-12 13:19:17 +02:00
|
|
|
|
return;
|
|
|
|
|
case ELOWSKILL:
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2004-06-21 18:45:27 +02:00
|
|
|
|
msg_feedback(u, u->thisorder, "manufacture_skills", "skill minskill product",
|
2004-06-12 13:19:17 +02:00
|
|
|
|
sk, minskill, itype->rtype, 1));
|
|
|
|
|
return;
|
|
|
|
|
case ENOMATERIALS:
|
2007-05-28 23:10:59 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_materials_required(u, u->thisorder, itype->construction, want));
|
2007-05-28 18:03:48 +02:00
|
|
|
|
return;
|
2004-06-12 13:19:17 +02:00
|
|
|
|
}
|
|
|
|
|
if (n>0) {
|
|
|
|
|
i_change(&u->items, itype, n);
|
|
|
|
|
if (want==INT_MAX) want = n;
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("manufacture",
|
|
|
|
|
"unit region amount wanted resource", u, u->region, n, want, itype->rtype));
|
|
|
|
|
} else {
|
2007-05-22 19:09:23 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, u->thisorder, "error_cannotmake", ""));
|
2004-06-12 13:19:17 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
typedef struct allocation {
|
|
|
|
|
struct allocation * next;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
int want, get;
|
|
|
|
|
double save;
|
|
|
|
|
unsigned int flags;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
unit * unit;
|
|
|
|
|
} allocation;
|
|
|
|
|
#define new_allocation() calloc(sizeof(allocation), 1)
|
|
|
|
|
#define free_allocation(a) free(a)
|
|
|
|
|
|
|
|
|
|
typedef struct allocation_list {
|
|
|
|
|
struct allocation_list * next;
|
|
|
|
|
allocation * data;
|
|
|
|
|
const resource_type * type;
|
|
|
|
|
} allocation_list;
|
|
|
|
|
|
|
|
|
|
static allocation_list * allocations;
|
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
static boolean
|
|
|
|
|
can_guard(const unit * guard, const unit * u)
|
|
|
|
|
{
|
|
|
|
|
if (fval(guard, UFL_ISNEW)) return false;
|
|
|
|
|
if (guard->number<=0 || !cansee(guard->faction, guard->region, u, 0)) return false;
|
2007-01-01 13:06:29 +01:00
|
|
|
|
if (besieged(guard) || !(fval(guard->race, RCF_UNARMEDGUARD) || armedmen(guard))) return false;
|
2006-02-11 17:11:16 +01:00
|
|
|
|
|
|
|
|
|
return !alliedunit(guard, u->faction, HELP_GUARD);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
AFL_DONE = 1<<0,
|
|
|
|
|
AFL_LOWSKILL = 1<<1
|
|
|
|
|
};
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
allocate_resource(unit * u, const resource_type * rtype, int want)
|
|
|
|
|
{
|
2007-01-01 13:06:29 +01:00
|
|
|
|
const item_type * itype = resource2item(rtype);
|
|
|
|
|
region * r = u->region;
|
|
|
|
|
int busy = u->number;
|
|
|
|
|
int dm = 0;
|
|
|
|
|
allocation_list * alist;
|
|
|
|
|
allocation * al;
|
2006-02-11 17:11:16 +01:00
|
|
|
|
attrib * a = a_find(rtype->attribs, &at_resourcelimit);
|
|
|
|
|
resource_limit * rdata = (resource_limit*)a->data.v;
|
|
|
|
|
int amount, skill;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-01-01 13:06:29 +01:00
|
|
|
|
/* momentan kann man keine ressourcen abbauen, wenn man daf<61>r
|
|
|
|
|
* Materialverbrauch hat: */
|
|
|
|
|
assert(itype!=NULL && (itype->construction==NULL || itype->construction->materials==NULL));
|
2006-02-11 17:11:16 +01:00
|
|
|
|
assert(rdata!=NULL);
|
2007-01-01 13:06:29 +01:00
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
if (rdata->limit!=NULL) {
|
|
|
|
|
int avail = rdata->limit(r, rtype);
|
|
|
|
|
if (avail<=0) {
|
|
|
|
|
cmistake(u, u->thisorder, 121, MSG_PRODUCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-01-01 13:06:29 +01:00
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
if (itype == olditemtype[I_LAEN]) {
|
2007-01-01 13:06:29 +01:00
|
|
|
|
struct building * b = inside_building(u);
|
|
|
|
|
const struct building_type * btype = b?b->type:NULL;
|
|
|
|
|
if (btype != bt_find("mine")) {
|
|
|
|
|
cmistake(u, u->thisorder, 104, MSG_PRODUCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (besieged(u)) {
|
|
|
|
|
cmistake(u, u->thisorder, 60, MSG_PRODUCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
if (rdata->guard!=0) {
|
|
|
|
|
unit * u2;
|
2007-01-01 13:06:29 +01:00
|
|
|
|
for (u2 = r->units; u2; u2 = u2->next) {
|
|
|
|
|
if ((getguard(u2) & rdata->guard) && can_guard(u2, u)) {
|
2006-02-11 17:11:16 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2007-01-01 13:06:29 +01:00
|
|
|
|
msg_feedback(u, u->thisorder, "region_guarded", "guard", u2));
|
2006-02-11 17:11:16 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2007-01-01 13:06:29 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Bergw<67>chter k<>nnen Abbau von Eisen/Laen durch Bewachen verhindern.
|
|
|
|
|
* Als magische Wesen 'sehen' Bergw<EFBFBD>chter alles und werden durch
|
|
|
|
|
* Belagerung nicht aufgehalten. (Ansonsten wie oben bei Elfen anpassen).
|
|
|
|
|
*/
|
|
|
|
|
if (itype == olditemtype[I_IRON] || itype == olditemtype[I_LAEN]) {
|
2006-02-11 17:11:16 +01:00
|
|
|
|
unit * u2;
|
2007-01-01 13:06:29 +01:00
|
|
|
|
for (u2 = r->units; u2; u2 = u2->next ) {
|
|
|
|
|
if (getguard(u) & GUARD_MINING
|
|
|
|
|
&& !fval(u2, UFL_ISNEW)
|
|
|
|
|
&& u2->number
|
|
|
|
|
&& !alliedunit(u2, u->faction, HELP_GUARD))
|
|
|
|
|
{
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, u->thisorder, "region_guarded", "guard", u2));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assert(itype->construction->skill!=0 || "limited resource needs a required skill for making it");
|
|
|
|
|
skill = eff_skill(u, itype->construction->skill, u->region);
|
|
|
|
|
if (skill == 0) {
|
|
|
|
|
skill_t sk = itype->construction->skill;
|
|
|
|
|
add_message(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, u->thisorder, "skill_needed", "skill", sk));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (skill < itype->construction->minskill) {
|
|
|
|
|
skill_t sk = itype->construction->skill;
|
|
|
|
|
add_message(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, u->thisorder, "manufacture_skills", "skill minskill product",
|
|
|
|
|
sk, itype->construction->minskill, itype->rtype));
|
|
|
|
|
return;
|
|
|
|
|
} else {
|
|
|
|
|
struct building * b = inside_building(u);
|
|
|
|
|
const struct building_type * btype = b?b->type:NULL;
|
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
if (rdata->modifiers) {
|
|
|
|
|
resource_mod * mod = rdata->modifiers;
|
|
|
|
|
for (;mod->flags!=0;++mod) {
|
|
|
|
|
if (mod->flags & RMF_SKILL) {
|
|
|
|
|
if (mod->btype==NULL || mod->btype==btype) {
|
|
|
|
|
if (mod->race==NULL || mod->race==u->race) {
|
|
|
|
|
skill += mod->value.i;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-01-01 13:06:29 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
amount = skill * u->number;
|
|
|
|
|
/* nun ist amount die Gesamtproduktion der Einheit (in punkten) */
|
|
|
|
|
|
|
|
|
|
/* mit Flinkfingerring verzehnfacht sich die Produktion */
|
|
|
|
|
amount += skill * min(u->number, get_item(u,I_RING_OF_NIMBLEFINGER)) * 9;
|
|
|
|
|
|
|
|
|
|
/* Schaffenstrunk: */
|
|
|
|
|
if ((dm = get_effect(u, oldpotiontype[P_DOMORE])) != 0) {
|
|
|
|
|
dm = min(dm, u->number);
|
|
|
|
|
change_effect(u, oldpotiontype[P_DOMORE], -dm);
|
|
|
|
|
amount += dm * skill; /* dm Personen produzieren doppelt */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
amount /= itype->construction->minskill;
|
|
|
|
|
|
|
|
|
|
/* Limitierung durch Parameter m. */
|
|
|
|
|
if (want > 0 && want < amount) amount = want;
|
|
|
|
|
|
|
|
|
|
busy = (amount + skill - 1) / skill; /* wieviel leute tun etwas? */
|
|
|
|
|
|
|
|
|
|
alist = allocations;
|
|
|
|
|
while (alist && alist->type!=rtype) alist = alist->next;
|
|
|
|
|
if (!alist) {
|
|
|
|
|
alist = calloc(sizeof(struct allocation_list), 1);
|
|
|
|
|
alist->next = allocations;
|
|
|
|
|
alist->type = rtype;
|
|
|
|
|
allocations = alist;
|
|
|
|
|
}
|
|
|
|
|
al = new_allocation();
|
|
|
|
|
al->want = amount;
|
|
|
|
|
al->save = 1.0;
|
|
|
|
|
al->next = alist->data;
|
|
|
|
|
al->unit = u;
|
|
|
|
|
alist->data = al;
|
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
if (rdata->modifiers) {
|
|
|
|
|
struct building * b = inside_building(u);
|
|
|
|
|
const struct building_type * btype = b?b->type:NULL;
|
2007-01-01 13:06:29 +01:00
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
resource_mod * mod = rdata->modifiers;
|
|
|
|
|
for (;mod->flags!=0;++mod) {
|
2006-02-12 12:05:43 +01:00
|
|
|
|
if (mod->flags & RMF_SAVEMATERIAL) {
|
2006-02-11 17:11:16 +01:00
|
|
|
|
if (mod->btype==NULL || mod->btype==btype) {
|
|
|
|
|
if (mod->race==NULL || mod->race==u->race) {
|
|
|
|
|
al->save *= mod->value.f;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if (itype==olditemtype[I_IRON]) {
|
2007-01-01 13:06:29 +01:00
|
|
|
|
struct building * b = inside_building(u);
|
|
|
|
|
const struct building_type * btype = b?b->type:NULL;
|
|
|
|
|
if (btype==bt_find("mine"))
|
|
|
|
|
al->save *= 0.5;
|
|
|
|
|
if (u->race == new_race[RC_DWARF]) {
|
|
|
|
|
al->save *= 0.75;
|
|
|
|
|
}
|
|
|
|
|
} else if (itype==olditemtype[I_STONE]) {
|
|
|
|
|
struct building * b = inside_building(u);
|
|
|
|
|
const struct building_type * btype = b?b->type:NULL;
|
|
|
|
|
if (btype==bt_find("quarry"))
|
|
|
|
|
al->save = al->save*0.5;
|
|
|
|
|
if (u->race == new_race[RC_TROLL])
|
|
|
|
|
al->save = al->save*0.75;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
static int
|
|
|
|
|
required(int want, double save)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
2006-02-11 17:11:16 +01:00
|
|
|
|
int norders = (int)(want * save);
|
|
|
|
|
if (norders < want*save) ++norders;
|
|
|
|
|
return norders;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2006-02-11 17:11:16 +01:00
|
|
|
|
leveled_allocation(const resource_type * rtype, region * r, allocation * alist)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
2006-02-11 17:11:16 +01:00
|
|
|
|
const item_type * itype = resource2item(rtype);
|
|
|
|
|
rawmaterial * rm = rm_get(r, rtype);
|
|
|
|
|
int need;
|
|
|
|
|
boolean first = true;
|
|
|
|
|
|
|
|
|
|
if (rm!=NULL) {
|
|
|
|
|
do {
|
|
|
|
|
int avail = rm->amount;
|
|
|
|
|
int norders = 0;
|
|
|
|
|
allocation * al;
|
|
|
|
|
|
|
|
|
|
if(avail <= 0) {
|
|
|
|
|
for (al=alist;al;al=al->next) {
|
|
|
|
|
al->get = 0;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2001-12-15 13:26:04 +01:00
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
assert(avail>0);
|
|
|
|
|
|
|
|
|
|
for (al=alist;al;al=al->next) if (!fval(al, AFL_DONE)) {
|
|
|
|
|
int req = required(al->want-al->get, al->save);
|
|
|
|
|
assert(al->get<=al->want && al->get >= 0);
|
|
|
|
|
if (eff_skill(al->unit, itype->construction->skill, r)
|
|
|
|
|
>= rm->level+itype->construction->minskill-1) {
|
|
|
|
|
if (req) {
|
|
|
|
|
norders += req;
|
|
|
|
|
} else {
|
|
|
|
|
fset(al, AFL_DONE);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
fset(al, AFL_DONE);
|
|
|
|
|
if (first) fset(al, AFL_LOWSKILL);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
need = norders;
|
|
|
|
|
|
|
|
|
|
avail = min(avail, norders);
|
|
|
|
|
if (need>0) {
|
|
|
|
|
int use = 0;
|
|
|
|
|
for (al=alist;al;al=al->next) if (!fval(al, AFL_DONE)) {
|
|
|
|
|
if (avail > 0) {
|
|
|
|
|
int want = required(al->want-al->get, al->save);
|
|
|
|
|
int x = avail*want/norders;
|
|
|
|
|
/* Wenn Rest, dann w<>rfeln, ob ich was bekomme: */
|
2006-02-19 23:43:56 +01:00
|
|
|
|
if (rng_int() % norders < (avail*want) % norders)
|
2006-02-11 17:11:16 +01:00
|
|
|
|
++x;
|
|
|
|
|
avail -= x;
|
|
|
|
|
use += x;
|
|
|
|
|
norders -= want;
|
|
|
|
|
need -= x;
|
|
|
|
|
al->get = min(al->want, al->get+(int)(x/al->save));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (use) {
|
|
|
|
|
assert(use<=rm->amount);
|
|
|
|
|
rm->type->use(rm, r, use);
|
|
|
|
|
}
|
|
|
|
|
assert(avail==0 || norders==0);
|
|
|
|
|
}
|
|
|
|
|
first = false;
|
|
|
|
|
} while (need>0);
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2006-02-11 17:11:16 +01:00
|
|
|
|
attrib_allocation(const resource_type * rtype, region * r, allocation * alist)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
2006-02-11 17:11:16 +01:00
|
|
|
|
allocation * al;
|
|
|
|
|
int norders = 0;
|
|
|
|
|
attrib * a = a_find(rtype->attribs, &at_resourcelimit);
|
|
|
|
|
resource_limit * rdata = (resource_limit*)a->data.v;
|
|
|
|
|
int avail = rdata->value;
|
|
|
|
|
|
|
|
|
|
for (al=alist;al;al=al->next) {
|
|
|
|
|
norders += required(al->want, al->save);
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
if (rdata->limit) {
|
|
|
|
|
avail = rdata->limit(r, rtype);
|
|
|
|
|
if (avail < 0) avail = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
avail = min(avail, norders);
|
|
|
|
|
for (al=alist;al;al=al->next) {
|
|
|
|
|
if (avail > 0) {
|
|
|
|
|
int want = required(al->want, al->save);
|
|
|
|
|
int x = avail*want/norders;
|
|
|
|
|
/* Wenn Rest, dann w<>rfeln, ob ich was bekomme: */
|
2006-02-19 23:43:56 +01:00
|
|
|
|
if (rng_int() % norders < (avail*want) % norders)
|
2006-02-11 17:11:16 +01:00
|
|
|
|
++x;
|
|
|
|
|
avail -= x;
|
|
|
|
|
norders -= want;
|
|
|
|
|
al->get = min(al->want, (int)(x/al->save));
|
|
|
|
|
if (rdata->produce) {
|
|
|
|
|
int use = required(al->get, al->save);
|
|
|
|
|
if (use) rdata->produce(r, rtype, use);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
assert(avail==0 || norders==0);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2006-02-11 17:11:16 +01:00
|
|
|
|
typedef void (*allocate_function)(const resource_type *, struct region *, struct allocation *);
|
|
|
|
|
|
|
|
|
|
static allocate_function
|
|
|
|
|
get_allocator(const struct resource_type * rtype)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
2006-02-11 17:11:16 +01:00
|
|
|
|
attrib * a = a_find(rtype->attribs, &at_resourcelimit);
|
|
|
|
|
|
|
|
|
|
if (a!=NULL) {
|
|
|
|
|
resource_limit * rdata = (resource_limit*)a->data.v;
|
|
|
|
|
if (rdata->value>0 || rdata->limit!=NULL) {
|
|
|
|
|
return attrib_allocation;
|
|
|
|
|
}
|
|
|
|
|
return leveled_allocation;
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2007-02-04 14:13:48 +01:00
|
|
|
|
void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
split_allocations(region * r)
|
|
|
|
|
{
|
2007-02-04 14:13:48 +01:00
|
|
|
|
allocation_list ** p_alist=&allocations;
|
2007-06-20 02:34:02 +02:00
|
|
|
|
freset(r, RF_SELECT);
|
2007-02-04 14:13:48 +01:00
|
|
|
|
while (*p_alist) {
|
|
|
|
|
allocation_list * alist = *p_alist;
|
|
|
|
|
const resource_type * rtype = alist->type;
|
|
|
|
|
allocate_function alloc = get_allocator(rtype);
|
|
|
|
|
const item_type * itype = resource2item(rtype);
|
|
|
|
|
allocation ** p_al = &alist->data;
|
|
|
|
|
|
2007-06-20 02:34:02 +02:00
|
|
|
|
freset(r, RF_SELECT);
|
2007-02-04 14:13:48 +01:00
|
|
|
|
alloc(rtype, r, alist->data);
|
|
|
|
|
|
|
|
|
|
while (*p_al) {
|
|
|
|
|
allocation * al = *p_al;
|
|
|
|
|
if (al->get) {
|
|
|
|
|
assert(itype || !"not implemented for non-items");
|
|
|
|
|
i_change(&al->unit->items, itype, al->get);
|
|
|
|
|
produceexp(al->unit, itype->construction->skill, al->unit->number);
|
2007-06-20 02:34:02 +02:00
|
|
|
|
fset(r, RF_SELECT);
|
2007-02-04 14:13:48 +01:00
|
|
|
|
}
|
|
|
|
|
if (al->want==INT_MAX) al->want = al->get;
|
|
|
|
|
if (fval(al, AFL_LOWSKILL)) {
|
|
|
|
|
ADDMSG(&al->unit->faction->msgs,
|
|
|
|
|
msg_message("produce_lowskill", "unit region resource",
|
|
|
|
|
al->unit, al->unit->region, rtype));
|
|
|
|
|
} else {
|
|
|
|
|
ADDMSG(&al->unit->faction->msgs, msg_message("produce",
|
2006-01-02 23:29:07 +01:00
|
|
|
|
"unit region amount wanted resource",
|
2007-02-04 14:13:48 +01:00
|
|
|
|
al->unit, al->unit->region, al->get, al->want, rtype));
|
|
|
|
|
}
|
|
|
|
|
*p_al=al->next;
|
|
|
|
|
free_allocation(al);
|
|
|
|
|
}
|
|
|
|
|
*p_alist=alist->next;
|
|
|
|
|
free(alist);
|
|
|
|
|
}
|
|
|
|
|
allocations = NULL;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
create_potion(unit * u, const potion_type * ptype, int want)
|
|
|
|
|
{
|
|
|
|
|
int built;
|
|
|
|
|
|
2007-05-28 23:10:59 +02:00
|
|
|
|
if (want==0) {
|
|
|
|
|
want = maxbuild(u, ptype->itype->construction);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
built = build(u, ptype->itype->construction, 0, want);
|
|
|
|
|
switch (built) {
|
2002-04-21 19:22:48 +02:00
|
|
|
|
case ELOWSKILL:
|
|
|
|
|
case ENEEDSKILL:
|
|
|
|
|
/* no skill, or not enough skill points to build */
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, u->thisorder, 50, MSG_PRODUCE);
|
2002-04-21 19:22:48 +02:00
|
|
|
|
break;
|
|
|
|
|
case ECOMPLETE:
|
|
|
|
|
assert(0);
|
|
|
|
|
break;
|
|
|
|
|
case ENOMATERIALS:
|
|
|
|
|
/* something missing from the list of materials */
|
2007-05-28 23:10:59 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_materials_required(u, u->thisorder, ptype->itype->construction, want));
|
2007-05-28 18:03:48 +02:00
|
|
|
|
return;
|
2002-04-21 19:22:48 +02:00
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
i_change(&u->items, ptype->itype, built);
|
|
|
|
|
if (want==INT_MAX) want = built;
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("manufacture",
|
2003-09-21 10:47:28 +02:00
|
|
|
|
"unit region amount wanted resource", u, u->region, built, want, ptype->itype->rtype));
|
2002-04-21 19:22:48 +02:00
|
|
|
|
break;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2002-04-23 19:13:45 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
create_item(unit * u, const item_type * itype, int want)
|
|
|
|
|
{
|
2007-03-25 01:04:40 +01:00
|
|
|
|
if (fval(itype->rtype, RTF_LIMITED)) {
|
2002-04-23 19:13:45 +02:00
|
|
|
|
#if GUARD_DISABLES_PRODUCTION == 1
|
2007-03-25 01:04:40 +01:00
|
|
|
|
if(is_guarded(u->region, u, GUARD_PRODUCE)) {
|
|
|
|
|
cmistake(u, u->thisorder, 70, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2002-04-21 19:22:48 +02:00
|
|
|
|
#endif
|
2007-03-25 01:04:40 +01:00
|
|
|
|
allocate_resource(u, itype->rtype, want);
|
|
|
|
|
} else {
|
|
|
|
|
const potion_type * ptype = resource2potion(itype->rtype);
|
|
|
|
|
if (ptype!=NULL) create_potion(u, ptype, want);
|
|
|
|
|
else if (itype->construction && itype->construction->materials) manufacture(u, itype, want);
|
2007-05-22 19:09:23 +02:00
|
|
|
|
else {
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, u->thisorder, "error_cannotmake", ""));
|
|
|
|
|
}
|
2007-03-25 01:04:40 +01:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2007-02-01 00:41:55 +01:00
|
|
|
|
int
|
2004-06-21 18:45:27 +02:00
|
|
|
|
make_cmd(unit * u, struct order * ord)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region * r = u->region;
|
2007-02-25 09:58:01 +01:00
|
|
|
|
const building_type * btype;
|
|
|
|
|
const ship_type * stype;
|
|
|
|
|
param_t p;
|
|
|
|
|
int m;
|
|
|
|
|
const item_type * itype;
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char *s;
|
2004-09-30 00:17:49 +02:00
|
|
|
|
const struct locale * lang = u->faction->locale;
|
2007-06-27 02:08:46 +02:00
|
|
|
|
char ibuf[16];
|
2007-02-25 09:58:01 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
|
|
|
|
s = getstrtoken();
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
m = atoi((const char *)s);
|
2007-06-27 02:08:46 +02:00
|
|
|
|
sprintf(ibuf, "%d", m);
|
|
|
|
|
if (!strcmp(ibuf, (const char *)s)) {
|
2007-02-25 09:58:01 +01:00
|
|
|
|
/* first came a want-paramter */
|
|
|
|
|
s = getstrtoken();
|
|
|
|
|
} else {
|
|
|
|
|
m = INT_MAX;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-02-25 09:58:01 +01:00
|
|
|
|
p = findparam(s, u->faction->locale);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-02-25 09:58:01 +01:00
|
|
|
|
/* MACHE TEMP kann hier schon gar nicht auftauchen, weil diese nicht in
|
|
|
|
|
* thisorder abgespeichert werden - und auf den ist getstrtoken() beim
|
|
|
|
|
* aufruf von make geeicht */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-02-25 09:58:01 +01:00
|
|
|
|
if (p == P_ROAD) {
|
|
|
|
|
if(r->planep && fval(r->planep, PFL_NOBUILD)) {
|
|
|
|
|
cmistake(u, ord, 275, MSG_PRODUCE);
|
2007-02-01 00:41:55 +01:00
|
|
|
|
} else {
|
|
|
|
|
direction_t d = finddirection(getstrtoken(), u->faction->locale);
|
|
|
|
|
if (d!=NODIRECTION) {
|
|
|
|
|
if(r->planep && fval(r->planep, PFL_NOBUILD)) {
|
|
|
|
|
cmistake(u, ord, 94, MSG_PRODUCE);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
build_road(r, u, m, d);
|
|
|
|
|
} else {
|
2007-02-23 22:18:35 +01:00
|
|
|
|
/* Die Richtung wurde nicht erkannt */
|
2007-02-01 00:41:55 +01:00
|
|
|
|
cmistake(u, ord, 71, MSG_PRODUCE);
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-02-25 09:58:01 +01:00
|
|
|
|
return 0;
|
|
|
|
|
} else if (p == P_SHIP) {
|
|
|
|
|
if(r->planep && fval(r->planep, PFL_NOBUILD)) {
|
|
|
|
|
cmistake(u, ord, 276, MSG_PRODUCE);
|
2007-02-01 00:41:55 +01:00
|
|
|
|
} else {
|
|
|
|
|
continue_ship(r, u, m);
|
|
|
|
|
}
|
2007-02-25 09:58:01 +01:00
|
|
|
|
return 0;
|
|
|
|
|
} else if (p == P_HERBS) {
|
|
|
|
|
herbsearch(r, u, m);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-09-30 00:17:49 +02:00
|
|
|
|
/* since the string can match several objects, like in 'academy' and
|
2007-02-25 09:58:01 +01:00
|
|
|
|
* 'academy of arts', we need to figure out what the player meant.
|
|
|
|
|
* This is not 100% safe.
|
|
|
|
|
*/
|
|
|
|
|
stype = findshiptype(s, lang);
|
|
|
|
|
btype = findbuildingtype(s, lang);
|
|
|
|
|
itype = finditemtype(s, lang);
|
|
|
|
|
|
2004-09-30 00:17:49 +02:00
|
|
|
|
if (itype!=NULL && (btype!=NULL || stype!=NULL)) {
|
|
|
|
|
if (itype->construction==NULL) {
|
|
|
|
|
/* if the item cannot be made, we probably didn't mean to make it */
|
|
|
|
|
itype = NULL;
|
|
|
|
|
} else if (stype!=NULL) {
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char * sname = LOC(lang, stype->name[0]);
|
|
|
|
|
const char * iname = LOC(lang, resourcename(itype->rtype, 0));
|
|
|
|
|
if (strlen(iname)<strlen(sname)) stype = NULL;
|
2004-09-30 00:17:49 +02:00
|
|
|
|
else itype = NULL;
|
|
|
|
|
} else {
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char * bname = LOC(lang, btype->_name);
|
|
|
|
|
const char * iname = LOC(lang, resourcename(itype->rtype, 0));
|
|
|
|
|
if (strlen(iname)<strlen(bname)) btype = NULL;
|
2004-09-30 00:17:49 +02:00
|
|
|
|
else itype = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (btype!=NULL && stype!=NULL) {
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char * bname = LOC(lang, btype->_name);
|
|
|
|
|
const char * sname = LOC(lang, stype->name[0]);
|
|
|
|
|
if (strlen(sname)<strlen(bname)) btype = NULL;
|
2004-09-30 00:17:49 +02:00
|
|
|
|
else stype = NULL;
|
|
|
|
|
}
|
2007-02-25 09:58:01 +01:00
|
|
|
|
|
|
|
|
|
if (stype != NOSHIP) {
|
|
|
|
|
if(r->planep && fval(r->planep, PFL_NOBUILD)) {
|
|
|
|
|
cmistake(u, ord, 276, MSG_PRODUCE);
|
2007-02-01 00:41:55 +01:00
|
|
|
|
} else {
|
|
|
|
|
create_ship(r, u, stype, m, ord);
|
|
|
|
|
}
|
2007-02-25 09:58:01 +01:00
|
|
|
|
} else if (btype != NOBUILDING) {
|
|
|
|
|
if(r->planep && fval(r->planep, PFL_NOBUILD)) {
|
|
|
|
|
cmistake(u, ord, 94, MSG_PRODUCE);
|
2007-02-01 00:41:55 +01:00
|
|
|
|
} else {
|
|
|
|
|
build_building(u, btype, m, ord);
|
|
|
|
|
}
|
2007-02-25 09:58:01 +01:00
|
|
|
|
}
|
|
|
|
|
else if (itype!=NULL) {
|
|
|
|
|
create_item(u, itype, m);
|
2007-02-01 00:41:55 +01:00
|
|
|
|
} else {
|
2007-05-22 19:09:23 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "error_cannotmake", ""));
|
2007-02-01 00:41:55 +01:00
|
|
|
|
}
|
2004-09-30 00:17:49 +02:00
|
|
|
|
|
2007-02-01 00:41:55 +01:00
|
|
|
|
return 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2005-05-08 02:16:32 +02:00
|
|
|
|
static void
|
|
|
|
|
free_luxuries(struct attrib * a)
|
|
|
|
|
{
|
2005-12-11 10:58:17 +01:00
|
|
|
|
i_freeall((item**)&a->data.v);
|
2005-05-08 02:16:32 +02:00
|
|
|
|
}
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
const attrib_type at_luxuries = {
|
2005-05-08 02:16:32 +02:00
|
|
|
|
"luxuries", NULL, free_luxuries, NULL, NULL, NULL
|
2001-01-25 10:37:55 +01:00
|
|
|
|
};
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
expandbuying(region * r, request * buyorders)
|
|
|
|
|
{
|
|
|
|
|
int max_products;
|
|
|
|
|
unit *u;
|
|
|
|
|
static struct trade {
|
|
|
|
|
const luxury_type * type;
|
|
|
|
|
int number;
|
|
|
|
|
int multi;
|
|
|
|
|
} *trades, *trade;
|
|
|
|
|
static int ntrades=0;
|
|
|
|
|
int i, j;
|
|
|
|
|
const luxury_type * ltype;
|
|
|
|
|
|
|
|
|
|
if (ntrades==0) {
|
|
|
|
|
for (ltype=luxurytypes;ltype;ltype=ltype->next)
|
|
|
|
|
++ntrades;
|
|
|
|
|
trades = gc_add(calloc(sizeof(struct trade), ntrades));
|
|
|
|
|
for (i=0, ltype=luxurytypes;i!=ntrades;++i, ltype=ltype->next)
|
|
|
|
|
trades[i].type = ltype;
|
|
|
|
|
}
|
|
|
|
|
for (i=0;i!=ntrades;++i) {
|
|
|
|
|
trades[i].number = 0;
|
|
|
|
|
trades[i].multi = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!buyorders) return;
|
|
|
|
|
|
|
|
|
|
/* Initialisation. multiplier ist der Multiplikator auf den
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* Verkaufspreis. F<EFBFBD>r max_products Produkte kauft man das Produkt zum
|
|
|
|
|
* einfachen Verkaufspreis, danach erh<EFBFBD>ht sich der Multiplikator um 1.
|
|
|
|
|
* counter ist ein Z<EFBFBD>hler, der die gekauften Produkte z<EFBFBD>hlt. money
|
|
|
|
|
* wird f<EFBFBD>r die debug message gebraucht. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
max_products = rpeasants(r) / TRADE_FRACTION;
|
|
|
|
|
|
2006-02-19 23:43:56 +01:00
|
|
|
|
/* Kauf - auch so programmiert, da<64> er leicht erweiterbar auf mehrere
|
|
|
|
|
* G<EFBFBD>ter pro Monat ist. j sind die Befehle, i der Index des
|
|
|
|
|
* gehandelten Produktes. */
|
2004-05-13 22:10:20 +02:00
|
|
|
|
if (max_products>0) {
|
2006-02-19 23:43:56 +01:00
|
|
|
|
expandorders(r, buyorders);
|
|
|
|
|
if (!norders) return;
|
|
|
|
|
|
|
|
|
|
for (j = 0; j != norders; j++) {
|
|
|
|
|
int price, multi;
|
|
|
|
|
ltype = oa[j].type.ltype;
|
|
|
|
|
trade = trades;
|
|
|
|
|
while (trade->type!=ltype) ++trade;
|
|
|
|
|
multi = trade->multi;
|
|
|
|
|
price = ltype->price * multi;
|
|
|
|
|
|
|
|
|
|
if (get_pooled(oa[j].unit, oldresourcetype[R_SILVER], GET_DEFAULT, price) >= price) {
|
|
|
|
|
unit * u = oa[j].unit;
|
|
|
|
|
|
|
|
|
|
/* litems z<>hlt die G<>ter, die verkauft wurden, u->n das Geld, das
|
|
|
|
|
* verdient wurde. Dies mu<EFBFBD> gemacht werden, weil der Preis st<EFBFBD>ndig sinkt,
|
|
|
|
|
* man sich also das verdiente Geld und die verkauften Produkte separat
|
|
|
|
|
* merken mu<EFBFBD>. */
|
|
|
|
|
attrib * a = a_find(u->attribs, &at_luxuries);
|
|
|
|
|
if (a==NULL) a = a_add(&u->attribs, a_new(&at_luxuries));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-10-05 22:21:01 +02:00
|
|
|
|
i_change((item**)&a->data.v, ltype->itype, 1);
|
|
|
|
|
i_change(&oa[j].unit->items, ltype->itype, 1);
|
|
|
|
|
use_pooled(u, oldresourcetype[R_SILVER], GET_DEFAULT, price);
|
|
|
|
|
if (u->n < 0)
|
|
|
|
|
u->n = 0;
|
|
|
|
|
u->n += price;
|
|
|
|
|
|
|
|
|
|
rsetmoney(r, rmoney(r) + price);
|
|
|
|
|
|
|
|
|
|
/* Falls mehr als max_products Bauern ein Produkt verkauft haben, steigt
|
|
|
|
|
* der Preis Multiplikator f<EFBFBD>r das Produkt um den Faktor 1. Der Z<EFBFBD>hler
|
|
|
|
|
* wird wieder auf 0 gesetzt. */
|
|
|
|
|
if (++trade->number == max_products) {
|
|
|
|
|
trade->number = 0;
|
|
|
|
|
++trade->multi;
|
|
|
|
|
}
|
2007-03-11 23:18:58 +01:00
|
|
|
|
fset(u, UFL_LONGACTION|UFL_NOTMOVING);
|
2006-10-05 22:21:01 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
free(oa);
|
|
|
|
|
|
|
|
|
|
/* Ausgabe an Einheiten */
|
|
|
|
|
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
attrib * a = a_find(u->attribs, &at_luxuries);
|
|
|
|
|
item * itm;
|
|
|
|
|
if (a==NULL) continue;
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("buy", "unit money", u, u->n));
|
|
|
|
|
for (itm=(item*)a->data.v; itm; itm=itm->next) {
|
|
|
|
|
if (itm->number) {
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("buyamount",
|
|
|
|
|
"unit amount resource", u, itm->number, itm->type->rtype));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
a_remove(&u->attribs, a);
|
|
|
|
|
}
|
2004-05-13 22:10:20 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
attrib_type at_trades = {
|
|
|
|
|
"trades",
|
2003-09-21 10:47:28 +02:00
|
|
|
|
DEFAULT_INIT,
|
|
|
|
|
DEFAULT_FINALIZE,
|
|
|
|
|
DEFAULT_AGE,
|
|
|
|
|
NO_WRITE,
|
|
|
|
|
NO_READ
|
2001-01-25 10:37:55 +01:00
|
|
|
|
};
|
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2004-06-21 18:45:27 +02:00
|
|
|
|
buy(unit * u, request ** buyorders, struct order * ord)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region * r = u->region;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
int n, k;
|
|
|
|
|
request *o;
|
|
|
|
|
attrib * a;
|
|
|
|
|
const item_type * itype = NULL;
|
|
|
|
|
const luxury_type * ltype = NULL;
|
|
|
|
|
if (u->ship && is_guarded(r, u, GUARD_CREWS)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 69, MSG_INCOME);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (u->ship && is_guarded(r, u, GUARD_CREWS)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 69, MSG_INCOME);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
/* Im Augenblick kann man nur 1 Produkt kaufen. expandbuying ist aber
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* schon daf<EFBFBD>r ausger<EFBFBD>stet, mehrere Produkte zu kaufen. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
2007-06-26 11:32:28 +02:00
|
|
|
|
n = getuint();
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (!n) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 26, MSG_COMMERCE);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (besieged(u)) {
|
|
|
|
|
/* Belagerte Einheiten k<>nnen nichts kaufen. */
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 60, MSG_COMMERCE);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (u->race == new_race[RC_INSECT]) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* entweder man ist insekt, oder... */
|
2005-11-26 00:52:53 +01:00
|
|
|
|
if (r->terrain != newterrain(T_SWAMP) && rterrain(r) != T_DESERT && !rbuildings(r)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 119, MSG_COMMERCE);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
/* ...oder in der Region mu<6D> es eine Burg geben. */
|
2005-04-03 10:32:16 +02:00
|
|
|
|
building * b;
|
|
|
|
|
static const struct building_type * bt_castle;
|
|
|
|
|
if (!bt_castle) bt_castle = bt_find("castle");
|
|
|
|
|
for (b=r->buildings;b;b=b->next) {
|
2005-04-09 16:46:45 +02:00
|
|
|
|
if (b->type==bt_castle && b->size>=2) break;
|
2005-04-03 10:32:16 +02:00
|
|
|
|
}
|
|
|
|
|
if (b==NULL) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 119, MSG_COMMERCE);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Ein H<>ndler kann nur 10 G<>ter pro Talentpunkt handeln. */
|
|
|
|
|
k = u->number * 10 * eff_skill(u, SK_TRADE, r);
|
|
|
|
|
|
|
|
|
|
/* hat der H<>ndler bereits gehandelt, muss die Menge der bereits
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* verkauften/gekauften G<EFBFBD>ter abgezogen werden */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
a = a_find(u->attribs, &at_trades);
|
|
|
|
|
if (!a) {
|
|
|
|
|
a = a_add(&u->attribs, a_new(&at_trades));
|
|
|
|
|
} else {
|
|
|
|
|
k -= a->data.i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
n = min(n, k);
|
|
|
|
|
|
|
|
|
|
if (!n) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 102, MSG_COMMERCE);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assert(n>=0);
|
|
|
|
|
/* die Menge der verkauften G<>ter merken */
|
|
|
|
|
a->data.i += n;
|
|
|
|
|
|
|
|
|
|
itype = finditemtype(getstrtoken(), u->faction->locale);
|
|
|
|
|
if (itype!=NULL) {
|
|
|
|
|
ltype = resource2luxury(itype->rtype);
|
|
|
|
|
if (ltype==NULL) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 124, MSG_COMMERCE);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (r_demand(r, ltype)) {
|
2006-01-29 02:55:36 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2004-06-21 18:45:27 +02:00
|
|
|
|
msg_feedback(u, ord, "luxury_notsold", ""));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
o = (request *) calloc(1, sizeof(request));
|
|
|
|
|
o->type.ltype = ltype; /* sollte immer gleich sein */
|
|
|
|
|
|
|
|
|
|
o->unit = u;
|
|
|
|
|
o->qty = n;
|
|
|
|
|
addlist(buyorders, o);
|
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
/* Steuers<72>tze in % bei Burggr<67><72>e */
|
2001-12-10 01:13:39 +01:00
|
|
|
|
static int tax_per_size[7] =
|
|
|
|
|
{0, 6, 12, 18, 24, 30, 36};
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2004-07-14 09:12:36 +02:00
|
|
|
|
expandselling(region * r, request * sellorders, int limit)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
int money, price, j, max_products;
|
|
|
|
|
/* int m, n = 0; */
|
|
|
|
|
int maxsize = 0, maxeffsize = 0;
|
|
|
|
|
int taxcollected = 0;
|
|
|
|
|
int hafencollected = 0;
|
|
|
|
|
unit *maxowner = (unit *) NULL;
|
|
|
|
|
building *maxb = (building *) NULL;
|
|
|
|
|
building *b;
|
|
|
|
|
unit *u;
|
|
|
|
|
unit *hafenowner;
|
|
|
|
|
static int *counter;
|
|
|
|
|
static int ncounter = 0;
|
|
|
|
|
|
|
|
|
|
if (ncounter==0) {
|
|
|
|
|
const luxury_type * ltype;
|
|
|
|
|
for (ltype=luxurytypes;ltype;ltype=ltype->next) ++ncounter;
|
|
|
|
|
counter=(int*)gc_add(calloc(sizeof(int), ncounter));
|
|
|
|
|
} else {
|
|
|
|
|
memset(counter, 0, sizeof(int)*ncounter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!sellorders) /* NEIN, denn Insekten k<>nnen in || !r->buildings) */
|
|
|
|
|
return; /* S<>mpfen und W<>sten auch so handeln */
|
|
|
|
|
|
|
|
|
|
/* Stelle Eigent<6E>mer der gr<67><72>ten Burg fest. Bekommt Steuern aus jedem
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* Verkauf. Wenn zwei Burgen gleicher Gr<EFBFBD><EFBFBD>e bekommt gar keiner etwas. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
for (b = rbuildings(r); b; b = b->next) {
|
|
|
|
|
if (b->size > maxsize && buildingowner(r, b) != NULL
|
2003-09-21 10:47:28 +02:00
|
|
|
|
&& b->type == bt_find("castle")) {
|
|
|
|
|
maxb = b;
|
|
|
|
|
maxsize = b->size;
|
|
|
|
|
maxowner = buildingowner(r, b);
|
|
|
|
|
} else if (b->size == maxsize && b->type == bt_find("castle")) {
|
|
|
|
|
maxb = (building *) NULL;
|
|
|
|
|
maxowner = (unit *) NULL;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-03-27 22:49:27 +01:00
|
|
|
|
hafenowner = owner_buildingtyp(r, bt_find("harbour"));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
if (maxb != (building *) NULL && maxowner != (unit *) NULL) {
|
|
|
|
|
maxeffsize = buildingeffsize(maxb, false);
|
|
|
|
|
if (maxeffsize == 0) {
|
|
|
|
|
maxb = (building *) NULL;
|
|
|
|
|
maxowner = (unit *) NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* Die Region muss genug Geld haben, um die Produkte kaufen zu k<>nnen. */
|
|
|
|
|
|
|
|
|
|
money = rmoney(r);
|
|
|
|
|
|
|
|
|
|
/* max_products sind 1/100 der Bev<65>lkerung, falls soviele Produkte
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* verkauft werden - counter[] - sinkt die Nachfrage um 1 Punkt.
|
|
|
|
|
* multiplier speichert r->demand f<EFBFBD>r die debug message ab. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
max_products = rpeasants(r) / TRADE_FRACTION;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (max_products <= 0) return;
|
|
|
|
|
|
2002-03-27 22:49:27 +01:00
|
|
|
|
if (rterrain(r) == T_DESERT && buildingtype_exists(r, bt_find("caravan"))) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
max_products = rpeasants(r) * 2 / TRADE_FRACTION;
|
|
|
|
|
}
|
|
|
|
|
/* Verkauf: so programmiert, dass er leicht auf mehrere Gueter pro
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* Runde erweitert werden kann. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
expandorders(r, sellorders);
|
|
|
|
|
if (!norders) return;
|
|
|
|
|
|
|
|
|
|
for (j = 0; j != norders; j++) {
|
|
|
|
|
static const luxury_type * search=NULL;
|
|
|
|
|
const luxury_type * ltype = oa[j].type.ltype;
|
|
|
|
|
int multi = r_demand(r, ltype);
|
|
|
|
|
static int i=-1;
|
|
|
|
|
int use = 0;
|
|
|
|
|
if (search!=ltype) {
|
|
|
|
|
i=0;
|
|
|
|
|
for (search=luxurytypes;search!=ltype;search=search->next) ++i;
|
|
|
|
|
}
|
2004-07-14 09:12:36 +02:00
|
|
|
|
if (counter[i]>=limit) continue;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (counter[i]+1 > max_products && multi > 1) --multi;
|
|
|
|
|
price = ltype->price * multi;
|
|
|
|
|
|
|
|
|
|
if (money >= price) {
|
|
|
|
|
int abgezogenhafen = 0;
|
|
|
|
|
int abgezogensteuer = 0;
|
|
|
|
|
unit * u = oa[j].unit;
|
|
|
|
|
attrib * a = a_find(u->attribs, &at_luxuries);
|
|
|
|
|
if (a==NULL) a = a_add(&u->attribs, a_new(&at_luxuries));
|
|
|
|
|
i_change((item**)&a->data.v, ltype->itype, 1);
|
|
|
|
|
++use;
|
|
|
|
|
if (u->n < 0)
|
|
|
|
|
u->n = 0;
|
|
|
|
|
|
|
|
|
|
if (hafenowner != NULL) {
|
|
|
|
|
if (hafenowner->faction != u->faction) {
|
|
|
|
|
abgezogenhafen = price / 10;
|
|
|
|
|
hafencollected += abgezogenhafen;
|
|
|
|
|
price -= abgezogenhafen;
|
|
|
|
|
money -= abgezogenhafen;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (maxb != NULL) {
|
|
|
|
|
if (maxowner->faction != u->faction) {
|
|
|
|
|
abgezogensteuer = price * tax_per_size[maxeffsize] / 100;
|
|
|
|
|
taxcollected += abgezogensteuer;
|
|
|
|
|
price -= abgezogensteuer;
|
|
|
|
|
money -= abgezogensteuer;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
u->n += price;
|
|
|
|
|
change_money(u, price);
|
2007-03-11 23:18:58 +01:00
|
|
|
|
fset(u, UFL_LONGACTION|UFL_NOTMOVING);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
/* r->money -= price; --- dies wird eben nicht ausgef<65>hrt, denn die
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* Produkte k<EFBFBD>nnen auch als Steuern eingetrieben werden. In der Region
|
|
|
|
|
* wurden Silberst<EFBFBD>cke gegen Luxusg<EFBFBD>ter des selben Wertes eingetauscht!
|
|
|
|
|
* Falls mehr als max_products Kunden ein Produkt gekauft haben, sinkt
|
|
|
|
|
* die Nachfrage f<EFBFBD>r das Produkt um 1. Der Z<EFBFBD>hler wird wieder auf 0
|
|
|
|
|
* gesetzt. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
if (++counter[i] > max_products) {
|
|
|
|
|
int d = r_demand(r, ltype);
|
2005-11-26 02:24:28 +01:00
|
|
|
|
if (d > 1) {
|
|
|
|
|
r_setdemand(r, ltype, d-1);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
counter[i] = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (use>0) {
|
|
|
|
|
#ifdef NDEBUG
|
2005-10-30 16:42:15 +01:00
|
|
|
|
use_pooled(oa[j].unit, ltype->itype->rtype, GET_DEFAULT, use);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#else
|
2005-10-30 16:42:15 +01:00
|
|
|
|
/* int i = */ use_pooled(oa[j].unit, ltype->itype->rtype, GET_DEFAULT, use);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* assert(i==use); */
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
free(oa);
|
|
|
|
|
|
|
|
|
|
/* Steuern. Hier werden die Steuern dem Besitzer der gr<67><72>ten Burg gegeben. */
|
|
|
|
|
|
|
|
|
|
if (maxowner) {
|
|
|
|
|
if (taxcollected > 0) {
|
|
|
|
|
change_money(maxowner, (int) taxcollected);
|
|
|
|
|
add_income(maxowner, IC_TRADETAX, taxcollected, taxcollected);
|
2005-01-19 21:33:13 +01:00
|
|
|
|
/* TODO: Meldung
|
2005-07-25 21:56:44 +02:00
|
|
|
|
* "%s verdient %d Silber durch den Handel in %s.",
|
2005-01-19 21:33:13 +01:00
|
|
|
|
* unitname(maxowner), (int) taxcollected, regionname(r)); */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (hafenowner) {
|
|
|
|
|
if (hafencollected > 0) {
|
|
|
|
|
change_money(hafenowner, (int) hafencollected);
|
|
|
|
|
add_income(hafenowner, IC_TRADETAX, hafencollected, hafencollected);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* Berichte an die Einheiten */
|
|
|
|
|
|
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
|
|
|
|
|
attrib * a = a_find(u->attribs, &at_luxuries);
|
|
|
|
|
item * itm;
|
|
|
|
|
if (a==NULL) continue;
|
|
|
|
|
for (itm=(item*)a->data.v; itm; itm=itm->next) {
|
|
|
|
|
if (itm->number) {
|
2004-07-14 09:12:36 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("sellamount",
|
|
|
|
|
"unit amount resource", u, itm->number, itm->type->rtype));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
a_remove(&u->attribs, a);
|
|
|
|
|
add_income(u, IC_TRADE, u->n, u->n);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2004-07-14 09:12:36 +02:00
|
|
|
|
static boolean
|
2004-06-21 18:45:27 +02:00
|
|
|
|
sell(unit * u, request ** sellorders, struct order * ord)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-07-14 09:12:36 +02:00
|
|
|
|
boolean unlimited = true;
|
2006-02-19 23:43:56 +01:00
|
|
|
|
const item_type * itype;
|
|
|
|
|
const luxury_type * ltype=NULL;
|
|
|
|
|
int n;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region * r = u->region;
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char *s;
|
2006-02-19 23:43:56 +01:00
|
|
|
|
|
|
|
|
|
if (u->ship && is_guarded(r, u, GUARD_CREWS)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 69, MSG_INCOME);
|
2004-07-14 09:12:36 +02:00
|
|
|
|
return false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* sellorders sind KEIN array, weil f<>r alle items DIE SELBE resource
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* (das geld der region) aufgebraucht wird. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
2001-01-25 10:37:55 +01:00
|
|
|
|
s = getstrtoken();
|
2004-06-21 18:45:27 +02:00
|
|
|
|
|
|
|
|
|
if (findparam(s, u->faction->locale) == P_ANY) {
|
2004-07-14 09:12:36 +02:00
|
|
|
|
unlimited = false;
|
2002-02-17 09:57:39 +01:00
|
|
|
|
n = rpeasants(r) / TRADE_FRACTION;
|
2002-03-27 22:49:27 +01:00
|
|
|
|
if (rterrain(r) == T_DESERT && buildingtype_exists(r, bt_find("caravan")))
|
2001-01-25 10:37:55 +01:00
|
|
|
|
n *= 2;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (n==0) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 303, MSG_COMMERCE);
|
2004-07-14 09:12:36 +02:00
|
|
|
|
return false;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
} else {
|
2007-06-26 11:32:28 +02:00
|
|
|
|
n = atoi((const char *)s);
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (n==0) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 27, MSG_COMMERCE);
|
2004-07-14 09:12:36 +02:00
|
|
|
|
return false;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* Belagerte Einheiten k<>nnen nichts verkaufen. */
|
|
|
|
|
|
|
|
|
|
if (besieged(u)) {
|
2006-01-29 02:55:36 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2004-06-21 18:45:27 +02:00
|
|
|
|
msg_feedback(u, ord, "error60", ""));
|
2004-07-14 09:12:36 +02:00
|
|
|
|
return false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* In der Region mu<6D> es eine Burg geben. */
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (u->race == new_race[RC_INSECT]) {
|
2005-11-26 00:52:53 +01:00
|
|
|
|
if (r->terrain != newterrain(T_SWAMP) && rterrain(r) != T_DESERT && !rbuildings(r)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 119, MSG_COMMERCE);
|
2004-07-14 09:12:36 +02:00
|
|
|
|
return false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
} else {
|
2005-04-09 16:46:45 +02:00
|
|
|
|
/* ...oder in der Region mu<6D> es eine Burg geben. */
|
|
|
|
|
building * b;
|
|
|
|
|
static const struct building_type * bt_castle;
|
|
|
|
|
if (!bt_castle) bt_castle = bt_find("castle");
|
2006-02-19 23:43:56 +01:00
|
|
|
|
for (b=r->buildings;b;b=b->next) {
|
|
|
|
|
if (b->type==bt_castle && b->size>=2) break;
|
2005-04-09 16:46:45 +02:00
|
|
|
|
}
|
2006-02-19 23:43:56 +01:00
|
|
|
|
if (b==NULL) {
|
|
|
|
|
cmistake(u, ord, 119, MSG_COMMERCE);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Ein H<>ndler kann nur 10 G<>ter pro Talentpunkt verkaufen. */
|
|
|
|
|
|
|
|
|
|
n = min(n, u->number * 10 * eff_skill(u, SK_TRADE, r));
|
|
|
|
|
|
|
|
|
|
if (!n) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 54, MSG_COMMERCE);
|
2004-07-14 09:12:36 +02:00
|
|
|
|
return false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
s=getstrtoken();
|
|
|
|
|
itype = finditemtype(s, u->faction->locale);
|
|
|
|
|
if (itype!=NULL) ltype = resource2luxury(itype->rtype);
|
|
|
|
|
if (ltype==NULL) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 126, MSG_COMMERCE);
|
2004-07-14 09:12:36 +02:00
|
|
|
|
return false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
2006-02-19 23:43:56 +01:00
|
|
|
|
else {
|
|
|
|
|
attrib * a;
|
2004-07-14 09:12:36 +02:00
|
|
|
|
request *o;
|
2006-02-19 23:43:56 +01:00
|
|
|
|
int k, available;
|
|
|
|
|
|
|
|
|
|
if (!r_demand(r, ltype)) {
|
|
|
|
|
cmistake(u, ord, 263, MSG_COMMERCE);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
available = get_pooled(u, itype->rtype, GET_DEFAULT, INT_MAX);
|
|
|
|
|
|
|
|
|
|
/* Wenn andere Einheiten das selbe verkaufen, mu<6D> ihr Zeug abgezogen
|
|
|
|
|
* werden damit es nicht zweimal verkauft wird: */
|
|
|
|
|
for (o=*sellorders;o;o=o->next) {
|
|
|
|
|
if (o->type.ltype==ltype && o->unit->faction == u->faction) {
|
|
|
|
|
int fpool = o->qty - get_pooled(o->unit, itype->rtype, GET_RESERVE, INT_MAX);
|
|
|
|
|
available -= max(0, fpool);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
n = min(n, available);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
if (n <= 0) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 264, MSG_COMMERCE);
|
2004-07-14 09:12:36 +02:00
|
|
|
|
return false;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* Hier wird request->type verwendet, weil die obere limit durch
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* das silber gegeben wird (region->money), welches f<EFBFBD>r alle
|
|
|
|
|
* (!) produkte als summe gilt, als nicht wie bei der
|
|
|
|
|
* produktion, wo f<EFBFBD>r jedes produkt einzeln eine obere limite
|
|
|
|
|
* existiert, so dass man arrays von orders machen kann. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
/* Ein H<>ndler kann nur 10 G<>ter pro Talentpunkt handeln. */
|
|
|
|
|
k = u->number * 10 * eff_skill(u, SK_TRADE, r);
|
|
|
|
|
|
|
|
|
|
/* hat der H<>ndler bereits gehandelt, muss die Menge der bereits
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* verkauften/gekauften G<EFBFBD>ter abgezogen werden */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
a = a_find(u->attribs, &at_trades);
|
|
|
|
|
if (!a) {
|
|
|
|
|
a = a_add(&u->attribs, a_new(&at_trades));
|
|
|
|
|
} else {
|
|
|
|
|
k -= a->data.i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
n = min(n, k);
|
|
|
|
|
assert(n>=0);
|
|
|
|
|
/* die Menge der verkauften G<>ter merken */
|
|
|
|
|
a->data.i += n;
|
|
|
|
|
o = (request *) calloc(1, sizeof(request));
|
|
|
|
|
o->unit = u;
|
|
|
|
|
o->qty = n;
|
|
|
|
|
o->type.ltype = ltype;
|
|
|
|
|
addlist(sellorders, o);
|
|
|
|
|
|
2004-07-14 09:12:36 +02:00
|
|
|
|
return unlimited;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
expandstealing(region * r, request * stealorders)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
expandorders(r, stealorders);
|
|
|
|
|
if (!norders) return;
|
|
|
|
|
|
|
|
|
|
/* F<>r jede unit in der Region wird Geld geklaut, wenn sie Opfer eines
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* Beklauen-Orders ist. Jedes Opfer mu<EFBFBD> einzeln behandelt werden.
|
|
|
|
|
*
|
|
|
|
|
* u ist die beklaute unit. oa.unit ist die klauende unit.
|
|
|
|
|
*/
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-02-19 23:43:56 +01:00
|
|
|
|
for (i = 0; i != norders && oa[i].unit->n <= oa[i].unit->wants; i++) {
|
|
|
|
|
unit *u = findunitg(oa[i].no, r);
|
|
|
|
|
int n = 0;
|
|
|
|
|
if (u && u->region==r) {
|
|
|
|
|
n = get_pooled(u, r_silver, GET_ALL, INT_MAX);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#ifndef GOBLINKILL
|
2006-02-19 23:43:56 +01:00
|
|
|
|
if (oa[i].type.goblin) { /* Goblin-Spezialklau */
|
|
|
|
|
int uct = 0;
|
|
|
|
|
unit *u2;
|
2007-06-20 02:34:02 +02:00
|
|
|
|
assert(effskill(oa[i].unit, SK_STEALTH)>=4 || !"this goblin\'s skill is too low");
|
2005-11-25 23:09:59 +01:00
|
|
|
|
for (u2 = r->units; u2; u2 = u2->next) {
|
|
|
|
|
if (u2->faction == u->faction) {
|
|
|
|
|
uct += maintenance_cost(u2);
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-02-19 23:43:56 +01:00
|
|
|
|
n -= uct * 2;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#endif
|
2006-02-19 23:43:56 +01:00
|
|
|
|
if (n>10 && rplane(r) && (rplane(r)->flags & PFL_NOALLIANCES)) {
|
|
|
|
|
/* In Questen nur reduziertes Klauen */
|
|
|
|
|
n = 10;
|
|
|
|
|
}
|
|
|
|
|
if (n > 0) {
|
|
|
|
|
n = min(n, oa[i].unit->wants);
|
|
|
|
|
use_pooled(u, r_silver, GET_ALL, n);
|
|
|
|
|
oa[i].unit->n = n;
|
|
|
|
|
change_money(oa[i].unit, n);
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("stealeffect", "unit region amount", u, u->region, n));
|
|
|
|
|
}
|
|
|
|
|
add_income(oa[i].unit, IC_STEAL, oa[i].unit->wants, oa[i].unit->n);
|
2007-03-11 23:18:58 +01:00
|
|
|
|
fset(oa[i].unit, UFL_LONGACTION|UFL_NOTMOVING);
|
2006-02-19 23:43:56 +01:00
|
|
|
|
}
|
|
|
|
|
free(oa);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
/* ------------------------------------------------------------- */
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2001-12-10 01:13:39 +01:00
|
|
|
|
plant(region *r, unit *u, int raw)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-02-19 23:43:56 +01:00
|
|
|
|
int n, i, skill, planted = 0;
|
2005-10-30 10:51:02 +01:00
|
|
|
|
const item_type * itype;
|
|
|
|
|
static const resource_type * rt_water = NULL;
|
|
|
|
|
if (rt_water==NULL) rt_water = rt_find("p2");
|
2006-02-19 23:43:56 +01:00
|
|
|
|
|
2005-10-30 10:51:02 +01:00
|
|
|
|
assert(rt_water!=NULL);
|
2006-02-19 23:43:56 +01:00
|
|
|
|
if (!fval(r->terrain, LAND_REGION)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (rherbtype(r) == NULL) {
|
|
|
|
|
cmistake(u, u->thisorder, 108, MSG_PRODUCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Skill pr<70>fen */
|
|
|
|
|
skill = eff_skill(u, SK_HERBALISM, r);
|
|
|
|
|
itype = rherbtype(r);
|
|
|
|
|
if (skill < 6) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, u->thisorder, "plant_skills",
|
2005-10-25 14:38:01 +02:00
|
|
|
|
"skill minskill product", SK_HERBALISM, 6, itype->rtype, 1));
|
2006-02-19 23:43:56 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
/* Wasser des Lebens pr<70>fen */
|
|
|
|
|
if (get_pooled(u, rt_water, GET_DEFAULT, 1) == 0) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, u->thisorder, "resource_missing", "missing", rt_water));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
n = get_pooled(u, itype->rtype, GET_DEFAULT, skill*u->number);
|
|
|
|
|
/* Kr<4B>uter pr<70>fen */
|
|
|
|
|
if (n==0) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, u->thisorder, "resource_missing", "missing",
|
2005-10-25 14:38:01 +02:00
|
|
|
|
itype->rtype));
|
2006-02-19 23:43:56 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-02-19 23:43:56 +01:00
|
|
|
|
n = min(skill*u->number, n);
|
|
|
|
|
n = min(raw, n);
|
|
|
|
|
/* F<>r jedes Kraut Talent*10% Erfolgschance. */
|
|
|
|
|
for(i = n; i>0; i--) {
|
|
|
|
|
if (rng_int()%10 < skill) planted++;
|
|
|
|
|
}
|
|
|
|
|
produceexp(u, SK_HERBALISM, u->number);
|
|
|
|
|
|
|
|
|
|
/* Alles ok. Abziehen. */
|
|
|
|
|
use_pooled(u, rt_water, GET_DEFAULT, 1);
|
|
|
|
|
use_pooled(u, itype->rtype, GET_DEFAULT, n);
|
|
|
|
|
rsetherbs(r, rherbs(r)+planted);
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("plant", "unit region amount herb",
|
|
|
|
|
u, r, planted, itype->rtype));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2001-12-10 01:13:39 +01:00
|
|
|
|
planttrees(region *r, unit *u, int raw)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-02-19 23:43:56 +01:00
|
|
|
|
int n, i, skill, planted = 0;
|
|
|
|
|
const resource_type * rtype;
|
|
|
|
|
|
|
|
|
|
if (!fval(r->terrain, LAND_REGION)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Mallornb<6E>ume kann man nur in Mallornregionen z<>chten */
|
|
|
|
|
if (fval(r, RF_MALLORN)) {
|
|
|
|
|
rtype = rt_mallornseed;
|
|
|
|
|
} else {
|
|
|
|
|
rtype = rt_seed;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Skill pr<70>fen */
|
|
|
|
|
skill = eff_skill(u, SK_HERBALISM, r);
|
|
|
|
|
if (skill < 6) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, u->thisorder, "plant_skills",
|
2005-10-25 23:24:15 +02:00
|
|
|
|
"skill minskill product", SK_HERBALISM, 6, rtype, 1));
|
2006-02-19 23:43:56 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (fval(r, RF_MALLORN) && skill < 7 ) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, u->thisorder, "plant_skills",
|
2005-10-25 23:24:15 +02:00
|
|
|
|
"skill minskill product", SK_HERBALISM, 7, rtype, 1));
|
2006-02-19 23:43:56 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* wenn eine Anzahl angegeben wurde, nur soviel verbrauchen */
|
|
|
|
|
raw = min(raw, skill*u->number);
|
|
|
|
|
n = get_pooled(u, rtype, GET_DEFAULT, raw);
|
|
|
|
|
if (n==0) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, u->thisorder, "resource_missing", "missing", rtype));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
n = min(raw, n);
|
|
|
|
|
|
|
|
|
|
/* F<>r jeden Samen Talent*10% Erfolgschance. */
|
|
|
|
|
for(i = n; i>0; i--) {
|
|
|
|
|
if (rng_int()%10 < skill) planted++;
|
|
|
|
|
}
|
|
|
|
|
rsettrees(r, 0, rtrees(r, 0)+planted);
|
|
|
|
|
|
|
|
|
|
/* Alles ok. Abziehen. */
|
|
|
|
|
produceexp(u, SK_HERBALISM, u->number);
|
|
|
|
|
use_pooled(u, rtype, GET_DEFAULT, n);
|
|
|
|
|
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("plant",
|
|
|
|
|
"unit region amount herb", u, r, planted, rtype));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* z<>chte b<>ume */
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2001-12-10 01:13:39 +01:00
|
|
|
|
breedtrees(region *r, unit *u, int raw)
|
|
|
|
|
{
|
2006-02-19 23:43:56 +01:00
|
|
|
|
int n, i, skill, planted = 0;
|
|
|
|
|
const resource_type * rtype;
|
|
|
|
|
static int current_season = -1;
|
2005-06-05 15:34:08 +02:00
|
|
|
|
|
|
|
|
|
if (current_season<0) current_season = get_gamedate(turn, NULL)->season;
|
2006-02-19 23:43:56 +01:00
|
|
|
|
|
|
|
|
|
/* B<>ume z<>chten geht nur im Fr<46>hling */
|
|
|
|
|
if (current_season != SEASON_SPRING){
|
|
|
|
|
planttrees(r, u, raw);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!fval(r->terrain, LAND_REGION)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Mallornb<6E>ume kann man nur in Mallornregionen z<>chten */
|
|
|
|
|
if (fval(r, RF_MALLORN)) {
|
|
|
|
|
rtype = rt_mallornseed;
|
|
|
|
|
} else {
|
|
|
|
|
rtype = rt_seed;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Skill pr<70>fen */
|
|
|
|
|
skill = eff_skill(u, SK_HERBALISM, r);
|
|
|
|
|
if (skill < 12) {
|
|
|
|
|
planttrees(r, u, raw);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* wenn eine Anzahl angegeben wurde, nur soviel verbrauchen */
|
|
|
|
|
raw = min(skill*u->number, raw);
|
|
|
|
|
n = get_pooled(u, rtype, GET_DEFAULT, raw);
|
|
|
|
|
/* Samen pr<70>fen */
|
|
|
|
|
if (n==0) {
|
|
|
|
|
ADDMSG(&u->faction->msgs,
|
|
|
|
|
msg_feedback(u, u->thisorder, "resource_missing", "missing", rtype));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
n = min(raw, n);
|
|
|
|
|
|
|
|
|
|
/* F<>r jeden Samen Talent*5% Erfolgschance. */
|
|
|
|
|
for(i = n; i>0; i--) {
|
|
|
|
|
if (rng_int()%100 < skill*5) planted++;
|
|
|
|
|
}
|
|
|
|
|
rsettrees(r, 1, rtrees(r, 1)+planted);
|
|
|
|
|
|
|
|
|
|
/* Alles ok. Abziehen. */
|
|
|
|
|
produceexp(u, SK_HERBALISM, u->number);
|
|
|
|
|
use_pooled(u, rtype, GET_DEFAULT, n);
|
|
|
|
|
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("plant",
|
2006-01-08 12:33:10 +01:00
|
|
|
|
"unit region amount herb", u, r, planted, rtype));
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
2002-03-06 21:18:37 +01:00
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
/* z<>chte pferde */
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2001-12-10 01:13:39 +01:00
|
|
|
|
breedhorses(region *r, unit *u)
|
|
|
|
|
{
|
2007-01-01 13:14:56 +01:00
|
|
|
|
int n, c;
|
|
|
|
|
int gezuechtet = 0;
|
|
|
|
|
struct building * b = inside_building(u);
|
|
|
|
|
const struct building_type * btype = b?b->type:NULL;
|
|
|
|
|
|
|
|
|
|
if (btype!=bt_find("stables")) {
|
|
|
|
|
cmistake(u, u->thisorder, 122, MSG_PRODUCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (get_item(u, I_HORSE) < 2) {
|
|
|
|
|
cmistake(u, u->thisorder, 107, MSG_PRODUCE);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
n = min(u->number * eff_skill(u, SK_HORSE_TRAINING, r), get_item(u, I_HORSE));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-01-01 13:14:56 +01:00
|
|
|
|
for (c = 0; c < n; c++) {
|
|
|
|
|
if (rng_int() % 100 < eff_skill(u, SK_HORSE_TRAINING, r)) {
|
|
|
|
|
i_change(&u->items, olditemtype[I_HORSE], 1);
|
|
|
|
|
gezuechtet++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
produceexp(u, SK_HORSE_TRAINING, u->number);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-01-08 12:33:10 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("raised",
|
|
|
|
|
"unit amount", u, gezuechtet));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2004-06-21 18:45:27 +02:00
|
|
|
|
breed_cmd(unit *u, struct order * ord)
|
2001-12-10 01:13:39 +01:00
|
|
|
|
{
|
2007-01-01 13:14:56 +01:00
|
|
|
|
int m;
|
2007-08-10 09:03:23 +02:00
|
|
|
|
const char *s;
|
2007-01-01 13:14:56 +01:00
|
|
|
|
param_t p;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region *r = u->region;
|
2007-05-22 19:09:23 +02:00
|
|
|
|
const resource_type * rtype = NULL;
|
|
|
|
|
|
|
|
|
|
if (r->land==NULL) {
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "error_onlandonly", ""));
|
|
|
|
|
return;
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2007-01-01 13:14:56 +01:00
|
|
|
|
/* z<>chte [<anzahl>] <parameter> */
|
2004-06-21 18:45:27 +02:00
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
2007-01-01 13:14:56 +01:00
|
|
|
|
s = getstrtoken();
|
2004-06-21 18:45:27 +02:00
|
|
|
|
|
2007-06-26 11:32:28 +02:00
|
|
|
|
m = atoi((const char *)s);
|
2007-05-22 19:09:23 +02:00
|
|
|
|
if (m!=0) {
|
2007-01-01 13:14:56 +01:00
|
|
|
|
/* first came a want-paramter */
|
|
|
|
|
s = getstrtoken();
|
|
|
|
|
} else {
|
|
|
|
|
m = INT_MAX;
|
|
|
|
|
}
|
|
|
|
|
|
2007-05-22 19:09:23 +02:00
|
|
|
|
if (!s[0]) {
|
2007-01-01 13:14:56 +01:00
|
|
|
|
p = P_ANY;
|
|
|
|
|
} else {
|
|
|
|
|
p = findparam(s, u->faction->locale);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (p) {
|
|
|
|
|
case P_HERBS:
|
|
|
|
|
plant(r, u, m);
|
2007-05-22 19:09:23 +02:00
|
|
|
|
break;
|
2007-01-01 13:14:56 +01:00
|
|
|
|
case P_TREES:
|
|
|
|
|
breedtrees(r, u, m);
|
2007-05-22 19:09:23 +02:00
|
|
|
|
break;
|
2007-01-01 13:14:56 +01:00
|
|
|
|
default:
|
2007-05-22 19:09:23 +02:00
|
|
|
|
if (p!=P_ANY) {
|
|
|
|
|
rtype = findresourcetype(s, u->faction->locale);
|
|
|
|
|
if (rtype==rt_mallornseed || rtype==rt_seed) {
|
|
|
|
|
breedtrees(r, u, m);
|
|
|
|
|
break;
|
|
|
|
|
} else if (rtype!=oldresourcetype[R_HORSE]) {
|
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "error_cannotmake", ""));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-01-01 13:14:56 +01:00
|
|
|
|
breedhorses(r, u);
|
2007-05-22 19:09:23 +02:00
|
|
|
|
break;
|
2007-01-01 13:14:56 +01:00
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static const char *
|
2001-01-25 10:37:55 +01:00
|
|
|
|
rough_amount(int a, int m)
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
int p = (a * 100)/m;
|
|
|
|
|
|
|
|
|
|
if (p < 10) {
|
|
|
|
|
return "sehr wenige";
|
|
|
|
|
} else if (p < 30) {
|
|
|
|
|
return "wenige";
|
|
|
|
|
} else if (p < 60) {
|
|
|
|
|
return "relativ viele";
|
|
|
|
|
} else if (p < 90) {
|
|
|
|
|
return "viele";
|
|
|
|
|
}
|
|
|
|
|
return "sehr viele";
|
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2004-06-21 18:45:27 +02:00
|
|
|
|
research_cmd(unit *u, struct order * ord)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region *r = u->region;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
2004-07-08 00:02:42 +02:00
|
|
|
|
/*
|
|
|
|
|
const char *s = getstrtoken();
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-07-08 00:02:42 +02:00
|
|
|
|
if (findparam(s, u->faction->locale) == P_HERBS) { */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-07-08 00:02:42 +02:00
|
|
|
|
if (eff_skill(u, SK_HERBALISM, r) < 7) {
|
|
|
|
|
cmistake(u, ord, 227, MSG_EVENT);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2001-12-10 01:13:39 +01:00
|
|
|
|
|
2004-07-08 00:02:42 +02:00
|
|
|
|
produceexp(u, SK_HERBALISM, u->number);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-07-08 00:02:42 +02:00
|
|
|
|
if (rherbs(r) > 0) {
|
2005-10-25 14:38:01 +02:00
|
|
|
|
const item_type *itype = rherbtype(r);
|
2004-07-08 00:02:42 +02:00
|
|
|
|
|
2005-10-25 14:38:01 +02:00
|
|
|
|
if (itype != NULL) {
|
2006-01-08 12:33:10 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("researchherb",
|
|
|
|
|
"unit region amount herb",
|
|
|
|
|
u, r, rough_amount(rherbs(r), 100), itype->rtype));
|
2004-07-08 00:02:42 +02:00
|
|
|
|
} else {
|
2006-01-08 12:33:10 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("researchherb_none",
|
|
|
|
|
"unit region", u, r));
|
2004-07-08 00:02:42 +02:00
|
|
|
|
}
|
|
|
|
|
} else {
|
2006-01-08 12:33:10 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("researchherb_none",
|
|
|
|
|
"unit region", u, r));
|
2004-07-08 00:02:42 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static int
|
2001-01-25 10:37:55 +01:00
|
|
|
|
wahrnehmung(region * r, faction * f)
|
|
|
|
|
{
|
2007-02-17 13:34:07 +01:00
|
|
|
|
unit *u;
|
|
|
|
|
int w = 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-02-17 13:34:07 +01:00
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if (u->faction == f) {
|
|
|
|
|
if (eff_skill(u, SK_OBSERVATION, r) > w) {
|
|
|
|
|
w = eff_skill(u, SK_OBSERVATION, r);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-02-17 13:34:07 +01:00
|
|
|
|
return w;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2003-07-29 11:48:03 +02:00
|
|
|
|
static void
|
2004-06-21 18:45:27 +02:00
|
|
|
|
steal_cmd(unit * u, struct order * ord, request ** stealorders)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2007-02-17 13:34:07 +01:00
|
|
|
|
int n, i, id;
|
|
|
|
|
boolean goblin = false;
|
|
|
|
|
request * o;
|
|
|
|
|
unit * u2 = NULL;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region * r = u->region;
|
2007-02-17 13:34:07 +01:00
|
|
|
|
faction * f = NULL;
|
|
|
|
|
|
|
|
|
|
if (!fval(u->race, RCF_CANSTEAL)) {
|
2007-02-17 13:51:40 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "race_nosteal", "race", u->race));
|
2007-02-17 13:34:07 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-10-25 14:38:01 +02:00
|
|
|
|
if (fval(r->terrain, SEA_REGION) && u->race != new_race[RC_AQUARIAN]) {
|
2007-05-22 19:09:23 +02:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "error_onlandonly", ""));
|
2005-04-03 10:32:16 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2007-02-17 13:34:07 +01:00
|
|
|
|
if (r->planep && fval(r->planep, PFL_NOATTACK)) {
|
|
|
|
|
cmistake(u, ord, 270, MSG_INCOME);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
|
|
|
|
id = read_unitid(u->faction, r);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
u2 = findunitr(r, id);
|
|
|
|
|
|
|
|
|
|
if (u2 && u2->region==u->region) {
|
|
|
|
|
f = u2->faction;
|
|
|
|
|
} else {
|
|
|
|
|
f = dfindhash(id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (u2=r->units;u2;u2=u2->next) {
|
|
|
|
|
if (u2->faction == f && cansee(u->faction, r, u2, 0)) break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!u2) {
|
2007-02-10 22:47:51 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "feedback_unit_not_found", ""));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2005-01-08 00:19:10 +01:00
|
|
|
|
if (u2->faction->age < NewbieImmunity()) {
|
2006-01-29 02:55:36 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2004-06-21 18:45:27 +02:00
|
|
|
|
msg_feedback(u, ord, "newbie_immunity_error",
|
2005-01-08 00:19:10 +01:00
|
|
|
|
"turns", NewbieImmunity()));
|
2002-10-05 16:35:04 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2004-06-27 18:56:01 +02:00
|
|
|
|
if (u->faction->alliance!=NULL && u->faction->alliance == u2->faction->alliance) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 47, MSG_INCOME);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assert(u->region==u2->region);
|
|
|
|
|
if (!can_contact(r, u, u2)) {
|
2006-01-29 02:55:36 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "error60", ""));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2002-10-05 16:35:04 +02:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
n = eff_skill(u, SK_STEALTH, r) - wahrnehmung(r, f);
|
|
|
|
|
|
2006-07-26 21:02:55 +02:00
|
|
|
|
if (n <= 0) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
/* Wahrnehmung == Tarnung */
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (u->race != new_race[RC_GOBLIN] || eff_skill(u, SK_STEALTH, r) <= 3) {
|
2006-01-08 12:33:10 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("stealfail", "unit target", u, u2));
|
2006-07-26 21:02:55 +02:00
|
|
|
|
if (n==0) {
|
|
|
|
|
ADDMSG(&u2->faction->msgs, msg_message("stealdetect", "unit", u2));
|
|
|
|
|
} else {
|
|
|
|
|
ADDMSG(&u2->faction->msgs, msg_message("thiefdiscover", "unit target", u, u2));
|
|
|
|
|
}
|
2006-01-08 12:33:10 +01:00
|
|
|
|
return;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
} else {
|
2006-01-08 12:33:10 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs, msg_message("stealfatal", "unit target", u, u2));
|
|
|
|
|
ADDMSG(&u2->faction->msgs, msg_message("thiefdiscover", "unit target", u, u2));
|
2006-07-26 21:02:55 +02:00
|
|
|
|
n = 1;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
goblin = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
i = min(u->number, get_item(u,I_RING_OF_NIMBLEFINGER));
|
|
|
|
|
if (i > 0) {
|
|
|
|
|
n *= STEALINCOME * (u->number + i * 9);
|
|
|
|
|
} else {
|
|
|
|
|
n *= u->number * STEALINCOME;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u->wants = n;
|
|
|
|
|
|
|
|
|
|
/* wer dank unsichtbarkeitsringen klauen kann, muss nicht unbedingt ein
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* guter dieb sein, schliesslich macht man immer noch sehr viel laerm */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
o = (request *) calloc(1, sizeof(request));
|
|
|
|
|
o->unit = u;
|
|
|
|
|
o->qty = 1; /* Betrag steht in u->wants */
|
|
|
|
|
o->no = u2->no;
|
|
|
|
|
o->type.goblin = goblin; /* Merken, wenn Goblin-Spezialklau */
|
|
|
|
|
addlist(stealorders, o);
|
|
|
|
|
|
|
|
|
|
/* Nur soviel PRODUCEEXP wie auch tatsaechlich gemacht wurde */
|
|
|
|
|
|
2002-02-06 09:06:02 +01:00
|
|
|
|
produceexp(u, SK_STEALTH, min(n, u->number));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
expandentertainment(region * r)
|
|
|
|
|
{
|
|
|
|
|
unit *u;
|
|
|
|
|
int m = entertainmoney(r);
|
|
|
|
|
request *o;
|
|
|
|
|
|
|
|
|
|
for (o = &entertainers[0]; o != nextentertainer; ++o) {
|
|
|
|
|
double part = m / (double) entertaining;
|
|
|
|
|
u = o->unit;
|
|
|
|
|
if (entertaining <= m)
|
|
|
|
|
u->n = o->qty;
|
|
|
|
|
else
|
|
|
|
|
u->n = (int) (o->qty * part);
|
|
|
|
|
change_money(u, u->n);
|
|
|
|
|
rsetmoney(r, rmoney(r) - u->n);
|
|
|
|
|
m -= u->n;
|
2004-07-09 20:28:38 +02:00
|
|
|
|
entertaining -= o->qty;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
/* Nur soviel PRODUCEEXP wie auch tats<74>chlich gemacht wurde */
|
2002-02-06 09:06:02 +01:00
|
|
|
|
produceexp(u, SK_ENTERTAINMENT, min(u->n, u->number));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
add_income(u, IC_ENTERTAIN, o->qty, u->n);
|
2007-03-11 23:18:58 +01:00
|
|
|
|
fset(u, UFL_LONGACTION|UFL_NOTMOVING);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
void
|
2004-06-21 18:45:27 +02:00
|
|
|
|
entertain_cmd(unit * u, struct order * ord)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region * r = u->region;
|
2004-04-10 12:43:46 +02:00
|
|
|
|
int max_e;
|
|
|
|
|
request *o;
|
|
|
|
|
static int entertainbase = 0;
|
|
|
|
|
static int entertainperlevel = 0;
|
2004-05-08 12:32:46 +02:00
|
|
|
|
|
2004-04-10 12:43:46 +02:00
|
|
|
|
if (!entertainbase) {
|
|
|
|
|
const char * str = get_param(global.parameters, "entertain.base");
|
|
|
|
|
entertainbase = str?atoi(str):0;
|
|
|
|
|
}
|
|
|
|
|
if (!entertainperlevel) {
|
|
|
|
|
const char * str = get_param(global.parameters, "entertain.perlevel");
|
|
|
|
|
entertainperlevel = str?atoi(str):0;
|
|
|
|
|
}
|
|
|
|
|
if (fval(u, UFL_WERE)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 58, MSG_INCOME);
|
2004-04-10 12:43:46 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!effskill(u, SK_ENTERTAINMENT)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 58, MSG_INCOME);
|
2004-04-10 12:43:46 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (besieged(u)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 60, MSG_INCOME);
|
2004-04-10 12:43:46 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (u->ship && is_guarded(r, u, GUARD_CREWS)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 69, MSG_INCOME);
|
2004-04-10 12:43:46 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (is_cursed(r->attribs, C_DEPRESSION, 0)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 28, MSG_INCOME);
|
2004-04-10 12:43:46 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-05-08 12:32:46 +02:00
|
|
|
|
u->wants = u->number * (entertainbase + effskill(u, SK_ENTERTAINMENT)
|
|
|
|
|
* entertainperlevel);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
2007-06-26 11:32:28 +02:00
|
|
|
|
max_e = getuint();
|
2004-06-21 18:45:27 +02:00
|
|
|
|
if (max_e != 0) {
|
|
|
|
|
u->wants = min(u->wants,max_e);
|
|
|
|
|
}
|
2004-04-10 12:43:46 +02:00
|
|
|
|
o = nextentertainer++;
|
|
|
|
|
o->unit = u;
|
|
|
|
|
o->qty = u->wants;
|
|
|
|
|
entertaining += o->qty;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
expandwork(region * r)
|
|
|
|
|
{
|
|
|
|
|
int n, earnings;
|
|
|
|
|
/* n: verbleibende Einnahmen */
|
|
|
|
|
/* m: maximale Arbeiter */
|
|
|
|
|
int m = maxworkingpeasants(r);
|
2005-11-20 13:58:59 +01:00
|
|
|
|
int p_wage = wage(r, NULL, NULL);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
int verdienst = 0;
|
|
|
|
|
request *o;
|
|
|
|
|
|
|
|
|
|
for (o = &workers[0]; o != nextworker; ++o) {
|
|
|
|
|
unit * u = o->unit;
|
|
|
|
|
int workers;
|
|
|
|
|
|
|
|
|
|
if (u->number == 0) continue;
|
|
|
|
|
|
|
|
|
|
if (m>=working) workers = u->number;
|
|
|
|
|
else {
|
|
|
|
|
workers = u->number * m / working;
|
2006-02-19 23:43:56 +01:00
|
|
|
|
if (rng_int() % working < (u->number * m) % working) workers++;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
assert(workers>=0);
|
|
|
|
|
|
2005-11-20 13:58:59 +01:00
|
|
|
|
u->n = workers * wage(u->region, u->faction, u->race);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
m -= workers;
|
|
|
|
|
assert(m>=0);
|
|
|
|
|
|
|
|
|
|
change_money(u, u->n);
|
|
|
|
|
working -= o->unit->number;
|
|
|
|
|
add_income(u, IC_WORK, o->qty, u->n);
|
2007-03-11 23:18:58 +01:00
|
|
|
|
fset(u, UFL_LONGACTION|UFL_NOTMOVING);
|
2005-05-22 22:58:34 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
n = m * p_wage;
|
|
|
|
|
|
|
|
|
|
/* Der Rest wird von den Bauern verdient. n ist das uebriggebliebene
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* Geld. */
|
2007-06-27 22:39:49 +02:00
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
earnings = min(n, rpeasants(r) * p_wage) + verdienst;
|
|
|
|
|
/* Mehr oder weniger durch Trank "Riesengrass" oder "Faulobstschnaps" */
|
|
|
|
|
|
|
|
|
|
rsetmoney(r, rmoney(r) + earnings);
|
|
|
|
|
}
|
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
static void
|
2006-02-12 21:10:32 +01:00
|
|
|
|
work_cmd(unit * u, order * ord)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region * r = u->region;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
request *o;
|
|
|
|
|
int w;
|
|
|
|
|
|
2002-04-12 14:46:16 +02:00
|
|
|
|
if(fval(u, UFL_WERE)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 313, MSG_INCOME);
|
2002-04-12 14:46:16 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
if (besieged(u)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 60, MSG_INCOME);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (u->ship && is_guarded(r, u, GUARD_CREWS)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 69, MSG_INCOME);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2005-11-20 13:58:59 +01:00
|
|
|
|
w = wage(r, u->faction, u->race);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
u->wants = u->number * w;
|
|
|
|
|
o = nextworker++;
|
|
|
|
|
o->unit = u;
|
|
|
|
|
o->qty = u->number * w;
|
|
|
|
|
working += u->number;
|
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
static void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
expandtax(region * r, request * taxorders)
|
|
|
|
|
{
|
|
|
|
|
unit *u;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
expandorders(r, taxorders);
|
|
|
|
|
if (!norders) return;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i != norders && rmoney(r) > TAXFRACTION; i++) {
|
|
|
|
|
change_money(oa[i].unit, TAXFRACTION);
|
|
|
|
|
oa[i].unit->n += TAXFRACTION;
|
|
|
|
|
rsetmoney(r, rmoney(r) -TAXFRACTION);
|
|
|
|
|
}
|
|
|
|
|
free(oa);
|
|
|
|
|
|
2005-05-22 22:58:34 +02:00
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if (u->n >= 0) {
|
|
|
|
|
add_income(u, IC_TAX, u->wants, u->n);
|
2007-03-11 23:18:58 +01:00
|
|
|
|
fset(u, UFL_LONGACTION|UFL_NOTMOVING);
|
2005-05-22 22:58:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
void
|
2004-06-21 18:45:27 +02:00
|
|
|
|
tax_cmd(unit * u, struct order * ord, request ** taxorders)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
|
|
|
|
/* Steuern werden noch vor der Forschung eingetrieben */
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region * r = u->region;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
unit *u2;
|
|
|
|
|
int n;
|
|
|
|
|
request *o;
|
|
|
|
|
int max;
|
|
|
|
|
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (!humanoidrace(u->race) && u->faction != findfaction(MONSTER_FACTION)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 228, MSG_INCOME);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2002-04-21 19:22:48 +02:00
|
|
|
|
|
2002-04-12 14:46:16 +02:00
|
|
|
|
if (fval(u, UFL_WERE)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 228, MSG_INCOME);
|
2002-04-12 14:46:16 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
if (besieged(u)) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 60, MSG_INCOME);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
n = armedmen(u);
|
|
|
|
|
|
|
|
|
|
if (!n) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, ord, 48, MSG_INCOME);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
2004-06-21 18:45:27 +02:00
|
|
|
|
|
|
|
|
|
init_tokens(ord);
|
|
|
|
|
skip_token();
|
2007-06-26 11:32:28 +02:00
|
|
|
|
max = getuint();
|
2004-06-21 18:45:27 +02:00
|
|
|
|
|
|
|
|
|
if (max == 0) max = INT_MAX;
|
2001-12-10 01:13:39 +01:00
|
|
|
|
if (!playerrace(u->race)) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
u->wants = min(income(u), max);
|
|
|
|
|
} else {
|
|
|
|
|
u->wants = min(n * eff_skill(u, SK_TAXING, r) * 20, max);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
u2 = is_guarded(r, u, GUARD_TAX);
|
|
|
|
|
if (u2) {
|
2006-01-29 02:55:36 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2004-06-21 18:45:27 +02:00
|
|
|
|
msg_feedback(u, ord, "region_guarded", "guard", u2));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* die einnahmen werden in fraktionen von 10 silber eingeteilt: diese
|
2003-09-21 10:47:28 +02:00
|
|
|
|
* fraktionen werden dann bei eintreiben unter allen eintreibenden
|
|
|
|
|
* einheiten aufgeteilt. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
o = (request *) calloc(1, sizeof(request));
|
|
|
|
|
o->qty = u->wants / TAXFRACTION;
|
|
|
|
|
o->unit = u;
|
|
|
|
|
addlist(taxorders, o);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
/* ------------------------------------------------------------- */
|
|
|
|
|
|
2002-09-02 22:36:12 +02:00
|
|
|
|
void
|
2001-01-25 10:37:55 +01:00
|
|
|
|
produce(void)
|
|
|
|
|
{
|
2004-06-21 18:45:27 +02:00
|
|
|
|
region *r;
|
|
|
|
|
request *taxorders, *sellorders, *stealorders, *buyorders;
|
|
|
|
|
unit *u;
|
|
|
|
|
int todo;
|
|
|
|
|
|
|
|
|
|
/* das sind alles befehle, die 30 tage brauchen, und die in thisorder
|
|
|
|
|
* stehen! von allen 30-tage befehlen wird einfach der letzte verwendet
|
|
|
|
|
* (dosetdefaults).
|
|
|
|
|
*
|
|
|
|
|
* kaufen vor einnahmequellen. da man in einer region dasselbe produkt
|
|
|
|
|
* nicht kaufen und verkaufen kann, ist die reihenfolge wegen der
|
|
|
|
|
* produkte egal. nicht so wegen dem geld.
|
|
|
|
|
*
|
|
|
|
|
* lehren vor lernen. */
|
|
|
|
|
|
|
|
|
|
for (r = regions; r; r = r->next) {
|
2004-07-14 09:12:36 +02:00
|
|
|
|
boolean limited = true;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
|
|
|
|
|
assert(rmoney(r) >= 0);
|
|
|
|
|
assert(rpeasants(r) >= 0);
|
|
|
|
|
|
|
|
|
|
buyorders = 0;
|
|
|
|
|
sellorders = 0;
|
|
|
|
|
nextworker = &workers[0];
|
|
|
|
|
working = 0;
|
|
|
|
|
nextentertainer = &entertainers[0];
|
|
|
|
|
entertaining = 0;
|
|
|
|
|
taxorders = 0;
|
|
|
|
|
stealorders = 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
for (u = r->units; u; u = u->next) {
|
|
|
|
|
if (u->race == new_race[RC_SPELL] || fval(u, UFL_LONGACTION))
|
|
|
|
|
continue;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
if (u->race == new_race[RC_INSECT] && r_insectstalled(r) &&
|
|
|
|
|
!is_cursed(u->attribs, C_KAELTESCHUTZ,0))
|
|
|
|
|
continue;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-03-26 23:49:37 +02:00
|
|
|
|
if (fval(u, UFL_LONGACTION) && u->thisorder==NULL) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
cmistake(u, u->thisorder, 52, MSG_PRODUCE);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
if (!TradeDisabled()) {
|
|
|
|
|
order * ord;
|
2005-05-30 23:00:27 +02:00
|
|
|
|
boolean trader = false;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
for (ord = u->orders;ord;ord=ord->next) {
|
|
|
|
|
switch (get_keyword(ord)) {
|
|
|
|
|
case K_BUY:
|
|
|
|
|
buy(u, &buyorders, ord);
|
2005-05-30 23:00:27 +02:00
|
|
|
|
trader = true;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
break;
|
|
|
|
|
case K_SELL:
|
2004-07-14 09:12:36 +02:00
|
|
|
|
/* sell returns true if the sale is not limited
|
|
|
|
|
* by the region limit */
|
2004-08-21 03:55:56 +02:00
|
|
|
|
limited &= !sell(u, &sellorders, ord);
|
2005-05-30 23:00:27 +02:00
|
|
|
|
trader = true;
|
2004-06-21 18:45:27 +02:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-05-30 23:00:27 +02:00
|
|
|
|
if (trader) {
|
2004-06-21 18:45:27 +02:00
|
|
|
|
attrib * a = a_find(u->attribs, &at_trades);
|
|
|
|
|
if (a && a->data.i) {
|
|
|
|
|
produceexp(u, SK_TRADE, u->number);
|
|
|
|
|
}
|
2007-03-11 23:18:58 +01:00
|
|
|
|
fset(u, UFL_LONGACTION|UFL_NOTMOVING);
|
2004-06-21 18:45:27 +02:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
todo = get_keyword(u->thisorder);
|
|
|
|
|
if (todo == NOKEYWORD) continue;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2005-10-25 14:38:01 +02:00
|
|
|
|
if (fval(r->terrain, SEA_REGION) && u->race != new_race[RC_AQUARIAN]
|
2004-06-21 18:45:27 +02:00
|
|
|
|
&& !(u->race->flags & RCF_SWIM)
|
|
|
|
|
&& todo != K_STEAL && todo != K_SPY && todo != K_SABOTAGE)
|
|
|
|
|
continue;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
switch (todo) {
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
case K_ENTERTAIN:
|
|
|
|
|
entertain_cmd(u, u->thisorder);
|
|
|
|
|
break;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
case K_WORK:
|
2006-02-12 21:10:32 +01:00
|
|
|
|
if (playerrace(u->race)) work_cmd(u, u->thisorder);
|
2004-06-21 18:45:27 +02:00
|
|
|
|
else if (playerrace(u->faction->race)) {
|
2006-01-29 02:55:36 +01:00
|
|
|
|
ADDMSG(&u->faction->msgs,
|
2004-06-21 18:45:27 +02:00
|
|
|
|
msg_feedback(u, u->thisorder, "race_cantwork", "race", u->race));
|
|
|
|
|
}
|
|
|
|
|
break;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
case K_TAX:
|
|
|
|
|
tax_cmd(u, u->thisorder, &taxorders);
|
|
|
|
|
break;
|
2002-04-21 19:22:48 +02:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
case K_STEAL:
|
|
|
|
|
steal_cmd(u, u->thisorder, &stealorders);
|
|
|
|
|
break;
|
2002-04-21 19:22:48 +02:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
case K_SPY:
|
|
|
|
|
spy_cmd(u, u->thisorder);
|
|
|
|
|
break;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
case K_SABOTAGE:
|
|
|
|
|
sabotage_cmd(u, u->thisorder);
|
|
|
|
|
break;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2007-05-22 19:09:23 +02:00
|
|
|
|
case K_PLANT:
|
2005-05-11 22:24:51 +02:00
|
|
|
|
case K_BREED:
|
2004-06-21 18:45:27 +02:00
|
|
|
|
breed_cmd(u, u->thisorder);
|
|
|
|
|
break;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
case K_RESEARCH:
|
|
|
|
|
research_cmd(u, u->thisorder);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
/* Entertainment (expandentertainment) und Besteuerung (expandtax) vor den
|
|
|
|
|
* Befehlen, die den Bauern mehr Geld geben, damit man aus den Zahlen der
|
|
|
|
|
* letzten Runde berechnen kann, wieviel die Bauern f<EFBFBD>r Unterhaltung
|
|
|
|
|
* auszugeben bereit sind. */
|
|
|
|
|
if (entertaining) expandentertainment(r);
|
|
|
|
|
expandwork(r);
|
|
|
|
|
if (taxorders) expandtax(r, taxorders);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
/* An erster Stelle Kaufen (expandbuying), die Bauern so Geld bekommen, um
|
|
|
|
|
* nachher zu beim Verkaufen (expandselling) den Spielern abkaufen zu
|
|
|
|
|
* k<EFBFBD>nnen. */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
if (buyorders) expandbuying(r, buyorders);
|
2004-07-14 09:12:36 +02:00
|
|
|
|
|
|
|
|
|
if (sellorders) {
|
|
|
|
|
int limit = rpeasants(r) / TRADE_FRACTION;
|
2004-08-21 03:55:56 +02:00
|
|
|
|
if (rterrain(r) == T_DESERT && buildingtype_exists(r, bt_find("caravan")))
|
|
|
|
|
limit *= 2;
|
2004-07-14 09:12:36 +02:00
|
|
|
|
expandselling(r, sellorders, limited?limit:INT_MAX);
|
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
/* Die Spieler sollen alles Geld verdienen, bevor sie beklaut werden
|
|
|
|
|
* (expandstealing). */
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
if (stealorders) expandstealing(r, stealorders);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
assert(rmoney(r) >= 0);
|
|
|
|
|
assert(rpeasants(r) >= 0);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2004-06-21 18:45:27 +02:00
|
|
|
|
}
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|