2010-08-08 10:06:34 +02:00
|
|
|
/*
|
2019-07-14 09:37:35 +02:00
|
|
|
Copyright (c) 1998-2019, Enno Rehling <enno@eressea.de>
|
2015-01-30 20:37:14 +01:00
|
|
|
Katja Zedel <katze@felidae.kn-bremen.de
|
|
|
|
Christian Schlittchen <corwin@amber.kn-bremen.de>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
purpose with or without fee is hereby granted, provided that the above
|
|
|
|
copyright notice and this permission notice appear in all copies.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include <platform.h>
|
|
|
|
#include <kernel/config.h>
|
|
|
|
#include "terrain.h"
|
2017-08-21 20:18:19 +02:00
|
|
|
|
|
|
|
#include <attributes/racename.h>
|
|
|
|
#include <spells/regioncurse.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
#include "curse.h"
|
|
|
|
#include "region.h"
|
|
|
|
#include "resources.h"
|
2017-08-21 20:18:19 +02:00
|
|
|
#include "terrainid.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
#include <util/log.h>
|
2018-09-29 11:37:17 +02:00
|
|
|
#include <kernel/attrib.h>
|
2017-12-28 18:29:40 +01:00
|
|
|
#include <util/strings.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2018-03-04 20:03:13 +01:00
|
|
|
const char *terrainnames[MAXTERRAINS] = {
|
2015-01-30 20:37:14 +01:00
|
|
|
"ocean",
|
|
|
|
"plain",
|
|
|
|
"swamp",
|
|
|
|
"desert",
|
|
|
|
"highland",
|
|
|
|
"mountain",
|
|
|
|
"glacier",
|
|
|
|
"firewall",
|
|
|
|
"fog",
|
|
|
|
"thickfog",
|
|
|
|
"volcano",
|
|
|
|
"activevolcano",
|
|
|
|
"iceberg_sleep",
|
2015-03-29 14:03:56 +02:00
|
|
|
"iceberg"
|
2010-08-08 10:06:34 +02:00
|
|
|
};
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static terrain_type *registered_terrains;
|
2017-11-06 20:29:26 +01:00
|
|
|
static int terrain_changes = 1;
|
|
|
|
|
|
|
|
bool terrain_changed(int *cache) {
|
|
|
|
assert(cache);
|
|
|
|
if (*cache != terrain_changes) {
|
|
|
|
*cache = terrain_changes;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-12-30 02:02:10 +01:00
|
|
|
void free_terrains(void)
|
2011-03-13 01:59:32 +01:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
while (registered_terrains) {
|
|
|
|
terrain_type * t = registered_terrains;
|
|
|
|
registered_terrains = t->next;
|
|
|
|
free(t->_name);
|
2015-10-14 21:37:27 +02:00
|
|
|
free(t->herbs);
|
2015-09-11 12:17:35 +02:00
|
|
|
if (t->production) {
|
2018-02-11 15:43:24 +01:00
|
|
|
int n;
|
2015-09-11 12:17:35 +02:00
|
|
|
for (n = 0; t->production[n].type; ++n) {
|
|
|
|
free(t->production[n].base);
|
|
|
|
free(t->production[n].divisor);
|
|
|
|
free(t->production[n].startlevel);
|
|
|
|
}
|
|
|
|
free(t->production);
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
free(t);
|
|
|
|
}
|
2017-11-06 20:29:26 +01:00
|
|
|
++terrain_changes;
|
2011-03-13 01:59:32 +01:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
const terrain_type *terrains(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-06-14 02:36:05 +02:00
|
|
|
return registered_terrains;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static const char *plain_name(const struct region *r)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-06-14 02:36:05 +02:00
|
|
|
/* TODO: xml defined */
|
|
|
|
if (r_isforest(r)) {
|
2015-01-30 20:37:14 +01:00
|
|
|
return "forest";
|
2014-06-14 02:36:05 +02:00
|
|
|
}
|
|
|
|
return r->terrain->_name;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2014-06-14 02:36:05 +02:00
|
|
|
static terrain_type *terrain_find_i(const char *name)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-06-14 02:36:05 +02:00
|
|
|
terrain_type *terrain;
|
|
|
|
for (terrain = registered_terrains; terrain; terrain = terrain->next) {
|
2015-01-30 20:37:14 +01:00
|
|
|
if (strcmp(terrain->_name, name) == 0) {
|
|
|
|
break;
|
|
|
|
}
|
2014-06-14 02:36:05 +02:00
|
|
|
}
|
|
|
|
return terrain;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2018-03-04 20:03:13 +01:00
|
|
|
static terrain_type *get_terrain_i(const char *name) {
|
2014-06-14 02:36:05 +02:00
|
|
|
return terrain_find_i(name);
|
|
|
|
}
|
|
|
|
|
2018-03-04 20:03:13 +01:00
|
|
|
const terrain_type *get_terrain(const char *name) {
|
|
|
|
return get_terrain_i(name);
|
|
|
|
}
|
|
|
|
|
2014-06-16 06:19:47 +02:00
|
|
|
terrain_type * get_or_create_terrain(const char *name) {
|
2014-06-14 02:36:05 +02:00
|
|
|
terrain_type *terrain = terrain_find_i(name);
|
|
|
|
if (!terrain) {
|
2017-11-06 20:29:26 +01:00
|
|
|
++terrain_changes;
|
2018-12-15 19:38:40 +01:00
|
|
|
terrain = (terrain_type *)calloc(1, sizeof(terrain_type));
|
2014-06-14 02:36:05 +02:00
|
|
|
if (terrain) {
|
2017-12-28 18:29:40 +01:00
|
|
|
terrain->_name = str_strdup(name);
|
2014-06-14 02:36:05 +02:00
|
|
|
terrain->next = registered_terrains;
|
|
|
|
registered_terrains = terrain;
|
2018-03-04 20:03:13 +01:00
|
|
|
if (strcmp(terrainnames[T_PLAIN], name) == 0) {
|
2017-02-18 21:15:14 +01:00
|
|
|
/* TODO: this is awful, it belongs in config */
|
2014-06-14 02:36:05 +02:00
|
|
|
terrain->name = &plain_name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
return terrain;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2018-03-04 20:03:13 +01:00
|
|
|
static terrain_type *newterrains[MAXTERRAINS];
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
const struct terrain_type *newterrain(terrain_t t)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2018-03-04 20:03:13 +01:00
|
|
|
static int changed;
|
2017-11-05 17:00:40 +01:00
|
|
|
const struct terrain_type *result;
|
|
|
|
if (t == NOTERRAIN) {
|
2015-01-30 20:37:14 +01:00
|
|
|
return NULL;
|
2017-11-05 17:00:40 +01:00
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
assert(t >= 0);
|
|
|
|
assert(t < MAXTERRAINS);
|
2018-03-04 20:03:13 +01:00
|
|
|
if (terrain_changed(&changed)) {
|
|
|
|
memset(newterrains, 0, sizeof(newterrains));
|
|
|
|
result = NULL;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result = newterrains[t];
|
|
|
|
}
|
2017-11-05 17:00:40 +01:00
|
|
|
if (!result) {
|
2018-03-04 20:03:13 +01:00
|
|
|
result = newterrains[t] = get_terrain_i(terrainnames[t]);
|
2017-11-05 17:00:40 +01:00
|
|
|
}
|
|
|
|
if (!result) {
|
2017-11-06 21:04:30 +01:00
|
|
|
log_error("no such terrain: %s.", terrainnames[t]);
|
2017-11-05 16:43:01 +01:00
|
|
|
}
|
2017-11-05 17:00:40 +01:00
|
|
|
return result;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
terrain_t oldterrain(const struct terrain_type * terrain)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
terrain_t t;
|
2015-08-06 16:52:24 +02:00
|
|
|
assert(terrain);
|
2015-01-30 20:37:14 +01:00
|
|
|
for (t = 0; t != MAXTERRAINS; ++t) {
|
|
|
|
if (newterrains[t] == terrain)
|
|
|
|
return t;
|
|
|
|
}
|
2015-05-15 07:59:06 +02:00
|
|
|
log_debug("%s is not a classic terrain.\n", terrain->_name);
|
2011-03-07 08:02:35 +01:00
|
|
|
return NOTERRAIN;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
const char *terrain_name(const struct region *r)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
if (r->attribs) {
|
|
|
|
attrib *a = a_find(r->attribs, &at_racename);
|
|
|
|
if (a) {
|
|
|
|
const char *str = get_racename(a);
|
|
|
|
if (str)
|
|
|
|
return str;
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
if (r->terrain->name != NULL) {
|
|
|
|
return r->terrain->name(r);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
else if (fval(r->terrain, SEA_REGION)) {
|
2017-08-21 20:18:19 +02:00
|
|
|
if (curse_active(get_curse(r->attribs, &ct_maelstrom))) {
|
2015-01-30 20:37:14 +01:00
|
|
|
return "maelstrom";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return r->terrain->_name;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
void init_terrains(void)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
terrain_t t;
|
|
|
|
for (t = 0; t != MAXTERRAINS; ++t) {
|
2018-03-04 20:03:13 +01:00
|
|
|
terrain_type *newterrain = newterrains[t];
|
2015-01-30 20:37:14 +01:00
|
|
|
if (newterrain != NULL)
|
|
|
|
continue;
|
2017-11-06 21:04:30 +01:00
|
|
|
if (terrainnames[t] != NULL) {
|
2018-03-04 20:03:13 +01:00
|
|
|
newterrain = get_terrain_i(terrainnames[t]);
|
2015-01-30 20:37:14 +01:00
|
|
|
if (newterrain != NULL) {
|
|
|
|
newterrains[t] = newterrain;
|
|
|
|
}
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
}
|