server/src/common/kernel/ship.c

287 lines
5.4 KiB
C
Raw Normal View History

2001-01-25 10:37:55 +01:00
/* vi: set ts=2:
*
*
* Eressea PB(E)M host Copyright (C) 1998-2003
2001-01-25 10:37:55 +01:00
* Christian Schlittchen (corwin@amber.kn-bremen.de)
* Katja Zedel (katze@felidae.kn-bremen.de)
* Henning Peters (faroul@beyond.kn-bremen.de)
* Enno Rehling (enno@eressea-pbem.de)
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
*
* This program may not be used, modified or distributed without
* prior permission by the authors of Eressea.
*/
#include <config.h>
#include "eressea.h"
#include "ship.h"
/* kernel includes */
#include "build.h"
2001-01-25 10:37:55 +01:00
#include "unit.h"
#include "item.h"
#include "region.h"
#include "skill.h"
2001-01-25 10:37:55 +01:00
/* util includes */
#include <util/base36.h>
#include <util/event.h>
#include <util/language.h>
2007-06-20 02:34:02 +02:00
#include <util/lists.h>
#include <util/umlaut.h>
#include <util/xml.h>
2001-01-25 10:37:55 +01:00
/* libc includes */
#include <stdlib.h>
#include <string.h>
ship_typelist *shiptypes = NULL;
static local_names * snames;
const ship_type *
findshiptype(const xmlChar * name, const struct locale * lang)
{
local_names * sn = snames;
variant var;
while (sn) {
if (sn->lang==lang) break;
sn=sn->next;
}
if (!sn) {
struct ship_typelist * stl = shiptypes;
sn = calloc(sizeof(local_names), 1);
sn->next = snames;
sn->lang = lang;
while (stl) {
variant var;
const xmlChar * n = locale_string(lang, stl->type->name[0]);
var.v = (void*)stl->type;
addtoken(&sn->names, n, var);
stl = stl->next;
}
snames = sn;
}
if (findtoken(&sn->names, name, &var)==E_TOK_NOMATCH) return NULL;
return (const ship_type*)var.v;
}
2001-01-25 10:37:55 +01:00
const ship_type *
st_find(const char* name)
{
const struct ship_typelist * stl = shiptypes;
while (stl && strcmp(stl->type->name[0], name)) stl = stl->next;
2001-01-25 10:37:55 +01:00
return stl?stl->type:NULL;
}
void
st_register(const ship_type * type) {
struct ship_typelist * stl = malloc(sizeof(ship_type));
stl->type = type;
stl->next = shiptypes;
shiptypes = stl;
}
#define SMAXHASH 7919
2001-01-25 10:37:55 +01:00
ship *shiphash[SMAXHASH];
void
shash(ship * s)
{
ship *old = shiphash[s->no % SMAXHASH];
shiphash[s->no % SMAXHASH] = s;
s->nexthash = old;
}
void
sunhash(ship * s)
{
ship **show;
for (show = &shiphash[s->no % SMAXHASH]; *show; show = &(*show)->nexthash) {
if ((*show)->no == s->no)
break;
}
if (*show) {
assert(*show == s);
*show = (*show)->nexthash;
s->nexthash = 0;
}
}
static ship *
sfindhash(int i)
{
ship *old;
for (old = shiphash[i % SMAXHASH]; old; old = old->nexthash)
if (old->no == i)
return old;
return 0;
}
struct ship *
findship(int i)
{
return sfindhash(i);
}
struct ship *
findshipr(const region *r, int n)
{
ship * sh;
for (sh = r->ships; sh; sh = sh->next) {
if (sh->no == n) {
assert(sh->region == r);
return sh;
}
}
return 0;
}
2001-01-25 10:37:55 +01:00
void
damage_ship(ship * sh, double percent)
{
double damage = DAMAGE_SCALE * percent * sh->size + sh->damage;
sh->damage = (int)damage;
}
unit *
captain(ship *sh, region *r)
{
unit *u;
for(u = r->units; u; u = u->next)
if(u->ship == sh && fval(u, UFL_OWNER)) return u;
2001-01-25 10:37:55 +01:00
return NULL;
}
/* Alte Schiffstypen: */
ship *
new_ship(const ship_type * stype, const struct locale * lang, region * r)
2001-01-25 10:37:55 +01:00
{
static xmlChar buffer[7 + IDSIZE + 1];
2001-01-25 10:37:55 +01:00
ship *sh = (ship *) calloc(1, sizeof(ship));
sh->no = newcontainerid();
sh->coast = NODIRECTION;
sh->type = stype;
sh->region = r;
2001-01-25 10:37:55 +01:00
sprintf((char*)buffer, "%s %s", LOC(lang, stype->name[0]), shipid(sh));
sh->name = xstrdup(buffer);
2001-01-25 10:37:55 +01:00
shash(sh);
addlist(&r->ships, sh);
2001-01-25 10:37:55 +01:00
return sh;
}
void
destroy_ship(ship * sh)
2001-01-25 10:37:55 +01:00
{
region * r = sh->region;
unit * u = r->units;
2001-01-25 10:37:55 +01:00
while (u) {
if (u->ship == sh) {
leave_ship(u);
}
u = u->next;
}
sunhash(sh);
choplist(&r->ships, sh);
handle_event(sh->attribs, "destroy", sh);
2001-01-25 10:37:55 +01:00
}
const xmlChar *
write_shipname(const ship * sh, xmlChar * ibuf, size_t size)
{
snprintf((char*)ibuf, size, "%s (%s)", sh->name, itoa36(sh->no));
ibuf[size-1] = 0;
return ibuf;
}
const xmlChar *
2001-01-25 10:37:55 +01:00
shipname(const ship * sh)
{
typedef char name[OBJECTIDSIZE + 1];
static name idbuf[8];
static int nextbuf = 0;
char *ibuf = idbuf[(++nextbuf) % 8];
return write_shipname(sh, (xmlChar*)ibuf, sizeof(name));
2001-01-25 10:37:55 +01:00
}
int
shipcapacity (const ship * sh)
{
int i;
/* sonst ist construction:: size nicht ship_type::maxsize */
assert(!sh->type->construction || sh->type->construction->improvement==NULL);
if (sh->type->construction && sh->size!=sh->type->construction->maxsize)
return 0;
#ifdef SHIPDAMAGE
i = ((sh->size * DAMAGE_SCALE - sh->damage) / DAMAGE_SCALE)
* sh->type->cargo / sh->size;
i += ((sh->size * DAMAGE_SCALE - sh->damage) % DAMAGE_SCALE)
* sh->type->cargo / (sh->size*DAMAGE_SCALE);
#else
i = sh->type->cargo;
#endif
return i;
}
void
getshipweight(const ship * sh, int *sweight, int *scabins)
{
unit * u;
*sweight = 0;
*scabins = 0;
for (u = sh->region->units; u; u = u->next)
if (u->ship == sh) {
*sweight += weight(u);
*scabins += u->number;
}
}
2001-01-25 10:37:55 +01:00
unit *
2004-06-21 18:45:27 +02:00
shipowner(const ship * sh)
2001-01-25 10:37:55 +01:00
{
unit *u;
unit *first = NULL;
2004-06-21 18:45:27 +02:00
const region * r = sh->region;
2001-01-25 10:37:55 +01:00
2004-06-21 18:45:27 +02:00
/* Pr<50>fen ob Eigent<6E>mer am leben. */
2001-01-25 10:37:55 +01:00
for (u = r->units; u; u = u->next) {
if (u->ship == sh) {
if (!first && u->number > 0)
first = u;
if (fval(u, UFL_OWNER) && u->number > 0)
2001-01-25 10:37:55 +01:00
return u;
if (u->number == 0)
freset(u, UFL_OWNER);
2001-01-25 10:37:55 +01:00
}
}
/* Eigent<6E>mer tot oder kein Eigent<6E>mer vorhanden. Erste lebende Einheit
* nehmen. */
if (first)
fset(first, UFL_OWNER);
2001-01-25 10:37:55 +01:00
return first;
}
void
register_ships(void)
{
2002-02-03 09:31:39 +01:00
}