The static analyzer complained about use of 'obj' maybe being Null when used in an impossible warning, but that warning will never appear for the case where obj is actually Null. Add an assert() that should let it figure that out, and move the impossible check inside the 'else' clause where the check matters. (Either of those by itself ought to be adequate to pacify the analyzer.)
887 lines
29 KiB
C
887 lines
29 KiB
C
/* NetHack 3.7 light.c $NHDT-Date: 1702680171 2023/12/15 22:42:51 $ $NHDT-Branch: NetHack-3.7 $:$NHDT-Revision: 1.66 $ */
|
|
/* Copyright (c) Dean Luick, 1994 */
|
|
/* NetHack may be freely redistributed. See license for details. */
|
|
|
|
#include "hack.h"
|
|
#include <assert.h>
|
|
|
|
/*
|
|
* Mobile light sources.
|
|
*
|
|
* This implementation minimizes memory at the expense of extra
|
|
* recalculations.
|
|
*
|
|
* Light sources are "things" that have a physical position and range.
|
|
* They have a type, which gives us information about them. Currently
|
|
* they are only attached to objects and monsters. Note well: the
|
|
* polymorphed-player handling assumes that both gy.youmonst.m_id and
|
|
* gy.youmonst.mx will always remain 0.
|
|
*
|
|
* Light sources, like timers, either follow game play (RANGE_GLOBAL) or
|
|
* stay on a level (RANGE_LEVEL). Light sources are unique by their
|
|
* (type, id) pair. For light sources attached to objects, this id
|
|
* is a pointer to the object.
|
|
*
|
|
* The major working function is do_light_sources(). It is called
|
|
* when the vision system is recreating its "could see" array. Here
|
|
* we add a flag (TEMP_LIT) to the array for all locations that are lit
|
|
* via a light source. The bad part of this is that we have to
|
|
* re-calculate the LOS of each light source every time the vision
|
|
* system runs. Even if the light sources and any topology (vision blocking
|
|
* positions) have not changed. The good part is that no extra memory
|
|
* is used, plus we don't have to figure out how far the sources have moved,
|
|
* or if the topology has changed.
|
|
*
|
|
* The structure of the save/restore mechanism is amazingly similar to
|
|
* the timer save/restore. This is because they both have the same
|
|
* principals of having pointers into objects that must be recalculated
|
|
* across saves and restores.
|
|
*/
|
|
|
|
/* flags */
|
|
#define LSF_SHOW 0x1 /* display the light source */
|
|
#define LSF_NEEDS_FIXUP 0x2 /* need oid fixup */
|
|
|
|
static light_source *new_light_core(coordxy, coordxy, int, int, anything *);
|
|
static void discard_flashes(void);
|
|
static void write_ls(NHFILE *, light_source *);
|
|
static int maybe_write_ls(NHFILE *, int, boolean);
|
|
|
|
/* imported from vision.c, for small circles */
|
|
extern const coordxy circle_data[];
|
|
extern const coordxy circle_start[];
|
|
|
|
|
|
/* Create a new light source. Caller (and extern.h) doesn't need to know
|
|
anything about type 'light_source'. */
|
|
void
|
|
new_light_source(coordxy x, coordxy y, int range, int type, anything *id)
|
|
{
|
|
(void) new_light_core(x, y, range, type, id);
|
|
}
|
|
|
|
/* Create a new light source and return it. Only used within this file. */
|
|
static light_source *
|
|
new_light_core(coordxy x, coordxy y, int range, int type, anything *id)
|
|
{
|
|
light_source *ls;
|
|
|
|
if (range > MAX_RADIUS || range < 0
|
|
/* camera flash uses radius 0 and passes Null object */
|
|
|| (range == 0 && (type != LS_OBJECT || id->a_obj != 0))) {
|
|
impossible("new_light_source: illegal range %d", range);
|
|
return (light_source *) 0;
|
|
}
|
|
|
|
ls = (light_source *) alloc(sizeof *ls);
|
|
|
|
(void) memset((genericptr_t) ls, 0, sizeof (light_source));
|
|
ls->next = gl.light_base;
|
|
ls->x = x;
|
|
ls->y = y;
|
|
ls->range = range;
|
|
ls->type = type;
|
|
ls->id = *id;
|
|
ls->flags = 0;
|
|
gl.light_base = ls;
|
|
|
|
gv.vision_full_recalc = 1; /* make the source show up */
|
|
return ls;
|
|
}
|
|
|
|
/*
|
|
* Delete a light source. This assumes only one light source is attached
|
|
* to an object at a time.
|
|
*/
|
|
void
|
|
del_light_source(int type, anything *id)
|
|
{
|
|
light_source *curr, *prev;
|
|
anything tmp_id;
|
|
|
|
tmp_id = cg.zeroany;
|
|
/* need to be prepared for dealing a with light source which
|
|
has only been partially restored during a level change
|
|
(in particular: chameleon vs prot. from shape changers) */
|
|
switch (type) {
|
|
case LS_OBJECT:
|
|
tmp_id.a_uint = id->a_obj ? id->a_obj->o_id : 0;
|
|
break;
|
|
case LS_MONSTER:
|
|
tmp_id.a_uint = id->a_monst->m_id;
|
|
break;
|
|
default:
|
|
tmp_id.a_uint = 0;
|
|
break;
|
|
}
|
|
|
|
for (prev = 0, curr = gl.light_base; curr; prev = curr, curr = curr->next) {
|
|
if (curr->type != type)
|
|
continue;
|
|
if (curr->id.a_obj
|
|
== ((curr->flags & LSF_NEEDS_FIXUP) ? tmp_id.a_obj : id->a_obj)) {
|
|
if (prev)
|
|
prev->next = curr->next;
|
|
else
|
|
gl.light_base = curr->next;
|
|
|
|
free((genericptr_t) curr);
|
|
gv.vision_full_recalc = 1;
|
|
return;
|
|
}
|
|
}
|
|
impossible("del_light_source: not found type=%d, id=%s", type,
|
|
fmt_ptr((genericptr_t) id->a_obj));
|
|
}
|
|
|
|
/* Mark locations that are temporarily lit via mobile light sources. */
|
|
void
|
|
do_light_sources(seenV **cs_rows)
|
|
{
|
|
coordxy x, y, min_x, max_x, max_y;
|
|
int offset;
|
|
const coordxy *limits;
|
|
short at_hero_range = 0;
|
|
light_source *ls;
|
|
seenV *row;
|
|
|
|
for (ls = gl.light_base; ls; ls = ls->next) {
|
|
ls->flags &= ~LSF_SHOW;
|
|
|
|
/*
|
|
* Check for moved light sources. It may be possible to
|
|
* save some effort if an object has not moved, but not in
|
|
* the current setup -- we need to recalculate for every
|
|
* vision recalc.
|
|
*/
|
|
if (ls->type == LS_OBJECT) {
|
|
if (ls->range == 0 /* camera flash; caller has set ls->{x,y} */
|
|
|| get_obj_location(ls->id.a_obj, &ls->x, &ls->y, 0))
|
|
ls->flags |= LSF_SHOW;
|
|
} else if (ls->type == LS_MONSTER) {
|
|
if (get_mon_location(ls->id.a_monst, &ls->x, &ls->y, 0))
|
|
ls->flags |= LSF_SHOW;
|
|
}
|
|
|
|
/* minor optimization: don't bother with duplicate light sources
|
|
at hero */
|
|
if (u_at(ls->x, ls->y)) {
|
|
if (at_hero_range >= ls->range)
|
|
ls->flags &= ~LSF_SHOW;
|
|
else
|
|
at_hero_range = ls->range;
|
|
}
|
|
|
|
if (ls->flags & LSF_SHOW) {
|
|
/*
|
|
* Walk the points in the circle and see if they are
|
|
* visible from the center. If so, mark'em.
|
|
*
|
|
* Kevin's tests indicated that doing this brute-force
|
|
* method is faster for radius <= 3 (or so).
|
|
*/
|
|
limits = circle_ptr(ls->range);
|
|
if ((max_y = (ls->y + ls->range)) >= ROWNO)
|
|
max_y = ROWNO - 1;
|
|
if ((y = (ls->y - ls->range)) < 0)
|
|
y = 0;
|
|
for (; y <= max_y; y++) {
|
|
row = cs_rows[y];
|
|
offset = limits[abs(y - ls->y)];
|
|
if ((min_x = (ls->x - offset)) < 1)
|
|
min_x = 1;
|
|
if ((max_x = (ls->x + offset)) >= COLNO)
|
|
max_x = COLNO - 1;
|
|
|
|
if (u_at(ls->x, ls->y)) {
|
|
/*
|
|
* If the light source is located at the hero, then
|
|
* we can use the COULD_SEE bits already calculated
|
|
* by the vision system. More importantly than
|
|
* this optimization, is that it allows the vision
|
|
* system to correct problems with clear_path().
|
|
* The function clear_path() is a simple LOS
|
|
* path checker that doesn't go out of its way to
|
|
* make things look "correct". The vision system
|
|
* does this.
|
|
*/
|
|
for (x = min_x; x <= max_x; x++)
|
|
if (row[x] & COULD_SEE)
|
|
row[x] |= TEMP_LIT;
|
|
} else {
|
|
for (x = min_x; x <= max_x; x++)
|
|
if ((ls->x == x && ls->y == y)
|
|
|| clear_path((int) ls->x, (int) ls->y, x, y))
|
|
row[x] |= TEMP_LIT;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* lit 'obj' has been thrown or kicked and is passing through x,y on the
|
|
way to its destination; show its light so that hero has a chance to
|
|
remember terrain, objects, and monsters being revealed;
|
|
if 'obj' is Null, <x,y> is being hit by a camera's light flash */
|
|
void
|
|
show_transient_light(struct obj *obj, coordxy x, coordxy y)
|
|
{
|
|
light_source *ls = 0;
|
|
anything cameraflash;
|
|
struct monst *mon;
|
|
int radius_squared;
|
|
|
|
/* Null object indicates camera flash */
|
|
if (!obj) {
|
|
/* no need to temporarily light an already lit spot */
|
|
if (levl[x][y].lit)
|
|
return;
|
|
|
|
cameraflash = cg.zeroany;
|
|
/* radius 0 will just light <x,y>; cameraflash.a_obj is Null */
|
|
ls = new_light_core(x, y, 0, LS_OBJECT, &cameraflash);
|
|
/* pacify static analysis; 'ls' is never Null for
|
|
new_light_core(,,0,LS_OBJECT,&zeroany) */
|
|
assert(ls != NULL);
|
|
} else {
|
|
/* thrown or kicked object which is emitting light; validate its
|
|
light source to obtain its radius (for monster sightings) */
|
|
for (ls = gl.light_base; ls; ls = ls->next) {
|
|
if (ls->type != LS_OBJECT)
|
|
continue;
|
|
if (ls->id.a_obj == obj)
|
|
break;
|
|
}
|
|
assert(obj != NULL); /* necessary condition to get into this 'else' */
|
|
if (!ls || obj->where != OBJ_FREE) {
|
|
impossible("transient light %s %s %s not %s?",
|
|
obj->lamplit ? "lit" : "unlit",
|
|
simpleonames(obj), otense(obj, "are"),
|
|
!ls ? "a light source" : "free");
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (obj) /* put lit candle or lamp temporarily on the map */
|
|
place_object(obj, gb.bhitpos.x, gb.bhitpos.y);
|
|
else /* camera flash: no object; directly set light source's location */
|
|
ls->x = x, ls->y = y;
|
|
|
|
/* full recalc; runs do_light_sources() */
|
|
vision_recalc(0);
|
|
flush_screen(0);
|
|
|
|
radius_squared = ls->range * ls->range;
|
|
for (mon = fmon; mon; mon = mon->nmon) {
|
|
if (DEADMONSTER(mon) || (mon->isgd && !mon->mx))
|
|
continue;
|
|
/* light range is the radius of a circle and we're limiting
|
|
canseemon() to a square enclosing that circle, but setting
|
|
mtemplit 'erroneously' for a seen monster is not a problem;
|
|
it just flags monsters for another canseemon() check when
|
|
'obj' has reached its destination after missile traversal */
|
|
if (dist2(mon->mx, mon->my, x, y) <= radius_squared) {
|
|
if (canseemon(mon))
|
|
mon->mtemplit = 1;
|
|
}
|
|
/* [what about worm tails?] */
|
|
}
|
|
|
|
if (obj) { /* take thrown/kicked candle or lamp off the map */
|
|
nh_delay_output();
|
|
remove_object(obj);
|
|
}
|
|
}
|
|
|
|
/* delete any camera flash light sources and draw "remembered, unseen
|
|
monster" glyph at locations where a monster was flagged for being
|
|
visible during transient light movement but can't be seen now */
|
|
void
|
|
transient_light_cleanup(void)
|
|
{
|
|
struct monst *mon;
|
|
int mtempcount;
|
|
|
|
/* in case we're cleaning up a camera flash, remove all object light
|
|
sources which aren't associated with a specific object */
|
|
discard_flashes();
|
|
if (gv.vision_full_recalc) /* set by del_light_source() */
|
|
vision_recalc(0);
|
|
|
|
/* for thrown/kicked candle or lamp or for camera flash, some
|
|
monsters may have been mapped in light which has now gone away
|
|
so need to be replaced by "remembered, unseen monster" glyph */
|
|
mtempcount = 0;
|
|
for (mon = fmon; mon; mon = mon->nmon) {
|
|
if (DEADMONSTER(mon))
|
|
continue;
|
|
if (mon->mtemplit) {
|
|
mon->mtemplit = 0;
|
|
++mtempcount;
|
|
if (!canspotmon(mon))
|
|
map_invisible(mon->mx, mon->my);
|
|
}
|
|
}
|
|
if (mtempcount)
|
|
flush_screen(0);
|
|
}
|
|
|
|
/* camera flashes have Null object; caller wants to get rid of them now */
|
|
static void
|
|
discard_flashes(void)
|
|
{
|
|
light_source *ls, *nxt_ls;
|
|
|
|
for (ls = gl.light_base; ls; ls = nxt_ls) {
|
|
nxt_ls = ls->next;
|
|
if (ls->type != LS_OBJECT)
|
|
continue;
|
|
if (!ls->id.a_obj)
|
|
del_light_source(LS_OBJECT, &ls->id);
|
|
}
|
|
}
|
|
|
|
/* (mon->mx == 0) implies migrating */
|
|
#define mon_is_local(mon) ((mon)->mx > 0)
|
|
|
|
struct monst *
|
|
find_mid(unsigned nid, unsigned fmflags)
|
|
{
|
|
struct monst *mtmp;
|
|
|
|
if (!nid)
|
|
return &gy.youmonst;
|
|
if (fmflags & FM_FMON)
|
|
for (mtmp = fmon; mtmp; mtmp = mtmp->nmon)
|
|
if (!DEADMONSTER(mtmp) && mtmp->m_id == nid)
|
|
return mtmp;
|
|
if (fmflags & FM_MIGRATE)
|
|
for (mtmp = gm.migrating_mons; mtmp; mtmp = mtmp->nmon)
|
|
if (mtmp->m_id == nid)
|
|
return mtmp;
|
|
if (fmflags & FM_MYDOGS)
|
|
for (mtmp = gm.mydogs; mtmp; mtmp = mtmp->nmon)
|
|
if (mtmp->m_id == nid)
|
|
return mtmp;
|
|
return (struct monst *) 0;
|
|
}
|
|
|
|
/* Save all light sources of the given range. */
|
|
void
|
|
save_light_sources(NHFILE *nhfp, int range)
|
|
{
|
|
int count, actual, is_global;
|
|
light_source **prev, *curr;
|
|
|
|
/* camera flash light sources have Null object and would trigger
|
|
impossible("no id!") below; they can only happen here if we're
|
|
in the midst of a panic save and they wouldn't be useful after
|
|
restore so just throw any that are present away */
|
|
discard_flashes();
|
|
gv.vision_full_recalc = 0;
|
|
|
|
if (perform_bwrite(nhfp)) {
|
|
count = maybe_write_ls(nhfp, range, FALSE);
|
|
if (nhfp->structlevel) {
|
|
bwrite(nhfp->fd, (genericptr_t) &count, sizeof count);
|
|
}
|
|
actual = maybe_write_ls(nhfp, range, TRUE);
|
|
if (actual != count)
|
|
panic("counted %d light sources, wrote %d! [range=%d]", count,
|
|
actual, range);
|
|
}
|
|
|
|
if (release_data(nhfp)) {
|
|
for (prev = &gl.light_base; (curr = *prev) != 0; ) {
|
|
if (!curr->id.a_monst) {
|
|
impossible("save_light_sources: no id! [range=%d]", range);
|
|
is_global = 0;
|
|
} else
|
|
switch (curr->type) {
|
|
case LS_OBJECT:
|
|
is_global = !obj_is_local(curr->id.a_obj);
|
|
break;
|
|
case LS_MONSTER:
|
|
is_global = !mon_is_local(curr->id.a_monst);
|
|
break;
|
|
default:
|
|
is_global = 0;
|
|
impossible("save_light_sources: bad type (%d) [range=%d]",
|
|
curr->type, range);
|
|
break;
|
|
}
|
|
/* if global and not doing local, or vice versa, remove it */
|
|
if (is_global ^ (range == RANGE_LEVEL)) {
|
|
*prev = curr->next;
|
|
free((genericptr_t) curr);
|
|
} else {
|
|
prev = &(*prev)->next;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Pull in the structures from disk, but don't recalculate the object
|
|
* pointers.
|
|
*/
|
|
void
|
|
restore_light_sources(NHFILE *nhfp)
|
|
{
|
|
int count = 0;
|
|
light_source *ls;
|
|
|
|
/* restore elements */
|
|
if (nhfp->structlevel)
|
|
mread(nhfp->fd, (genericptr_t) &count, sizeof count);
|
|
|
|
while (count-- > 0) {
|
|
ls = (light_source *) alloc(sizeof(light_source));
|
|
if (nhfp->structlevel)
|
|
mread(nhfp->fd, (genericptr_t) ls, sizeof(light_source));
|
|
ls->next = gl.light_base;
|
|
gl.light_base = ls;
|
|
}
|
|
}
|
|
|
|
DISABLE_WARNING_FORMAT_NONLITERAL
|
|
|
|
/* to support '#stats' wizard-mode command */
|
|
void
|
|
light_stats(const char *hdrfmt, char *hdrbuf, long *count, long *size)
|
|
{
|
|
light_source *ls;
|
|
|
|
Sprintf(hdrbuf, hdrfmt, (long) sizeof (light_source));
|
|
*count = *size = 0L;
|
|
for (ls = gl.light_base; ls; ls = ls->next) {
|
|
++*count;
|
|
*size += (long) sizeof *ls;
|
|
}
|
|
}
|
|
|
|
RESTORE_WARNING_FORMAT_NONLITERAL
|
|
|
|
/* Relink all lights that are so marked. */
|
|
void
|
|
relink_light_sources(boolean ghostly)
|
|
{
|
|
char which;
|
|
unsigned nid;
|
|
light_source *ls;
|
|
|
|
for (ls = gl.light_base; ls; ls = ls->next) {
|
|
if (ls->flags & LSF_NEEDS_FIXUP) {
|
|
if (ls->type == LS_OBJECT || ls->type == LS_MONSTER) {
|
|
if (ghostly) {
|
|
if (!lookup_id_mapping(ls->id.a_uint, &nid))
|
|
impossible("relink_light_sources: no id mapping");
|
|
} else
|
|
nid = ls->id.a_uint;
|
|
if (ls->type == LS_OBJECT) {
|
|
which = 'o';
|
|
ls->id.a_obj = find_oid(nid);
|
|
} else {
|
|
which = 'm';
|
|
ls->id.a_monst = find_mid(nid, FM_EVERYWHERE);
|
|
}
|
|
if (!ls->id.a_monst)
|
|
impossible("relink_light_sources: cant find %c_id %d",
|
|
which, nid);
|
|
} else
|
|
impossible("relink_light_sources: bad type (%d)", ls->type);
|
|
|
|
ls->flags &= ~LSF_NEEDS_FIXUP;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Part of the light source save routine. Count up the number of light
|
|
* sources that would be written. If write_it is true, actually write
|
|
* the light source out.
|
|
*/
|
|
static int
|
|
maybe_write_ls(NHFILE *nhfp, int range, boolean write_it)
|
|
{
|
|
int count = 0, is_global;
|
|
light_source *ls;
|
|
|
|
for (ls = gl.light_base; ls; ls = ls->next) {
|
|
if (!ls->id.a_monst) {
|
|
impossible("maybe_write_ls: no id! [range=%d]", range);
|
|
continue;
|
|
}
|
|
switch (ls->type) {
|
|
case LS_OBJECT:
|
|
is_global = !obj_is_local(ls->id.a_obj);
|
|
break;
|
|
case LS_MONSTER:
|
|
is_global = !mon_is_local(ls->id.a_monst);
|
|
break;
|
|
default:
|
|
is_global = 0;
|
|
impossible("maybe_write_ls: bad type (%d) [range=%d]", ls->type,
|
|
range);
|
|
break;
|
|
}
|
|
/* if global and not doing local, or vice versa, count it */
|
|
if (is_global ^ (range == RANGE_LEVEL)) {
|
|
count++;
|
|
if (write_it)
|
|
write_ls(nhfp, ls);
|
|
}
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
void
|
|
light_sources_sanity_check(void)
|
|
{
|
|
light_source *ls;
|
|
struct monst *mtmp;
|
|
struct obj *otmp;
|
|
unsigned int auint;
|
|
|
|
for (ls = gl.light_base; ls; ls = ls->next) {
|
|
if (!ls->id.a_monst)
|
|
panic("insane light source: no id!");
|
|
if (ls->type == LS_OBJECT) {
|
|
otmp = ls->id.a_obj;
|
|
auint = otmp->o_id;
|
|
if (find_oid(auint) != otmp)
|
|
panic("insane light source: can't find obj #%u!", auint);
|
|
} else if (ls->type == LS_MONSTER) {
|
|
mtmp = (struct monst *) ls->id.a_monst;
|
|
auint = mtmp->m_id;
|
|
if (find_mid(auint, FM_EVERYWHERE) != mtmp)
|
|
panic("insane light source: can't find mon #%u!", auint);
|
|
} else {
|
|
panic("insane light source: bad ls type %d", ls->type);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Write a light source structure to disk. */
|
|
static void
|
|
write_ls(NHFILE *nhfp, light_source *ls)
|
|
{
|
|
anything arg_save;
|
|
struct obj *otmp;
|
|
struct monst *mtmp;
|
|
|
|
if (ls->type == LS_OBJECT || ls->type == LS_MONSTER) {
|
|
if (ls->flags & LSF_NEEDS_FIXUP) {
|
|
if (nhfp->structlevel)
|
|
bwrite(nhfp->fd, (genericptr_t) ls, sizeof(light_source));
|
|
} else {
|
|
/* replace object pointer with id for write, then put back */
|
|
arg_save = ls->id;
|
|
if (ls->type == LS_OBJECT) {
|
|
otmp = ls->id.a_obj;
|
|
ls->id = cg.zeroany;
|
|
ls->id.a_uint = otmp->o_id;
|
|
if (find_oid((unsigned) ls->id.a_uint) != otmp)
|
|
impossible("write_ls: can't find obj #%u!",
|
|
ls->id.a_uint);
|
|
} else { /* ls->type == LS_MONSTER */
|
|
mtmp = (struct monst *) ls->id.a_monst;
|
|
ls->id = cg.zeroany;
|
|
ls->id.a_uint = mtmp->m_id;
|
|
if (find_mid((unsigned) ls->id.a_uint, FM_EVERYWHERE) != mtmp)
|
|
impossible("write_ls: can't find mon #%u!",
|
|
ls->id.a_uint);
|
|
}
|
|
ls->flags |= LSF_NEEDS_FIXUP;
|
|
if (nhfp->structlevel)
|
|
bwrite(nhfp->fd, (genericptr_t) ls, sizeof(light_source));
|
|
ls->id = arg_save;
|
|
ls->flags &= ~LSF_NEEDS_FIXUP;
|
|
}
|
|
} else {
|
|
impossible("write_ls: bad type (%d)", ls->type);
|
|
}
|
|
}
|
|
|
|
/* Change light source's ID from src to dest. */
|
|
void
|
|
obj_move_light_source(struct obj *src, struct obj *dest)
|
|
{
|
|
light_source *ls;
|
|
|
|
for (ls = gl.light_base; ls; ls = ls->next)
|
|
if (ls->type == LS_OBJECT && ls->id.a_obj == src)
|
|
ls->id.a_obj = dest;
|
|
src->lamplit = 0;
|
|
dest->lamplit = 1;
|
|
}
|
|
|
|
/* return true if there exist any light sources */
|
|
boolean
|
|
any_light_source(void)
|
|
{
|
|
return (boolean) (gl.light_base != (light_source *) 0);
|
|
}
|
|
|
|
/*
|
|
* Snuff an object light source if at (x,y). This currently works
|
|
* only for burning light sources.
|
|
*/
|
|
void
|
|
snuff_light_source(coordxy x, coordxy y)
|
|
{
|
|
light_source *ls;
|
|
struct obj *obj;
|
|
|
|
for (ls = gl.light_base; ls; ls = ls->next)
|
|
/*
|
|
* Is this position check valid??? Can I assume that the positions
|
|
* will always be correct because the objects would have been
|
|
* updated with the last vision update? [Is that recent enough???]
|
|
*/
|
|
if (ls->type == LS_OBJECT && ls->x == x && ls->y == y) {
|
|
obj = ls->id.a_obj;
|
|
if (obj_is_burning(obj)) {
|
|
/* The only way to snuff Sunsword is to unwield it. Darkness
|
|
* scrolls won't affect it. (If we got here because it was
|
|
* dropped or thrown inside a monster, this won't matter
|
|
* anyway because it will go out when dropped.)
|
|
*/
|
|
if (artifact_light(obj))
|
|
continue;
|
|
end_burn(obj, obj->otyp != MAGIC_LAMP);
|
|
/*
|
|
* The current ls element has just been removed (and
|
|
* ls->next is now invalid). Return assuming that there
|
|
* is only one light source attached to each object.
|
|
*/
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Return TRUE if object sheds any light at all. */
|
|
boolean
|
|
obj_sheds_light(struct obj *obj)
|
|
{
|
|
/* so far, only burning objects shed light */
|
|
return obj_is_burning(obj);
|
|
}
|
|
|
|
/* Return TRUE if sheds light AND will be snuffed by end_burn(). */
|
|
boolean
|
|
obj_is_burning(struct obj *obj)
|
|
{
|
|
return (boolean) (obj->lamplit && (ignitable(obj)
|
|
|| artifact_light(obj)));
|
|
}
|
|
|
|
/* copy the light source(s) attached to src, and attach it/them to dest */
|
|
void
|
|
obj_split_light_source(struct obj *src, struct obj *dest)
|
|
{
|
|
light_source *ls, *new_ls;
|
|
|
|
for (ls = gl.light_base; ls; ls = ls->next)
|
|
if (ls->type == LS_OBJECT && ls->id.a_obj == src) {
|
|
/*
|
|
* Insert the new source at beginning of list. This will
|
|
* never interfere us walking down the list - we are already
|
|
* past the insertion point.
|
|
*/
|
|
new_ls = (light_source *) alloc(sizeof(light_source));
|
|
*new_ls = *ls;
|
|
if (Is_candle(src)) {
|
|
/* split candles may emit less light than original group */
|
|
ls->range = candle_light_range(src);
|
|
new_ls->range = candle_light_range(dest);
|
|
gv.vision_full_recalc = 1; /* in case range changed */
|
|
}
|
|
new_ls->id.a_obj = dest;
|
|
new_ls->next = gl.light_base;
|
|
gl.light_base = new_ls;
|
|
dest->lamplit = 1; /* now an active light source */
|
|
}
|
|
}
|
|
|
|
/* light source `src' has been folded into light source `dest';
|
|
used for merging lit candles and adding candle(s) to lit candelabrum */
|
|
void
|
|
obj_merge_light_sources(struct obj *src, struct obj *dest)
|
|
{
|
|
light_source *ls;
|
|
|
|
/* src == dest implies adding to candelabrum */
|
|
if (src != dest)
|
|
end_burn(src, TRUE); /* extinguish candles */
|
|
|
|
for (ls = gl.light_base; ls; ls = ls->next)
|
|
if (ls->type == LS_OBJECT && ls->id.a_obj == dest) {
|
|
ls->range = candle_light_range(dest);
|
|
gv.vision_full_recalc = 1; /* in case range changed */
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* light source `obj' is being made brighter or dimmer */
|
|
void
|
|
obj_adjust_light_radius(struct obj *obj, int new_radius)
|
|
{
|
|
light_source *ls;
|
|
|
|
for (ls = gl.light_base; ls; ls = ls->next)
|
|
if (ls->type == LS_OBJECT && ls->id.a_obj == obj) {
|
|
if (new_radius != ls->range)
|
|
gv.vision_full_recalc = 1;
|
|
ls->range = new_radius;
|
|
return;
|
|
}
|
|
impossible("obj_adjust_light_radius: can't find %s", xname(obj));
|
|
}
|
|
|
|
/* Candlelight is proportional to the number of candles;
|
|
minimum range is 2 rather than 1 for playability. */
|
|
int
|
|
candle_light_range(struct obj *obj)
|
|
{
|
|
int radius;
|
|
|
|
if (obj->otyp == CANDELABRUM_OF_INVOCATION) {
|
|
/*
|
|
* The special candelabrum emits more light than the
|
|
* corresponding number of candles would.
|
|
* 1..3 candles, range 2 (minimum range);
|
|
* 4..6 candles, range 3 (normal lamp range);
|
|
* 7 candles, range 4 (bright).
|
|
*/
|
|
radius = (obj->spe < 4) ? 2 : (obj->spe < 7) ? 3 : 4;
|
|
} else if (Is_candle(obj)) {
|
|
/*
|
|
* Range is incremented quadratically. You can get the same
|
|
* amount of light as from a lamp with 4 candles, and
|
|
* even better light with 9 candles, and so on.
|
|
* 1..3 candles, range 2;
|
|
* 4..8 candles, range 3;
|
|
* 9..15 candles, range 4; &c.
|
|
*/
|
|
long n = obj->quan;
|
|
|
|
radius = 1; /* always incremented at least once */
|
|
while(radius*radius <= n && radius < MAX_RADIUS) {
|
|
radius++;
|
|
}
|
|
} else {
|
|
/* we're only called for lit candelabrum or candles */
|
|
/* impossible("candlelight for %d?", obj->otyp); */
|
|
radius = 3; /* lamp's value */
|
|
}
|
|
return radius;
|
|
}
|
|
|
|
/* light emitting artifact's range depends upon its curse/bless state */
|
|
int
|
|
arti_light_radius(struct obj *obj)
|
|
{
|
|
int res;
|
|
|
|
/*
|
|
* Used by begin_burn() when setting up a new light source
|
|
* (obj->lamplit will already be set by this point) and
|
|
* also by bless()/unbless()/uncurse()/curse() to decide
|
|
* whether to call obj_adjust_light_radius().
|
|
*/
|
|
|
|
/* sanity check [simplifies usage by bless()/curse()/&c] */
|
|
if (!obj->lamplit || !artifact_light(obj))
|
|
return 0;
|
|
|
|
/* cursed radius of 1 is not noticeable for an item that's
|
|
carried by the hero but is if it's carried by a monster
|
|
or left lit on the floor (not applicable for Sunsword) */
|
|
res = (obj->blessed ? 3 : !obj->cursed ? 2 : 1);
|
|
|
|
/* if poly'd into gold dragon with embedded scales, make the scales
|
|
have minimum radiance (hero as light source will use light radius
|
|
based on monster form); otherwise, worn gold DSM gives off more
|
|
light than other light sources */
|
|
if (obj == uskin)
|
|
res = 1;
|
|
else if (obj->otyp == GOLD_DRAGON_SCALE_MAIL) /* DSM but not scales */
|
|
++res;
|
|
|
|
return res;
|
|
}
|
|
|
|
/* adverb describing lit artifact's light; radius varies depending upon
|
|
curse/bless state; also used for gold dragon scales/scale mail */
|
|
const char *
|
|
arti_light_description(struct obj *obj)
|
|
{
|
|
switch (arti_light_radius(obj)) {
|
|
case 4:
|
|
return "radiantly"; /* blessed gold dragon scale mail */
|
|
case 3:
|
|
return "brilliantly"; /* blessed artifact, uncursed gold DSM */
|
|
case 2:
|
|
return "brightly"; /* uncursed artifact, cursed gold DSM */
|
|
case 1:
|
|
return "dimly"; /* cursed artifact, embedded scales */
|
|
default:
|
|
break;
|
|
}
|
|
return "strangely";
|
|
}
|
|
|
|
/* the #lightsources command */
|
|
int
|
|
wiz_light_sources(void)
|
|
{
|
|
winid win;
|
|
char buf[BUFSZ];
|
|
light_source *ls;
|
|
|
|
win = create_nhwindow(NHW_MENU); /* corner text window */
|
|
if (win == WIN_ERR)
|
|
return ECMD_OK;
|
|
|
|
Sprintf(buf, "Mobile light sources: hero @ (%2d,%2d)", u.ux, u.uy);
|
|
putstr(win, 0, buf);
|
|
putstr(win, 0, "");
|
|
|
|
if (gl.light_base) {
|
|
putstr(win, 0, "location range flags type id");
|
|
putstr(win, 0, "-------- ----- ------ ---- -------");
|
|
for (ls = gl.light_base; ls; ls = ls->next) {
|
|
Sprintf(buf, " %2d,%2d %2d 0x%04x %s %s", ls->x, ls->y,
|
|
ls->range, ls->flags,
|
|
(ls->type == LS_OBJECT
|
|
? "obj"
|
|
: ls->type == LS_MONSTER
|
|
? (mon_is_local(ls->id.a_monst)
|
|
? "mon"
|
|
: (ls->id.a_monst == &gy.youmonst)
|
|
? "you"
|
|
/* migrating monster */
|
|
: "<m>")
|
|
: "???"),
|
|
fmt_ptr(ls->id.a_void));
|
|
putstr(win, 0, buf);
|
|
}
|
|
} else
|
|
putstr(win, 0, "<none>");
|
|
|
|
display_nhwindow(win, FALSE);
|
|
destroy_nhwindow(win);
|
|
|
|
return ECMD_OK;
|
|
}
|
|
|
|
/* for 'onefile' processing where end of this file isn't necessarily the
|
|
end of the source code seen by the compiler */
|
|
#undef LSF_SHOW
|
|
#undef LSF_NEEDS_FIXUP
|
|
#undef mon_is_local
|
|
|
|
/*light.c*/
|