mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 04:26:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			260 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			260 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|   OpenMW - The completely unofficial reimplementation of Morrowind
 | |
|   Copyright (C) 2008-2010  Nicolay Korslund
 | |
|   Email: < korslund@gmail.com >
 | |
|   WWW: http://openmw.sourceforge.net/
 | |
| 
 | |
|   This file (node.h) 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/ .
 | |
| 
 | |
|  */
 | |
| 
 | |
| #ifndef OPENMW_COMPONENTS_NIF_NODE_HPP
 | |
| #define OPENMW_COMPONENTS_NIF_NODE_HPP
 | |
| 
 | |
| #include <OgreMatrix4.h>
 | |
| 
 | |
| #include "controlled.hpp"
 | |
| #include "data.hpp"
 | |
| #include "property.hpp"
 | |
| 
 | |
| namespace Nif
 | |
| {
 | |
| 
 | |
| class NiNode;
 | |
| 
 | |
| /** A Node is an object that's part of the main NIF tree. It has
 | |
|     parent node (unless it's the root), and transformation (location
 | |
|     and rotation) relative to it's parent.
 | |
|  */
 | |
| class Node : public Named
 | |
| {
 | |
| public:
 | |
|     // Node flags. Interpretation depends somewhat on the type of node.
 | |
|     int flags;
 | |
|     Transformation trafo;
 | |
|     Ogre::Vector3 velocity; // Unused? Might be a run-time game state
 | |
|     PropertyList props;
 | |
| 
 | |
|     // Bounding box info
 | |
|     bool hasBounds;
 | |
|     Ogre::Vector3 boundPos;
 | |
|     Ogre::Matrix3 boundRot;
 | |
|     Ogre::Vector3 boundXYZ; // Box size
 | |
| 
 | |
|     void read(NIFStream *nif)
 | |
|     {
 | |
|         Named::read(nif);
 | |
| 
 | |
|         flags = nif->getUShort();
 | |
|         trafo = nif->getTrafo();
 | |
|         velocity = nif->getVector3();
 | |
|         props.read(nif);
 | |
| 
 | |
|         hasBounds = !!nif->getInt();
 | |
|         if(hasBounds)
 | |
|         {
 | |
|             nif->getInt(); // always 1
 | |
|             boundPos = nif->getVector3();
 | |
|             boundRot = nif->getMatrix3();
 | |
|             boundXYZ = nif->getVector3();
 | |
|         }
 | |
| 
 | |
|         parent = NULL;
 | |
| 
 | |
|         boneTrafo = NULL;
 | |
|         boneIndex = -1;
 | |
|     }
 | |
| 
 | |
|     void post(NIFFile *nif)
 | |
|     {
 | |
|         Named::post(nif);
 | |
|         props.post(nif);
 | |
|     }
 | |
| 
 | |
|     // Parent node, or NULL for the root node. As far as I'm aware, only
 | |
|     // NiNodes (or types derived from NiNodes) can be parents.
 | |
|     NiNode *parent;
 | |
| 
 | |
|     // Bone transformation. If set, node is a part of a skeleton.
 | |
|     const NiSkinData::BoneTrafo *boneTrafo;
 | |
| 
 | |
|     // Bone weight info, from NiSkinData
 | |
|     const NiSkinData::BoneInfo *boneInfo;
 | |
| 
 | |
|     // Bone index. If -1, this node is either not a bone, or if
 | |
|     // boneTrafo is set it is the root bone in the skeleton.
 | |
|     short boneIndex;
 | |
| 
 | |
|     void makeRootBone(const NiSkinData::BoneTrafo *tr)
 | |
|     {
 | |
|         boneTrafo = tr;
 | |
|         boneIndex = -1;
 | |
|     }
 | |
| 
 | |
|     void makeBone(short ind, const NiSkinData::BoneInfo &bi)
 | |
|     {
 | |
|         boneInfo = &bi;
 | |
|         boneTrafo = &bi.trafo;
 | |
|         boneIndex = ind;
 | |
|     }
 | |
| 
 | |
|     Ogre::Matrix4 getLocalTransform() const;
 | |
|     Ogre::Matrix4 getWorldTransform() const;
 | |
| };
 | |
| 
 | |
| struct NiNode : Node
 | |
| {
 | |
|     NodeList children;
 | |
|     NodeList effects;
 | |
| 
 | |
|     /* Known NiNode flags:
 | |
|         0x01 hidden
 | |
|         0x02 use mesh for collision
 | |
|         0x04 use bounding box for collision (?)
 | |
|         0x08 unknown, but common
 | |
|         0x20, 0x40, 0x80 unknown
 | |
|     */
 | |
| 
 | |
|     void read(NIFStream *nif)
 | |
|     {
 | |
|         Node::read(nif);
 | |
|         children.read(nif);
 | |
|         effects.read(nif);
 | |
|     }
 | |
| 
 | |
|     void post(NIFFile *nif)
 | |
|     {
 | |
|         Node::post(nif);
 | |
|         children.post(nif);
 | |
|         effects.post(nif);
 | |
| 
 | |
|         for(size_t i = 0;i < children.length();i++)
 | |
|         {
 | |
|             // Why would a unique list of children contain empty refs?
 | |
|             if(!children[i].empty())
 | |
|                 children[i]->parent = this;
 | |
|         }
 | |
|     }
 | |
| };
 | |
| 
 | |
| struct NiTriShape : Node
 | |
| {
 | |
|     /* Possible flags:
 | |
|         0x40 - mesh has no vertex normals ?
 | |
| 
 | |
|         Only flags included in 0x47 (ie. 0x01, 0x02, 0x04 and 0x40) have
 | |
|         been observed so far.
 | |
|     */
 | |
| 
 | |
|     NiTriShapeDataPtr data;
 | |
|     NiSkinInstancePtr skin;
 | |
| 
 | |
|     void read(NIFStream *nif)
 | |
|     {
 | |
|         Node::read(nif);
 | |
|         data.read(nif);
 | |
|         skin.read(nif);
 | |
|     }
 | |
| 
 | |
|     void post(NIFFile *nif)
 | |
|     {
 | |
|         Node::post(nif);
 | |
|         data.post(nif);
 | |
|         skin.post(nif);
 | |
|     }
 | |
| };
 | |
| 
 | |
| struct NiCamera : Node
 | |
| {
 | |
|     struct Camera
 | |
|     {
 | |
|         // Camera frustrum
 | |
|         float left, right, top, bottom, nearDist, farDist;
 | |
| 
 | |
|         // Viewport
 | |
|         float vleft, vright, vtop, vbottom;
 | |
| 
 | |
|         // Level of detail modifier
 | |
|         float LOD;
 | |
| 
 | |
|         void read(NIFStream *nif)
 | |
|         {
 | |
|             left = nif->getFloat();
 | |
|             right = nif->getFloat();
 | |
|             top = nif->getFloat();
 | |
|             bottom = nif->getFloat();
 | |
|             nearDist = nif->getFloat();
 | |
|             farDist = nif->getFloat();
 | |
| 
 | |
|             vleft = nif->getFloat();
 | |
|             vright = nif->getFloat();
 | |
|             vtop = nif->getFloat();
 | |
|             vbottom = nif->getFloat();
 | |
| 
 | |
|             LOD = nif->getFloat();
 | |
|         }
 | |
|     };
 | |
|     Camera cam;
 | |
| 
 | |
|     void read(NIFStream *nif)
 | |
|     {
 | |
|         Node::read(nif);
 | |
| 
 | |
|         cam.read(nif);
 | |
| 
 | |
|         nif->getInt(); // -1
 | |
|         nif->getInt(); // 0
 | |
|     }
 | |
| };
 | |
| 
 | |
| struct NiAutoNormalParticles : Node
 | |
| {
 | |
|     NiAutoNormalParticlesDataPtr data;
 | |
| 
 | |
|     void read(NIFStream *nif)
 | |
|     {
 | |
|         Node::read(nif);
 | |
|         data.read(nif);
 | |
|         nif->getInt(); // -1
 | |
|     }
 | |
| 
 | |
|     void post(NIFFile *nif)
 | |
|     {
 | |
|         Node::post(nif);
 | |
|         data.post(nif);
 | |
|     }
 | |
| };
 | |
| 
 | |
| struct NiRotatingParticles : Node
 | |
| {
 | |
|     NiRotatingParticlesDataPtr data;
 | |
| 
 | |
|     void read(NIFStream *nif)
 | |
|     {
 | |
|         Node::read(nif);
 | |
|         data.read(nif);
 | |
|         nif->getInt(); // -1
 | |
|     }
 | |
| 
 | |
|     void post(NIFFile *nif)
 | |
|     {
 | |
|         Node::post(nif);
 | |
|         data.post(nif);
 | |
|     }
 | |
| };
 | |
| 
 | |
| } // Namespace
 | |
| #endif
 |