1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-18 13:12:50 +00:00
2009-12-19 21:31:22 +00:00

593 lines
14 KiB
D

/*
OpenMW - The completely unofficial reimplementation of Morrowind
Copyright (C) 2008 Nicolay Korslund
Email: < korslund@gmail.com >
WWW: http://openmw.snaptoad.com/
This file (celldata.d) is part of the OpenMW package.
OpenMW is distributed as free software: you can redistribute it
and/or modify it under the terms of the GNU General Public License
version 3, as published by the Free Software Foundation.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
version 3 along with this program. If not, see
http://www.gnu.org/licenses/ .
*/
module scene.celldata;
import std.stdio;
public import esm.esmmain;
import core.memory;
import monster.monster;
import util.reglist;
import ogre.ogre;
import ogre.bindings;
import sound.audio;
import scene.player;
// Generic version of a "live" object. Do we even need this at all?
// No, I don't think so.
struct GenLive(T)
{
// Instance of class GameObject or a derived class (depending on
// object type)
MonsterObject *obj;
T *m;
Placement *getPos()
{
assert(obj !is null);
// This belongs in HACK-land
return cast(Placement*)obj.getFloatPtr("x");
}
float getScale()
{
assert(obj !is null);
return obj.getFloat("scale");
}
}
alias GenLive!(Static) LiveStatic;
alias GenLive!(NPC) LiveNPC;
alias GenLive!(Activator) LiveActivator;
alias GenLive!(Potion) LivePotion;
alias GenLive!(Apparatus) LiveApparatus;
alias GenLive!(Ingredient) LiveIngredient;
alias GenLive!(Armor) LiveArmor;
alias GenLive!(Weapon) LiveWeapon;
alias GenLive!(Book) LiveBook;
alias GenLive!(Clothing) LiveClothing;
alias GenLive!(Tool) LiveTool;
alias GenLive!(Creature) LiveCreature;
alias GenLive!(Door) LiveDoor;
alias GenLive!(Misc) LiveMisc;
alias GenLive!(Container) LiveContainer;
// TODO: This is sort of redundant. Eliminate or rework it later.
struct LiveLight
{
MonsterObject *obj;
Light *m;
Placement *getPos()
{
assert(obj !is null);
// This belongs in HACK-land
return cast(Placement*)obj.getFloatPtr("x");
}
float getScale()
{
assert(obj !is null);
return obj.getFloat("scale");
}
NodePtr lightNode;
SoundInstance *loopSound;
}
class CellData
{
private:
RegionManager reg;
public:
InteriorCell *inCell;
ExteriorCell *exCell;
// Ambient light data
AMBIStruct ambi;
// Water height
int water;
// Linked lists that hold references to all the objects in the cell
RegionList!(LiveStatic) statics;
RegionList!(LiveMisc) miscItems;
RegionList!(LiveLight) lights;
RegionList!(LiveLight) statLights;
RegionList!(LiveNPC) npcs;
RegionList!(LiveContainer) containers;
RegionList!(LiveDoor) doors;
RegionList!(LiveActivator) activators;
RegionList!(LivePotion) potions;
RegionList!(LiveApparatus) appas;
RegionList!(LiveIngredient) ingredients;
RegionList!(LiveArmor) armors;
RegionList!(LiveWeapon) weapons;
RegionList!(LiveBook) books;
RegionList!(LiveTool) tools;
RegionList!(LiveClothing) clothes;
RegionList!(LiveCreature) creatures;
this(RegionManager r)
{
reg = r;
killCell(); // Make sure all data is initialized.
}
// Kills all data and initialize the object for reuse.
void killCell()
{
inCell = null;
exCell = null;
// Reset the lists
statics.init(reg);
miscItems.init(reg);
lights.init(reg);
statLights.init(reg);
npcs.init(reg);
containers.init(reg);
doors.init(reg);
activators.init(reg);
potions.init(reg);
appas.init(reg);
ingredients.init(reg);
armors.init(reg);
weapons.init(reg);
books.init(reg);
tools.init(reg);
clothes.init(reg);
creatures.init(reg);
// Write some statistics
//writefln(reg);
reg.freeAll();
}
// Load an exterior cell
void loadExtCell(int x, int y)
{
exCell = cells.getExt(x, y);
writefln("Name: %s", exCell.name);
writefln("Region: %s", exCell.region.id);
esFile.restoreContext(exCell.context, reg);
// Always for exterior cells
water = 0;
Color mapColor;
if(esFile.isNextSub("NAM5"))
esFile.readHExact(&mapColor, mapColor.sizeof);
loadReferences();
const float cellWidth = 8192;
// TODO/FIXME: This is temporary
playerData.position.position[0] = x*cellWidth;
playerData.position.position[1] = y*cellWidth;
playerData.position.position[2] = 6000;
playerData.position.rotation[] = 0;
}
// Load an interior cell
void loadIntCell(char[] cName)
{
inCell = cells.getInt(cName);
/*
writefln("Cell id: '%s'", cName);
writefln("Cell name: '%s'", cell.id);
*/
esFile.restoreContext(inCell.context, reg);
// TODO: Read this in loadcell.d
if(esFile.isNextSub("INTV") || esFile.isNextSub("WHGT"))
water = esFile.getHInt();
//writefln("Water height: ", water);
if(inCell.flags & CellFlags.QuasiExt)
{
Region* reg = esFile.getHNOPtr!(Region)("RGNN", regions);
if(reg) writefln("Cell has region %s", reg.id);
else writefln("No region");
// Determine weather from this region
}
else
{
// Only for interior cells
esFile.readHNExact(&ambi, ambi.sizeof, "AMBI");
}
/*
writefln("Ambient light: ", ambi.ambient.array);
writefln("Sunlight: ", ambi.sunlight.array);
writefln("Fog color: ", ambi.ambient.array);
writefln("Fog density: ", ambi.fogDensity);
*/
loadReferences();
}
private:
void loadReferences()
{
with(esFile)
{
// Now read all the references
while(hasMoreSubs)
{
// Number of references in the cell? Maximum once in each
// cell, but not always at the beginning, and not always
// right. In other words, completely useless. Strange
// people...
getHNOInt("NAM0", 0);
int refnum = getHNInt("FRMR"); // Reference number
char[] refr = getHNString("NAME"); // ID of object
// Used internally for optimizing
bool container;
bool door;
bool stat;
bool activator;
// Identify the referenced object by looking up the id
// string 'ref' in the global database of all
// cell-referencable objects.
Item itm = cellRefs.lookup(refr);
ItemT it = ItemT(itm);
MonsterObject *mo;
// These should be ordered according to how commonly they
// occur.
// Static mesh - probably the most common object type
if(Static *s = it.getStatic())
{
LiveStatic ls;
ls.m = s;
ls.obj = s.proto.clone();
mo = ls.obj;
statics.insert(ls);
stat = true;
}
// Misc items are also pretty common
else if(Misc *m = it.getMisc())
{
LiveMisc ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
miscItems.insert(ls);
}
// Lights and containers too
else if(Light *m = it.getLight())
{
LiveLight ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
bool carry = (m.data.flags&Light.Flags.Carry) != 0;
if(carry)
lights.insert(ls);
else
statLights.insert(ls);
}
else if(Container *c = it.getContainer())
{
LiveContainer ls;
ls.m = c;
ls.obj = c.proto.clone();
mo = ls.obj;
containers.insert(ls);
container = true;
}
// From here on I doubt the order will matter much
else if(Door *d = it.getDoor())
{
LiveDoor ls;
ls.m = d;
ls.obj = d.proto.clone();
mo = ls.obj;
doors.insert(ls);
door = true;
}
// Activator?
else if(Activator *a = it.getActivator())
{
LiveActivator ls;
ls.m = a;
ls.obj = a.proto.clone();
mo = ls.obj;
activators.insert(ls);
activator = true;
}
// NPC?
else if(NPC *n = it.getNPC())
{
LiveNPC ls;
ls.m = n;
ls.obj = n.proto.clone();
mo = ls.obj;
npcs.insert(ls);
}
else if(Potion *p = it.getPotion())
{
LivePotion ls;
ls.m = p;
ls.obj = p.proto.clone();
mo = ls.obj;
potions.insert(ls);
}
else if(Apparatus *m = it.getApparatus())
{
LiveApparatus ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
appas.insert(ls);
}
else if(Ingredient *m = it.getIngredient())
{
LiveIngredient ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
ingredients.insert(ls);
}
else if(Armor *m = it.getArmor())
{
LiveArmor ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
armors.insert(ls);
}
else if(Weapon *m = it.getWeapon())
{
LiveWeapon ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
weapons.insert(ls);
}
else if(Book *m = it.getBook())
{
LiveBook ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
books.insert(ls);
}
else if(Clothing *m = it.getClothing())
{
LiveClothing ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
clothes.insert(ls);
}
else if(Tool *m = it.getPick())
{
LiveTool ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
tools.insert(ls);
}
else if(Tool *m = it.getProbe())
{
LiveTool ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
tools.insert(ls);
}
else if(Tool *m = it.getRepair())
{
LiveTool ls;
ls.m = m;
ls.obj = m.proto.clone();
mo = ls.obj;
tools.insert(ls);
}
else if(Creature *c = it.getCreature())
{
LiveCreature ls;
ls.m = c;
ls.obj = c.proto.clone();
mo = ls.obj;
creatures.insert(ls);
}
else if(LeveledCreatures *l = it.getCreatureList)
{
// Create a creature, based on current player level.
LiveCreature ls;
ls.m = l.instCreature(*playerData.level);
if(ls.m != null)
{
// Note that this clones a creature object, not a
// leveled list object.
ls.obj = ls.m.proto.clone(); mo = ls.obj;
creatures.insert(ls);
}
}
else fail(format(" UNKNOWN REFERENCE! Type ", cast(int)it.i.type));
// Now that the object has found it's place, load data
// into the object.
with(*mo)
{
// Scale. Multiply with the existing scale value.
float *scale = getFloatPtr("scale");
*scale *= getHNOFloat("XSCL", 1.0);
// Statics only need the position data. Skip the rest
// as an optimization.
if(stat) goto readpos;
// The NPC that owns this object (and will get angry
// if you steal it)
setString8("owner", getHNOString("ANAM"));
// I have no idea, link to a global variable perhaps?
setString8("glob", getHNOString("BNAM"));
// ID of creature trapped in a soul gem (?)
setString8("soulID", getHNOString("XSOL"));
// ?? CNAM has a faction name, might be for
// objects/beds etc belonging to a faction.
{
char[] cnam = getHNOString("CNAM");
setString8("cnam", cnam);
// INDX might be PC faction rank required to use the
// item? Sometimes is -1.
if(cnam.length) setInt("indx", getHNInt("INDX"));
}
// Possibly weapon health, number of uses left or
// weapon magic charge?
setFloat("xchg", getHNOFloat("XCHG", 0.0));
// I have no idea, these are present some times, often
// along with owner (ANAM) and sometimes otherwise. Is
// NAM9 is always 1? INTV is usually one, but big for
// lights. Perhaps something to do with remaining
// light "charge". I haven't tried reading it as a
// float in those cases.
setInt("intv", getHNOInt("INTV", 0));
setInt("nam9", getHNOInt("NAM9", 0));
// Present for doors that teleport you to another
// cell.
if(door && isNextSub("DODT"))
{
setBool("teleport", true);
// Warning, HACK! Will be fixed when we implement
// structs in Monster.
Placement *p = cast(Placement*)getFloatPtr("destx");
readHExact(p, Placement.sizeof);
// Destination cell (optional?)
setString8("destCell", getHNOString("DNAM"));
}
if(door || container)
{
// Lock level (I think)
setInt("lockLevel", getHNOInt("FLTV", 0));
// For locked doors and containers
setString8("key", getHNOString("KNAM"));
setString8("trap", getHNOString("TNAM"));
}
if(activator)
{
// Occurs ONCE in Morrowind.esm, for an activator.
setInt("unam", getHNOByte("UNAM", 0));
// Occurs in Tribunal.esm, eg. in the cell
// "Mournhold, Plaza Brindisi Dorom", where it has
// the value 100.
setInt("fltv", getHNOInt("FLTV", 0));
}
readpos:
// Position and rotation of this object within the
// cell
// TODO: This is a HACK. It assumes the class variable
// floats are placed consecutively in memory in the
// right order. This is true, but is a very bug prone
// method of doing this. Will fix when Monster gets
// structs. (See the DODT record above also.)
Placement *pos = cast(Placement*)getFloatPtr("x");
readHNExact(pos, Placement.sizeof, "DATA");
// TODO/FIXME: Very temporary. Set player position at
// the first door we find.
if(door && !playerData.posSet)
{
playerData.posSet = true;
*playerData.position = *pos;
}
}
} // End of while(hasMoreSubs)
}
} // End of loadReferences()
} // End of class CellData
CellFreelist cellList;
// Structure used as a free list for cell data objects and their
// respective regions.
struct CellFreelist
{
// We love static arrays! And 100 cells should be enough for
// everybody :)
CellData[100] list;
uint next;
// TODO: Figure out a good size to use here as well.
CellData get()
{
if(next) return list[--next];
// Since these are reused, there's no waste in allocating on the
// heap here. Also, this is only semi-runtime (executed when
// loading a cell), thus an occational GC slow down is not
// critical.
return new CellData(new RegionManager("CELL"));
}
void release(CellData r)
{
assert(next < list.length);
r.killCell();
list[next++] = r;
}
}