mirror of
https://github.com/TES3MP/openmw-tes3mp.git
synced 2025-01-22 00:53:50 +00:00
e197f5318b
conversion from 'const float' to 'int', possible loss of data conversion from 'double' to 'int', possible loss of data conversion from 'float' to 'int', possible loss of data
347 lines
11 KiB
C++
347 lines
11 KiB
C++
#include "characterpreview.hpp"
|
|
|
|
#include <OgreSceneManager.h>
|
|
#include <OgreRoot.h>
|
|
#include <OgreHardwarePixelBuffer.h>
|
|
#include <OgreCamera.h>
|
|
#include <OgreSceneNode.h>
|
|
#include <OgreTextureManager.h>
|
|
#include <OgreViewport.h>
|
|
#include <OgreRenderTexture.h>
|
|
|
|
#include <libs/openengine/ogre/selectionbuffer.hpp>
|
|
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
#include "../mwbase/world.hpp"
|
|
#include "../mwworld/class.hpp"
|
|
#include "../mwworld/inventorystore.hpp"
|
|
|
|
#include "renderconst.hpp"
|
|
#include "npcanimation.hpp"
|
|
|
|
namespace MWRender
|
|
{
|
|
|
|
CharacterPreview::CharacterPreview(MWWorld::Ptr character, int sizeX, int sizeY, const std::string& name,
|
|
Ogre::Vector3 position, Ogre::Vector3 lookAt)
|
|
: mSceneMgr (0)
|
|
, mPosition(position)
|
|
, mLookAt(lookAt)
|
|
, mCharacter(character)
|
|
, mAnimation(NULL)
|
|
, mName(name)
|
|
, mSizeX(sizeX)
|
|
, mSizeY(sizeY)
|
|
, mRenderTarget(NULL)
|
|
, mViewport(NULL)
|
|
, mCamera(NULL)
|
|
, mNode(NULL)
|
|
, mRecover(false)
|
|
{
|
|
mCharacter.mCell = NULL;
|
|
}
|
|
|
|
void CharacterPreview::onSetup()
|
|
{
|
|
|
|
}
|
|
|
|
void CharacterPreview::onFrame()
|
|
{
|
|
if (mRecover)
|
|
{
|
|
setupRenderTarget();
|
|
mRenderTarget->update();
|
|
mRecover = false;
|
|
}
|
|
}
|
|
|
|
void CharacterPreview::setup ()
|
|
{
|
|
mSceneMgr = Ogre::Root::getSingleton().createSceneManager(Ogre::ST_GENERIC);
|
|
|
|
// This is a dummy light to turn off shadows without having to use a separate set of shaders
|
|
Ogre::Light* l = mSceneMgr->createLight();
|
|
l->setType (Ogre::Light::LT_DIRECTIONAL);
|
|
l->setDiffuseColour (Ogre::ColourValue(0,0,0));
|
|
|
|
/// \todo Read the fallback values from INIImporter (Inventory:Directional*)
|
|
l = mSceneMgr->createLight();
|
|
l->setType (Ogre::Light::LT_DIRECTIONAL);
|
|
l->setDirection (Ogre::Vector3(0.3f, -0.7f, 0.3f));
|
|
l->setDiffuseColour (Ogre::ColourValue(1,1,1));
|
|
|
|
mSceneMgr->setAmbientLight (Ogre::ColourValue(0.25, 0.25, 0.25));
|
|
|
|
mCamera = mSceneMgr->createCamera (mName);
|
|
mCamera->setFOVy(Ogre::Degree(12.3f));
|
|
mCamera->setAspectRatio (float(mSizeX) / float(mSizeY));
|
|
|
|
Ogre::SceneNode* renderRoot = mSceneMgr->getRootSceneNode()->createChildSceneNode("renderRoot");
|
|
|
|
// leftover of old coordinate system. TODO: remove this and adjust positions/orientations to match
|
|
renderRoot->pitch(Ogre::Degree(-90));
|
|
|
|
mNode = renderRoot->createChildSceneNode();
|
|
|
|
mAnimation = new NpcAnimation(mCharacter, mNode,
|
|
0, true, true, (renderHeadOnly() ? NpcAnimation::VM_HeadOnly : NpcAnimation::VM_Normal));
|
|
|
|
Ogre::Vector3 scale = mNode->getScale();
|
|
mCamera->setPosition(mPosition * scale);
|
|
mCamera->lookAt(mLookAt * scale);
|
|
|
|
mCamera->setNearClipDistance (1);
|
|
mCamera->setFarClipDistance (1000);
|
|
|
|
mTexture = Ogre::TextureManager::getSingleton().createManual(mName,
|
|
Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, Ogre::TEX_TYPE_2D, mSizeX, mSizeY, 0, Ogre::PF_A8R8G8B8, Ogre::TU_RENDERTARGET, this);
|
|
|
|
setupRenderTarget();
|
|
|
|
onSetup ();
|
|
}
|
|
|
|
CharacterPreview::~CharacterPreview ()
|
|
{
|
|
if (mSceneMgr)
|
|
{
|
|
mSceneMgr->destroyAllCameras();
|
|
delete mAnimation;
|
|
Ogre::Root::getSingleton().destroySceneManager(mSceneMgr);
|
|
Ogre::TextureManager::getSingleton().remove(mName);
|
|
}
|
|
}
|
|
|
|
void CharacterPreview::rebuild()
|
|
{
|
|
delete mAnimation;
|
|
mAnimation = NULL;
|
|
mAnimation = new NpcAnimation(mCharacter, mNode,
|
|
0, true, true, (renderHeadOnly() ? NpcAnimation::VM_HeadOnly : NpcAnimation::VM_Normal));
|
|
|
|
float scale=1.f;
|
|
mCharacter.getClass().adjustScale(mCharacter, scale);
|
|
mNode->setScale(Ogre::Vector3(scale));
|
|
|
|
mCamera->setPosition(mPosition * mNode->getScale());
|
|
mCamera->lookAt(mLookAt * mNode->getScale());
|
|
|
|
onSetup();
|
|
}
|
|
|
|
void CharacterPreview::loadResource(Ogre::Resource *resource)
|
|
{
|
|
Ogre::Texture* tex = dynamic_cast<Ogre::Texture*>(resource);
|
|
if (!tex)
|
|
return;
|
|
|
|
tex->createInternalResources();
|
|
|
|
mRenderTarget = NULL;
|
|
mViewport = NULL;
|
|
mRecover = true;
|
|
}
|
|
|
|
void CharacterPreview::setupRenderTarget()
|
|
{
|
|
mRenderTarget = mTexture->getBuffer()->getRenderTarget();
|
|
mRenderTarget->removeAllViewports ();
|
|
mViewport = mRenderTarget->addViewport(mCamera);
|
|
mViewport->setOverlaysEnabled(false);
|
|
mViewport->setBackgroundColour(Ogre::ColourValue(0, 0, 0, 0));
|
|
mViewport->setShadowsEnabled(false);
|
|
mRenderTarget->setActive(true);
|
|
mRenderTarget->setAutoUpdated (false);
|
|
}
|
|
|
|
// --------------------------------------------------------------------------------------------------
|
|
|
|
|
|
InventoryPreview::InventoryPreview(MWWorld::Ptr character)
|
|
: CharacterPreview(character, 512, 1024, "CharacterPreview", Ogre::Vector3(0, 71, -700), Ogre::Vector3(0,71,0))
|
|
, mSelectionBuffer(NULL)
|
|
, mSizeX(0)
|
|
, mSizeY(0)
|
|
{
|
|
}
|
|
|
|
InventoryPreview::~InventoryPreview()
|
|
{
|
|
delete mSelectionBuffer;
|
|
}
|
|
|
|
void InventoryPreview::resize(int sizeX, int sizeY)
|
|
{
|
|
mSizeX = sizeX;
|
|
mSizeY = sizeY;
|
|
|
|
mViewport->setDimensions (0, 0, std::min(1.f, float(mSizeX) / float(512)), std::min(1.f, float(mSizeY) / float(1024)));
|
|
mTexture->load();
|
|
|
|
if (!mRenderTarget)
|
|
setupRenderTarget();
|
|
|
|
mRenderTarget->update();
|
|
}
|
|
|
|
void InventoryPreview::update()
|
|
{
|
|
if (!mAnimation)
|
|
return;
|
|
|
|
mAnimation->updateParts();
|
|
|
|
MWWorld::InventoryStore &inv = mCharacter.getClass().getInventoryStore(mCharacter);
|
|
MWWorld::ContainerStoreIterator iter = inv.getSlot(MWWorld::InventoryStore::Slot_CarriedRight);
|
|
std::string groupname;
|
|
bool showCarriedLeft = true;
|
|
if(iter == inv.end())
|
|
groupname = "inventoryhandtohand";
|
|
else
|
|
{
|
|
const std::string &type = iter->getTypeName();
|
|
if(type == typeid(ESM::Lockpick).name() || type == typeid(ESM::Probe).name())
|
|
groupname = "inventoryweapononehand";
|
|
else if(type == typeid(ESM::Weapon).name())
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref = iter->get<ESM::Weapon>();
|
|
|
|
int type = ref->mBase->mData.mType;
|
|
if(type == ESM::Weapon::ShortBladeOneHand ||
|
|
type == ESM::Weapon::LongBladeOneHand ||
|
|
type == ESM::Weapon::BluntOneHand ||
|
|
type == ESM::Weapon::AxeOneHand ||
|
|
type == ESM::Weapon::MarksmanThrown ||
|
|
type == ESM::Weapon::MarksmanCrossbow ||
|
|
type == ESM::Weapon::MarksmanBow)
|
|
groupname = "inventoryweapononehand";
|
|
else if(type == ESM::Weapon::LongBladeTwoHand ||
|
|
type == ESM::Weapon::BluntTwoClose ||
|
|
type == ESM::Weapon::AxeTwoHand)
|
|
groupname = "inventoryweapontwohand";
|
|
else if(type == ESM::Weapon::BluntTwoWide ||
|
|
type == ESM::Weapon::SpearTwoWide)
|
|
groupname = "inventoryweapontwowide";
|
|
else
|
|
groupname = "inventoryhandtohand";
|
|
|
|
showCarriedLeft = (iter->getClass().canBeEquipped(*iter, mCharacter).first != 2);
|
|
}
|
|
else
|
|
groupname = "inventoryhandtohand";
|
|
}
|
|
|
|
mAnimation->showCarriedLeft(showCarriedLeft);
|
|
|
|
mCurrentAnimGroup = groupname;
|
|
mAnimation->play(mCurrentAnimGroup, 1, Animation::Group_All, false, 1.0f, "start", "stop", 0.0f, 0);
|
|
|
|
MWWorld::ContainerStoreIterator torch = inv.getSlot(MWWorld::InventoryStore::Slot_CarriedLeft);
|
|
if(torch != inv.end() && torch->getTypeName() == typeid(ESM::Light).name() && showCarriedLeft)
|
|
{
|
|
if(!mAnimation->getInfo("torch"))
|
|
mAnimation->play("torch", 2, MWRender::Animation::Group_LeftArm, false,
|
|
1.0f, "start", "stop", 0.0f, ~0ul, true);
|
|
}
|
|
else if(mAnimation->getInfo("torch"))
|
|
mAnimation->disable("torch");
|
|
|
|
mAnimation->runAnimation(0.0f);
|
|
|
|
mNode->setOrientation (Ogre::Quaternion::IDENTITY);
|
|
|
|
mViewport->setDimensions (0, 0, std::min(1.f, float(mSizeX) / float(512)), std::min(1.f, float(mSizeY) / float(1024)));
|
|
mTexture->load();
|
|
|
|
if (!mRenderTarget)
|
|
setupRenderTarget();
|
|
|
|
mRenderTarget->update();
|
|
|
|
mSelectionBuffer->update();
|
|
}
|
|
|
|
void InventoryPreview::setupRenderTarget()
|
|
{
|
|
CharacterPreview::setupRenderTarget();
|
|
mViewport->setDimensions (0, 0, std::min(1.f, float(mSizeX) / float(512)), std::min(1.f, float(mSizeY) / float(1024)));
|
|
}
|
|
|
|
int InventoryPreview::getSlotSelected (int posX, int posY)
|
|
{
|
|
return mSelectionBuffer->getSelected (posX, posY);
|
|
}
|
|
|
|
void InventoryPreview::onSetup ()
|
|
{
|
|
delete mSelectionBuffer;
|
|
mSelectionBuffer = new OEngine::Render::SelectionBuffer(mCamera, 512, 1024, 0);
|
|
|
|
mAnimation->showWeapons(true);
|
|
|
|
mCurrentAnimGroup = "inventoryhandtohand";
|
|
mAnimation->play(mCurrentAnimGroup, 1, Animation::Group_All, false, 1.0f, "start", "stop", 0.0f, 0);
|
|
}
|
|
|
|
// --------------------------------------------------------------------------------------------------
|
|
|
|
RaceSelectionPreview::RaceSelectionPreview()
|
|
: CharacterPreview(MWBase::Environment::get().getWorld()->getPlayerPtr(),
|
|
512, 512, "CharacterHeadPreview", Ogre::Vector3(0, 8, -125), Ogre::Vector3(0,127,0))
|
|
, mBase (*mCharacter.get<ESM::NPC>()->mBase)
|
|
, mRef(&mBase)
|
|
, mPitch(Ogre::Degree(6))
|
|
{
|
|
mCharacter = MWWorld::Ptr(&mRef, NULL);
|
|
}
|
|
|
|
void RaceSelectionPreview::update(float angle)
|
|
{
|
|
mAnimation->runAnimation(0.0f);
|
|
|
|
mNode->setOrientation(Ogre::Quaternion(Ogre::Radian(angle), Ogre::Vector3::UNIT_Z)
|
|
* Ogre::Quaternion(mPitch, Ogre::Vector3::UNIT_X));
|
|
|
|
updateCamera();
|
|
}
|
|
|
|
void RaceSelectionPreview::render()
|
|
{
|
|
mTexture->load();
|
|
|
|
if (!mRenderTarget)
|
|
setupRenderTarget();
|
|
mRenderTarget->update();
|
|
}
|
|
|
|
void RaceSelectionPreview::setPrototype(const ESM::NPC &proto)
|
|
{
|
|
mBase = proto;
|
|
mBase.mId = "player";
|
|
rebuild();
|
|
mAnimation->runAnimation(0.0f);
|
|
updateCamera();
|
|
}
|
|
|
|
void RaceSelectionPreview::onSetup ()
|
|
{
|
|
mAnimation->play("idle", 1, Animation::Group_All, false, 1.0f, "start", "stop", 0.0f, 0);
|
|
|
|
updateCamera();
|
|
}
|
|
|
|
void RaceSelectionPreview::updateCamera()
|
|
{
|
|
Ogre::Vector3 scale = mNode->getScale();
|
|
Ogre::Node* headNode = mAnimation->getNode("Bip01 Head");
|
|
if (!headNode)
|
|
return;
|
|
Ogre::Vector3 headOffset = headNode->_getDerivedPosition();
|
|
headOffset = mNode->convertLocalToWorldPosition(headOffset);
|
|
|
|
mCamera->setPosition(headOffset + mPosition * scale);
|
|
mCamera->lookAt(headOffset + mPosition*Ogre::Vector3(0,1,0) * scale);
|
|
}
|
|
}
|