mirror of
https://port.numenaute.org/aleajactaest/khanat-opennel-code.git
synced 2025-01-28 02:08:41 +00:00
1486 lines
42 KiB
C++
1486 lines
42 KiB
C++
// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
|
||
// Copyright (C) 2010 Winch Gate Property Limited
|
||
//
|
||
// This program is free software: you can redistribute it and/or modify
|
||
// it under the terms of the GNU Affero General Public License as
|
||
// published by the Free Software Foundation, either version 3 of the
|
||
// License, or (at your option) any later version.
|
||
//
|
||
// 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 Affero General Public License for more details.
|
||
//
|
||
// You should have received a copy of the GNU Affero General Public License
|
||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
||
|
||
|
||
/////////////
|
||
// INCLUDE //
|
||
/////////////
|
||
#include "stdpch.h"
|
||
// Client
|
||
#include "misc.h"
|
||
#include "entity_animation_manager.h"
|
||
#include "entities.h"
|
||
#include "sheet_manager.h"
|
||
#include "interface_v3/interface_manager.h"
|
||
#include "continent_manager.h"
|
||
#include "world_database_manager.h"
|
||
#include "client_cfg.h"
|
||
#include "user_entity.h"
|
||
#include "net_manager.h"
|
||
#include "sky_render.h"
|
||
// Client Sheets
|
||
#include "client_sheets/item_sheet.h"
|
||
// 3D
|
||
#include "nel/3d/u_instance.h"
|
||
#include "nel/3d/u_instance_material.h"
|
||
#include "nel/3d/u_scene.h"
|
||
#include "nel/3d/u_driver.h"
|
||
#include "nel/3d/material.h"
|
||
// Misc
|
||
#include "nel/misc/file.h"
|
||
#include "nel/misc/i_xml.h"
|
||
#include "nel/misc/o_xml.h"
|
||
#include "nel/misc/fast_floor.h"
|
||
#include "nel/misc/noise_value.h"
|
||
#include "nel/misc/bitmap.h"
|
||
// Game Share
|
||
#include "game_share/player_visual_properties.h"
|
||
#include "game_share/seeds.h"
|
||
// Georges
|
||
#include "nel/georges/u_form.h"
|
||
#include "nel/georges/u_form_elm.h"
|
||
#include "nel/georges/u_form_loader.h"
|
||
|
||
#ifndef NL_OS_WINDOWS
|
||
#include <X11/Xlib.h>
|
||
#endif
|
||
|
||
///////////
|
||
// USING //
|
||
///////////
|
||
using namespace NL3D;
|
||
using namespace NLMISC;
|
||
using namespace std;
|
||
using namespace NLGEORGES;
|
||
|
||
////////////
|
||
// EXTERN //
|
||
////////////
|
||
extern CEntityAnimationManager *EAM;
|
||
extern UDriver *Driver;
|
||
extern UMaterial GenericMat;
|
||
|
||
|
||
////////////
|
||
// GLOBAL //
|
||
////////////
|
||
const uint RecordVersion = 1;
|
||
std::set<std::string> LodCharactersNotFound;
|
||
|
||
|
||
//////////////
|
||
// FUNCTION //
|
||
//////////////
|
||
//-----------------------------------------------
|
||
// keepIn_NegPi_Pi :
|
||
//-----------------------------------------------
|
||
double keepIn_NegPi_Pi(double angle)
|
||
{
|
||
if(angle > Pi)
|
||
angle = -2*Pi+angle;
|
||
else if(angle < -Pi)
|
||
angle = 2*Pi+angle;
|
||
|
||
return angle;
|
||
}// keepIn_NegPi_Pi //
|
||
|
||
//-----------------------------------------------
|
||
// angleBetween2Vect :
|
||
//-----------------------------------------------
|
||
double angleBetween2Vect(const CVectorD &from, const CVectorD &to)
|
||
{
|
||
// Get the final local head Yaw.
|
||
CVector vj = from;
|
||
vj.z = 0;
|
||
CVector vk(0,0,1);
|
||
CVector vi = vj^vk;
|
||
|
||
CMatrix mat;
|
||
mat.setRot(vi,vj,vk,true);
|
||
|
||
CVector localDir = mat.inverted() * to;
|
||
return atan2(-localDir.x, localDir.y);
|
||
}// angleBetween2Vect //
|
||
|
||
//-----------------------------------------------
|
||
// makeUp :
|
||
// \param face : pointer on the face to make up (must not be null).
|
||
// \param idMakeUp : index of the make-up to apply.
|
||
// \warning This function does not check if 'face' is valid.
|
||
//-----------------------------------------------
|
||
void makeUp(NL3D::UInstance face, sint idMakeUp)
|
||
{
|
||
static const char *tattooStr = "visage_makeup";
|
||
|
||
// look for tattoo texture
|
||
uint numMat = face.getNumMaterials();
|
||
std::string texFilename;
|
||
for(uint k = 0; k < numMat; ++k)
|
||
{
|
||
UInstanceMaterial im = face.getMaterial(k);
|
||
sint numTex = im.getLastTextureStage();
|
||
for(sint l = 0; l <= numTex; ++l)
|
||
{
|
||
if (im.isTextureFile(l)) // one texture from a file ?
|
||
{
|
||
// see if it is the texture used for tattoos
|
||
texFilename = im.getTextureFileName(l);
|
||
// nlinfo("visage tex = %s", texFilename.c_str());
|
||
std::string::size_type pos = texFilename.find(tattooStr, 0);
|
||
if (pos != std::string::npos)
|
||
{
|
||
uint charIndex = pos + strlen(tattooStr);
|
||
if (texFilename.length() >= charIndex + 2)
|
||
{
|
||
texFilename[charIndex] = '0' + (unsigned char) (idMakeUp / 10);
|
||
texFilename[charIndex + 1] = '0' + (unsigned char) (idMakeUp % 10);
|
||
im.setTextureFileName(texFilename, l);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}// makeUp //
|
||
|
||
//-----------------------------------------------
|
||
// qStart est le quaterion <20> t=0,
|
||
// qEnd est le quaterion <20> t=TAnimEnd,
|
||
// time est entre 0 et 1. (1=> TAnimEnd)
|
||
//-----------------------------------------------
|
||
CQuat applyRotationFactor(CQuat in, float rotFactor, CQuat qStart, CQuat qEnd, float time)
|
||
{
|
||
H_AUTO ( RZ_Client_Apply_Rotation_Factor )
|
||
|
||
CQuat qRotTotal1, qRotTotal2;
|
||
qStart.invert();
|
||
qRotTotal1= qEnd*qStart;
|
||
// qRotTotal2.makeClosest(CQuat::Identity);
|
||
qRotTotal2= CQuat::slerp(CQuat::Identity, qRotTotal1, rotFactor);
|
||
|
||
// apply animation factor
|
||
// qRotTotal1.makeClosest(CQuat::Identity);
|
||
// qRotTotal2.makeClosest(CQuat::Identity);
|
||
CQuat qRotDelta1= CQuat::slerp(CQuat::Identity, qRotTotal1, time);
|
||
CQuat qRotDelta2= CQuat::slerp(CQuat::Identity, qRotTotal2, time);
|
||
|
||
// remove normal rotation, and apply rotFactor-ed one.
|
||
qRotDelta1.invert();
|
||
return qRotDelta2 * qRotDelta1 * in;
|
||
}// applyRotationFactor //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// computeShortestAngle :
|
||
// Compute the angle between a source and a destination using the shortest way.
|
||
// \param from : angle between -Pi and Pi;
|
||
// \param to : angle between -Pi and Pi;
|
||
//-----------------------------------------------
|
||
double computeShortestAngle(double from, double to)
|
||
{
|
||
double difAngle = to - from;
|
||
if(difAngle < -Pi) // Angle in the wrong direction.
|
||
difAngle += 2.0*Pi;
|
||
else if(difAngle > Pi) // Angle in the wrong direction.
|
||
difAngle -= 2.0*Pi;
|
||
|
||
return difAngle;
|
||
}// computeShortestAngle //
|
||
|
||
//-----------------------------------------------
|
||
// readStringArray :
|
||
//-----------------------------------------------
|
||
void readStringArray(const std::string &filename, NLGEORGES::UFormLoader *formLoader, NLMISC::CSmartPtr<NLGEORGES::UForm> &form, std::map<std::string, std::string> &container)
|
||
{
|
||
// Read an array of strings
|
||
if(formLoader)
|
||
{
|
||
form = formLoader->loadForm(filename.c_str());
|
||
if(form)
|
||
{
|
||
// Get the root.
|
||
const UFormElm& rootElmt = form->getRootNode();
|
||
// Get animations.
|
||
const UFormElm *elmt = 0;
|
||
if(rootElmt.getNodeByName(&elmt, "array") == false)
|
||
nlwarning("readStringArray: the node 'array' is not referenced.");
|
||
// If the array is not empty (in fact exist).
|
||
if(elmt)
|
||
{
|
||
// Get the array size
|
||
uint arraySize;
|
||
elmt->getArraySize(arraySize);
|
||
// If there is at least 1 animation.
|
||
if(arraySize > 0)
|
||
{
|
||
// Get all animation for the State.
|
||
for(uint i = 0; i<arraySize; ++i)
|
||
{
|
||
// Get the Name
|
||
std::string nodeName;
|
||
std::string stringName;
|
||
if(elmt->getArrayNodeName(nodeName, i))
|
||
{
|
||
if(elmt->getArrayValue(stringName, i))
|
||
container.insert(make_pair(nodeName, stringName));
|
||
else
|
||
nlwarning("readStringArray: no string associated to the node '%d(%s)'.", i, nodeName.c_str());
|
||
}
|
||
else
|
||
nlwarning("readStringArray: node '%d', index valid.", i);
|
||
}
|
||
}
|
||
}
|
||
else
|
||
nlwarning("readStringArray: array node not allocated.");
|
||
}
|
||
else
|
||
nlwarning("readStringArray: cannot create the form from file '%s'.", filename.c_str());
|
||
}
|
||
else
|
||
nlwarning("readStringArray: the Loader is not allocated.");
|
||
}// readStringArray //
|
||
|
||
//-----------------------------------------------
|
||
// mode2Anim :
|
||
// Return the animset base name corresponding to the mode.
|
||
// \param mode : the mode to convert.
|
||
// \param result : this will be filed with the mode animset name.
|
||
// \return bool : 'true' if 'result' is filled, 'false' if left untouched.
|
||
//-----------------------------------------------
|
||
bool mode2Anim(MBEHAV::EMode mode, string &result)
|
||
{
|
||
static bool init = false;
|
||
static string mode2AnimArray[MBEHAV::NUMBER_OF_MODES];
|
||
|
||
// Is the mode valid
|
||
uint index = (uint)mode;
|
||
if(index >= MBEHAV::NUMBER_OF_MODES)
|
||
return false;
|
||
|
||
// Initialize
|
||
if(!init)
|
||
{
|
||
// Read mode2animset file
|
||
std::map<std::string, std::string> mode2Animset;
|
||
const std::string filename = "mode2animset.string_array";
|
||
NLGEORGES::UFormLoader *formLoader = UFormLoader::createLoader();
|
||
if(formLoader)
|
||
{
|
||
NLMISC::CSmartPtr<NLGEORGES::UForm> form;
|
||
readStringArray(filename, formLoader, form, mode2Animset);
|
||
}
|
||
else
|
||
nlwarning("mode2Anim: cannot create de loader");
|
||
NLGEORGES::UFormLoader::releaseLoader(formLoader);
|
||
// Initialize the static vector.
|
||
//-------------------------------
|
||
for(uint i=0; i<MBEHAV::NUMBER_OF_MODES; ++i)
|
||
{
|
||
const std::string modeName = MBEHAV::modeToString((MBEHAV::EMode)i);
|
||
std::map<std::string, std::string>::const_iterator it = mode2Animset.find(modeName);
|
||
if(it != mode2Animset.end())
|
||
{
|
||
mode2AnimArray[i] = (*it).second;
|
||
if(mode2AnimArray[i].empty())
|
||
nlwarning("mode2Anim: The mode '%d(%s)' has an empty animset associated.", i, modeName.c_str ());
|
||
}
|
||
// No animset for the mode.
|
||
else
|
||
{
|
||
mode2AnimArray[i] = "";
|
||
nlwarning("mode2Anim: no animset associated to the mode %d'%s'.", i, modeName.c_str());
|
||
}
|
||
}
|
||
// Init Done now.
|
||
init = true;
|
||
}
|
||
|
||
// Fill the result.
|
||
// Check name.
|
||
if(mode2AnimArray[index].empty())
|
||
result = mode2AnimArray[MBEHAV::NORMAL];
|
||
else
|
||
result = mode2AnimArray[index];
|
||
// Result filled.
|
||
return true;
|
||
}// mode2Anim //
|
||
|
||
//-----------------------------------------------
|
||
// computeAnimSet :
|
||
// Compute the animation set to use according to weapons, mode and race.
|
||
// \param animSet : result pointer.
|
||
// \param mode : the mode.
|
||
// \param animSetBaseName : basic name to construc the complet name of the animSet.
|
||
// \param leftHand : animSet name for the left hand.
|
||
// \param rightHand : animSet name for the right hand.
|
||
// \param lookAtItemsInHands : compute animset according to items in hands or not.
|
||
// \return bool : 'true' if the new animation set is the right one. 'false' if the one choosen is not the right one.
|
||
//-----------------------------------------------
|
||
bool computeAnimSet(const CAnimationSet *&animSet, MBEHAV::EMode mode, const string &animSetBaseName, const CItemSheet *itemLeftHand, const CItemSheet *itemRightHand, bool lookAtItemsInHands)
|
||
{
|
||
static std::set<std::string> UnknownAnimSet;
|
||
|
||
if(EAM == 0)
|
||
{
|
||
if(!ClientCfg.Light)
|
||
{
|
||
nlwarning("computeAnimSet: EAM not allocated -> cannot compute the anim set.");
|
||
}
|
||
return false;
|
||
}
|
||
|
||
string animSetRight, animSetLeft;
|
||
if(itemLeftHand)
|
||
animSetLeft = itemLeftHand->getAnimSet();
|
||
|
||
if(itemRightHand)
|
||
animSetRight = itemRightHand->getAnimSet();
|
||
|
||
// Get the animset name from the mode.
|
||
string result;
|
||
if(!mode2Anim(mode, result))
|
||
{
|
||
nlwarning("computeAnimSet: unknown mode '%d'.", mode);
|
||
result = "default";
|
||
}
|
||
|
||
// Compute the name.
|
||
if(lookAtItemsInHands)
|
||
result = animSetBaseName + "_" + result + "_" + animSetRight + "_" + animSetLeft;
|
||
else
|
||
result = animSetBaseName + "_" + result + "__";
|
||
|
||
// Get the animset.
|
||
const CAnimationSet *animSetTmp = EAM->getAnimSet(result);
|
||
if(animSetTmp)
|
||
{
|
||
animSet = animSetTmp;
|
||
return true;
|
||
}
|
||
// Bad one try something else.
|
||
else
|
||
{
|
||
// Up to 100 missing anim set (security).
|
||
if(UnknownAnimSet.size() < 100)
|
||
{
|
||
if(UnknownAnimSet.insert(result).second)
|
||
nlwarning("computeAnimSet: unknown Anim Set '%s' ('%u' unkowns).", result.c_str(), UnknownAnimSet.size());
|
||
}
|
||
// Try to compute the default one
|
||
result = animSetBaseName + "_" + "default" + "__";
|
||
animSetTmp = EAM->getAnimSet(result);
|
||
if(animSetTmp)
|
||
animSet = animSetTmp;
|
||
else
|
||
{
|
||
// Up to 100 missing anim set (security).
|
||
if(UnknownAnimSet.size() < 100)
|
||
{
|
||
if(UnknownAnimSet.insert(result).second)
|
||
nlwarning("computeAnimSet: unknown Anim Set '%s' ('%u' unkowns).", result.c_str(), UnknownAnimSet.size());
|
||
}
|
||
}
|
||
}
|
||
|
||
// Not Well done.
|
||
return false;
|
||
}// computeAnimSet //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// dump :
|
||
// Create a file with informations to debug.
|
||
//-----------------------------------------------
|
||
void dump(const std::string &name)
|
||
{
|
||
// Write informations to start as the version
|
||
COFile fStart;
|
||
if(fStart.open(name + "_start.rec", false, false))
|
||
{
|
||
CVectorD currentPos = UserEntity->pos();
|
||
fStart.serialVersion(RecordVersion);
|
||
fStart.serial(currentPos);
|
||
// Close the File.
|
||
fStart.close();
|
||
}
|
||
else
|
||
nlwarning("dump: cannot open/create the file '%s_start.rec'.", name.c_str());
|
||
|
||
// Write the DB
|
||
IngameDbMngr.write(name + "_db.rec");
|
||
// Open the file.
|
||
COFile f;
|
||
if(f.open(name + ".rec", false, false))
|
||
{
|
||
// Dump entities.
|
||
EntitiesMngr.dump(f);
|
||
|
||
// Dump Client CFG.
|
||
ClientCfg.serial(f);
|
||
|
||
// Close the File.
|
||
f.close();
|
||
}
|
||
else
|
||
nlwarning("dump: cannot open/create the file '%s.rec'.", name.c_str());
|
||
|
||
|
||
// Open the file.
|
||
if(f.open(name + ".xml", false, true))
|
||
{
|
||
// Create the XML stream
|
||
COXml output;
|
||
// Init
|
||
if(output.init (&f, "1.0"))
|
||
{
|
||
// Open the XML Dump.
|
||
output.xmlPush("XML");
|
||
|
||
// Dump Client CFG.
|
||
ClientCfg.serial(output);
|
||
|
||
// Dump entities.
|
||
EntitiesMngr.dumpXML(output);
|
||
|
||
// Close the XML Dump.
|
||
output.xmlPop();
|
||
|
||
// Flush the stream, write all the output file
|
||
output.flush();
|
||
}
|
||
else
|
||
nlwarning("dump: cannot initialize '%s.xml'.", name.c_str());
|
||
// Close the File.
|
||
f.close();
|
||
}
|
||
else
|
||
nlwarning("dump: cannot open/create the file '%s.xml'.", name.c_str());
|
||
}// dump //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// loadDump :
|
||
// Create a file with the current state of the client (good to report a bug).
|
||
//-----------------------------------------------
|
||
void loadDump(const std::string &name)
|
||
{
|
||
CVectorD currentPos;
|
||
|
||
// Load informations to start as the version
|
||
CIFile fStart;
|
||
if(fStart.open(name + "_start.rec", false))
|
||
{
|
||
fStart.serialVersion(RecordVersion);
|
||
fStart.serial(currentPos);
|
||
// Close the File.
|
||
fStart.close();
|
||
}
|
||
else
|
||
nlwarning("loadDump: cannot open the file '%s_start.rec'.", name.c_str());
|
||
|
||
// Update the position for the vision.
|
||
NetMngr.setReferencePosition(currentPos);
|
||
|
||
// Select the closest continent from the new position.
|
||
class CDummyProgress : public IProgressCallback
|
||
{
|
||
void progress (float /* value */) {}
|
||
};
|
||
CDummyProgress dummy;
|
||
ContinentMngr.select(currentPos, dummy);
|
||
|
||
// Load the DB
|
||
IngameDbMngr.read(name + "_db.rec");
|
||
|
||
// Open the file.
|
||
CIFile f;
|
||
if(f.open(name + ".rec", false))
|
||
{
|
||
// Dump entities.
|
||
EntitiesMngr.dump(f);
|
||
|
||
// Close the File.
|
||
f.close();
|
||
}
|
||
else
|
||
nlwarning("loadDump: cannot open '%s.rec'.", name.c_str());
|
||
}// loadDump //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// getLodCharacterId
|
||
// Get the lod character id from the scene, according to LodCharacterName. Cached.
|
||
// -1 if id not found.
|
||
//-----------------------------------------------
|
||
sint getLodCharacterId(UScene &scene, const string &lodCharacterName)
|
||
{
|
||
sint lodCharacterId = scene.getCLodShapeIdByName(lodCharacterName);
|
||
// display a warning for bad character Id, only if name was setup in the sheet
|
||
if(lodCharacterId==-1 && !lodCharacterName.empty() )
|
||
{
|
||
// Limited to 100 missing Lod to avoid memories problems
|
||
if(LodCharactersNotFound.size() < 100)
|
||
{
|
||
// Insert and display a waring
|
||
if(LodCharactersNotFound.insert(lodCharacterName).second)
|
||
nlwarning("getLodCharacterId: Not found A Character LodCharacter in the Manager: %s", lodCharacterName.c_str());
|
||
}
|
||
}
|
||
|
||
return lodCharacterId;
|
||
}// getLodCharacterId //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// getItem :
|
||
//-----------------------------------------------
|
||
CItemSheet *getItem(const CGenderInfo &genderInfo, SLOTTYPE::EVisualSlot slot)
|
||
{
|
||
CEntitySheet *faceItem = SheetMngr.get(CSheetId(genderInfo.getItemName(slot)));
|
||
return (dynamic_cast <CItemSheet *> (faceItem));
|
||
}// getItem //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// getColorIndex :
|
||
//-----------------------------------------------
|
||
sint getColorIndex(const CGenderInfo &genderInfo, SLOTTYPE::EVisualSlot slot)
|
||
{
|
||
CItemSheet *is = getItem(genderInfo, slot);
|
||
if(is == 0)
|
||
return 0;
|
||
return is->Color;
|
||
}// getColorIndex //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// buildPropVisualA :
|
||
//-----------------------------------------------
|
||
SPropVisualA buildPropVisualA(const CGenderInfo &genderInfo)
|
||
{
|
||
SPropVisualA dest;
|
||
dest.PropertyA = 0;
|
||
// setup items
|
||
dest.PropertySubData.JacketModel = SheetMngr.getVSIndex(genderInfo.Items[SLOTTYPE::CHEST_SLOT], SLOTTYPE::CHEST_SLOT);
|
||
dest.PropertySubData.TrouserModel = SheetMngr.getVSIndex(genderInfo.Items[SLOTTYPE::LEGS_SLOT], SLOTTYPE::LEGS_SLOT);
|
||
dest.PropertySubData.ArmModel = SheetMngr.getVSIndex(genderInfo.Items[SLOTTYPE::ARMS_SLOT], SLOTTYPE::ARMS_SLOT);
|
||
dest.PropertySubData.HatModel = SheetMngr.getVSIndex(genderInfo.Items[SLOTTYPE::HEAD_SLOT], SLOTTYPE::HEAD_SLOT);
|
||
|
||
// setup colors (fixed or user)
|
||
sint col = getColorIndex(genderInfo, SLOTTYPE::CHEST_SLOT);
|
||
dest.PropertySubData.JacketColor = (col == -1) ? 0 : col;
|
||
//
|
||
col = getColorIndex(genderInfo, SLOTTYPE::LEGS_SLOT);
|
||
dest.PropertySubData.TrouserColor = (col == -1) ? 0 : col;
|
||
//
|
||
col = getColorIndex(genderInfo, SLOTTYPE::ARMS_SLOT);
|
||
dest.PropertySubData.ArmColor = (col == -1) ? 0 : col;
|
||
//
|
||
col = getColorIndex(genderInfo, SLOTTYPE::HEAD_SLOT);
|
||
dest.PropertySubData.HatColor = (col == -1) ? 0 : col;
|
||
|
||
// sheath are not used
|
||
return dest;
|
||
}// buildPropVisualA //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// buildPropVisualB :
|
||
//-----------------------------------------------
|
||
SPropVisualB buildPropVisualB(const CGenderInfo &genderInfo)
|
||
{
|
||
// feet & hands
|
||
SPropVisualB dest;
|
||
dest.PropertyB = 0;
|
||
// setup items
|
||
dest.PropertySubData.HandsModel = SheetMngr.getVSIndex(genderInfo.Items[SLOTTYPE::HANDS_SLOT], SLOTTYPE::HANDS_SLOT);
|
||
dest.PropertySubData.FeetModel = SheetMngr.getVSIndex(genderInfo.Items[SLOTTYPE::FEET_SLOT], SLOTTYPE::FEET_SLOT);
|
||
|
||
// setup colors (fixed or user)
|
||
sint col = getColorIndex(genderInfo, SLOTTYPE::HANDS_SLOT);
|
||
dest.PropertySubData.HandsColor = (col == -1) ? 0 : col;
|
||
//
|
||
col = getColorIndex(genderInfo, SLOTTYPE::FEET_SLOT);
|
||
dest.PropertySubData.FeetColor = (col == -1) ? 0 : col;
|
||
|
||
return dest;
|
||
}// buildPropVisualB //
|
||
|
||
|
||
|
||
|
||
//-----------------------------------------------
|
||
// isUserColorSupported :
|
||
// Test wether user color is supported for this equipment
|
||
//-----------------------------------------------
|
||
bool isUserColorSupported(const CPlayerSheet::CEquipment &equip)
|
||
{
|
||
NLMISC::CSheetId si;
|
||
if(!si.buildSheetId(equip.Item))
|
||
return false;
|
||
|
||
CItemSheet *is = dynamic_cast<CItemSheet *>(SheetMngr.get(si));
|
||
if(!is)
|
||
return false;
|
||
|
||
return is->Color == -1; // user color supported by the item
|
||
}// isUserColorSupported //
|
||
|
||
//-----------------------------------------------
|
||
// isUserColorSupported :
|
||
// Test wether user color is supported for a given visual slot
|
||
//-----------------------------------------------
|
||
bool isUserColorSupported(const CPlayerSheet &playerSheet, SLOTTYPE::EVisualSlot vs)
|
||
{
|
||
switch(vs)
|
||
{
|
||
case SLOTTYPE::CHEST_SLOT: return isUserColorSupported(playerSheet.Body);
|
||
case SLOTTYPE::LEGS_SLOT: return isUserColorSupported(playerSheet.Legs);
|
||
case SLOTTYPE::HEAD_SLOT: return isUserColorSupported(playerSheet.Head);
|
||
case SLOTTYPE::ARMS_SLOT: return isUserColorSupported(playerSheet.Arms);
|
||
case SLOTTYPE::HANDS_SLOT: return isUserColorSupported(playerSheet.Hands);
|
||
case SLOTTYPE::FEET_SLOT: return isUserColorSupported(playerSheet.Feet);
|
||
case SLOTTYPE::RIGHT_HAND_SLOT: return isUserColorSupported(playerSheet.ObjectInRightHand);
|
||
case SLOTTYPE::LEFT_HAND_SLOT: return isUserColorSupported(playerSheet.ObjectInLeftHand);
|
||
}
|
||
return false;
|
||
}// isUserColorSupported //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// getColor :
|
||
//-----------------------------------------------
|
||
sint8 getColor(const CPlayerSheet::CEquipment &equip)
|
||
{
|
||
NLMISC::CSheetId si;
|
||
if(!si.buildSheetId(equip.Item))
|
||
return -2;
|
||
|
||
CItemSheet *is = dynamic_cast<CItemSheet *>(SheetMngr.get(si));
|
||
if(!is)
|
||
return -2;
|
||
|
||
if(is->Color == -1) // user color supported by the item
|
||
{
|
||
return equip.Color;
|
||
}
|
||
else
|
||
{
|
||
return is->Color;
|
||
}
|
||
}// getColor //
|
||
|
||
//-----------------------------------------------
|
||
// buildPropVisualA :
|
||
//-----------------------------------------------
|
||
SPropVisualA buildPropVisualA(const CPlayerSheet &playerSheet)
|
||
{
|
||
SPropVisualA vp;
|
||
// setup items
|
||
vp.PropertySubData.JacketModel = SheetMngr.getVSIndex(playerSheet.Body.Item, SLOTTYPE::CHEST_SLOT);
|
||
vp.PropertySubData.TrouserModel = SheetMngr.getVSIndex(playerSheet.Legs.Item, SLOTTYPE::LEGS_SLOT);
|
||
vp.PropertySubData.ArmModel = SheetMngr.getVSIndex(playerSheet.Arms.Item, SLOTTYPE::ARMS_SLOT);
|
||
vp.PropertySubData.HatModel = SheetMngr.getVSIndex(playerSheet.Head.Item, SLOTTYPE::HEAD_SLOT);
|
||
vp.PropertySubData.WeaponRightHand = SheetMngr.getVSIndex(playerSheet.ObjectInRightHand.Item, SLOTTYPE::RIGHT_HAND_SLOT);
|
||
vp.PropertySubData.WeaponLeftHand = SheetMngr.getVSIndex(playerSheet.ObjectInLeftHand.Item, SLOTTYPE::LEFT_HAND_SLOT);
|
||
|
||
// setup colors
|
||
vp.PropertySubData.JacketColor = getColor(playerSheet.Body);
|
||
vp.PropertySubData.TrouserColor = getColor(playerSheet.Legs);
|
||
vp.PropertySubData.ArmColor = getColor(playerSheet.Arms);
|
||
vp.PropertySubData.HatColor = getColor(playerSheet.Head);
|
||
|
||
return vp;
|
||
}// buildPropVisualA //
|
||
|
||
//-----------------------------------------------
|
||
// buildPropVisualB :
|
||
//-----------------------------------------------
|
||
SPropVisualB buildPropVisualB(const CPlayerSheet &playerSheet)
|
||
{
|
||
SPropVisualB vp;
|
||
// setup items
|
||
vp.PropertySubData.Name = 0;
|
||
vp.PropertySubData.HandsModel = SheetMngr.getVSIndex(playerSheet.Hands.Item, SLOTTYPE::HANDS_SLOT);
|
||
vp.PropertySubData.FeetModel = SheetMngr.getVSIndex(playerSheet.Feet.Item, SLOTTYPE::FEET_SLOT);
|
||
|
||
// setup colors
|
||
vp.PropertySubData.HandsColor = getColor(playerSheet.Hands);
|
||
vp.PropertySubData.FeetColor = getColor(playerSheet.Feet);
|
||
|
||
return vp;
|
||
}// buildPropVisualB //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// drawBox :
|
||
// Draw a Box from 2 vectors.
|
||
//-----------------------------------------------
|
||
void drawBox(const CVector &vMin, const CVector &vMax, const CRGBA &color)
|
||
{
|
||
CLineColor line;
|
||
line.Color0 = color;
|
||
line.Color1 = color;
|
||
// Bottom quad
|
||
line = CLine(CVector(vMin.x,vMin.y,vMin.z), CVector(vMax.x,vMin.y,vMin.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
line = CLine(CVector(vMax.x,vMin.y,vMin.z), CVector(vMax.x,vMax.y,vMin.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
line = CLine(CVector(vMax.x,vMax.y,vMin.z), CVector(vMin.x,vMax.y,vMin.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
line = CLine(CVector(vMin.x,vMax.y,vMin.z), CVector(vMin.x,vMin.y,vMin.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
// Top quad
|
||
line = CLine(CVector(vMin.x,vMin.y,vMax.z), CVector(vMax.x,vMin.y,vMax.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
line = CLine(CVector(vMax.x,vMin.y,vMax.z), CVector(vMax.x,vMax.y,vMax.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
line = CLine(CVector(vMax.x,vMax.y,vMax.z), CVector(vMin.x,vMax.y,vMax.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
line = CLine(CVector(vMin.x,vMax.y,vMax.z), CVector(vMin.x,vMin.y,vMax.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
// Sides Quad
|
||
line = CLine(CVector(vMin.x,vMin.y,vMin.z), CVector(vMin.x,vMin.y,vMax.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
line = CLine(CVector(vMax.x,vMin.y,vMin.z), CVector(vMax.x,vMin.y,vMax.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
line = CLine(CVector(vMax.x,vMax.y,vMin.z), CVector(vMax.x,vMax.y,vMax.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
line = CLine(CVector(vMin.x,vMax.y,vMin.z), CVector(vMin.x,vMax.y,vMax.z));
|
||
Driver->drawLine(line, GenericMat);
|
||
}// drawBox //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// drawSphere :
|
||
// Draw a Sphere
|
||
//-----------------------------------------------
|
||
void drawSphere(const NLMISC::CVector ¢er, float radius, const NLMISC::CRGBA &color)
|
||
{
|
||
const uint numSegs= 12;
|
||
CLineColor line;
|
||
line.Color0 = color;
|
||
line.Color1 = color;
|
||
// For all faces
|
||
for(uint face=0;face<3;face++)
|
||
{
|
||
for(uint i=0;i<numSegs;i++)
|
||
{
|
||
float angStart= float(i*2*Pi/numSegs);
|
||
float angEnd= float(((i+1)%numSegs)*2*Pi/numSegs);
|
||
line.V0= radius*CVector(cosf(angStart), sinf(angStart), 0.f);
|
||
line.V1= radius*CVector(cosf(angEnd), sinf(angEnd), 0.f);
|
||
if(face==1)
|
||
{
|
||
swap(line.V0.x, line.V0.z);
|
||
swap(line.V1.x, line.V1.z);
|
||
}
|
||
else if(face==2)
|
||
{
|
||
swap(line.V0.y, line.V0.z);
|
||
swap(line.V1.y, line.V1.z);
|
||
}
|
||
line.V0+= center;
|
||
line.V1+= center;
|
||
Driver->drawLine(line, GenericMat);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
//-----------------------------------------------
|
||
// getSeedsFromDB :
|
||
//-----------------------------------------------
|
||
void getSeedsFromDB(CSeeds &dest)
|
||
{
|
||
CInterfaceManager *im =CInterfaceManager::getInstance();
|
||
nlctassert(sizeof(CSeeds::TUInt) == 4); // excpect that the number of each seed type is encoded on 32 bits
|
||
// if this assert at compile, change the following code
|
||
string ls = im->getDefine("money_1");
|
||
string ms = im->getDefine("money_2");
|
||
string bs = im->getDefine("money_3");
|
||
string vbs = im->getDefine("money_4");
|
||
|
||
dest = CSeeds(im->getDbProp(ls)->getValue32(),
|
||
im->getDbProp(ms)->getValue32(),
|
||
im->getDbProp(bs)->getValue32(),
|
||
im->getDbProp(vbs)->getValue32());
|
||
} // getSeedsFromDB //
|
||
|
||
|
||
//-----------------------------------------------
|
||
// setVect :
|
||
// Change a 'direction' vector.
|
||
// \param vectToChange : the vector to change.
|
||
// \param vect : new vector to use.
|
||
// \param compute : adjust the param 'vect' to be valid or leave the old one unchanged if impossible.
|
||
// \param check : warning if the param 'vect' is not valid (vector Null) even with compute=true.
|
||
// \return bool : 'true' if the vectToChange has been filled, else 'false'.
|
||
//-----------------------------------------------
|
||
bool setVect(CVector &vectToChange, const CVector &vect, bool compute, bool check)
|
||
{
|
||
// Compute the vector.
|
||
if(compute)
|
||
{
|
||
CVector vectTmp(vect);
|
||
// No need of the Z component for the vector.
|
||
vectTmp.z = 0.0f;
|
||
// Check vector.
|
||
if(vectTmp != CVector::Null)
|
||
{
|
||
vectTmp.normalize();
|
||
vectToChange = vectTmp;
|
||
return true;
|
||
}
|
||
// Bad Vector -> vectToChange remains the same
|
||
else
|
||
{
|
||
// Warning
|
||
if(check)
|
||
nlwarning("setVect: cannot compute the vector, keep the old one.");
|
||
return false;
|
||
}
|
||
}
|
||
|
||
// Bad Vector -> vectToChange remains the same
|
||
if(vect == CVector::Null)
|
||
{
|
||
// Warning
|
||
if(check)
|
||
nlwarning("setVect: param is a vector Null, vectToChange remains the same.");
|
||
return false;
|
||
}
|
||
|
||
// Set the new front vector.
|
||
vectToChange = vect;
|
||
return true;
|
||
}// setVect //
|
||
|
||
NLMISC::CRGBA interpClientCfgColor(const ucstring &src, ucstring &dest)
|
||
{
|
||
CRGBA color = CRGBA::White;
|
||
if (src.size() >= 3)
|
||
{
|
||
if (src[0] == (ucchar) '&')
|
||
{
|
||
ucstring::size_type nextPos = src.find((ucchar) '&', 1);
|
||
if (nextPos != ucstring::npos)
|
||
{
|
||
std::string colorCode;
|
||
colorCode.resize(nextPos - 1);
|
||
for(uint k = 0; k < nextPos - 1; ++k)
|
||
{
|
||
colorCode[k] = tolower((char) src[k + 1]);
|
||
}
|
||
std::map<std::string, CClientConfig::SSysInfoParam>::const_iterator it = ClientCfg.SystemInfoParams.find(colorCode);
|
||
if (it != ClientCfg.SystemInfoParams.end())
|
||
{
|
||
color = it->second.Color;
|
||
}
|
||
dest = src.substr(nextPos + 1);
|
||
}
|
||
else
|
||
{
|
||
dest = src;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
dest = src;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
dest = src;
|
||
}
|
||
return color;
|
||
}
|
||
|
||
std::string getStringCategory(const ucstring &src, ucstring &dest, bool alwaysAddSysByDefault)
|
||
{
|
||
std::string str = getStringCategoryIfAny(src, dest);
|
||
if (alwaysAddSysByDefault)
|
||
return str.empty()?"SYS":str;
|
||
else
|
||
return str;
|
||
}
|
||
|
||
|
||
std::string getStringCategoryIfAny(const ucstring &src, ucstring &dest)
|
||
{
|
||
std::string colorCode = "";
|
||
if (src.size() >= 3)
|
||
{
|
||
uint startPos = 0;
|
||
|
||
// Skip <NEW> or <CHG> if present at beginning
|
||
ucstring preTag;
|
||
const uint PreTagSize = 5;
|
||
const ucstring newTag("<NEW>");
|
||
if ( (src.size() >= PreTagSize) && (src.substr( 0, PreTagSize ) == newTag) )
|
||
{
|
||
startPos = PreTagSize;
|
||
preTag = newTag;
|
||
}
|
||
const ucstring chgTag("<CHG>");
|
||
if ( (src.size() >= PreTagSize) && (src.substr( 0, PreTagSize ) == chgTag) )
|
||
{
|
||
startPos = PreTagSize;
|
||
preTag = chgTag;
|
||
}
|
||
|
||
if (src[startPos] == (ucchar) '&')
|
||
{
|
||
ucstring::size_type nextPos = src.find((ucchar) '&', startPos+1);
|
||
if (nextPos != ucstring::npos)
|
||
{
|
||
uint codeSize = nextPos - startPos - 1;
|
||
colorCode.resize( codeSize );
|
||
for(uint k = 0; k < codeSize; ++k)
|
||
{
|
||
colorCode[k] = tolower((char) src[k + startPos + 1]);
|
||
}
|
||
ucstring destTmp;
|
||
if ( startPos != 0 )
|
||
destTmp = preTag; // leave <NEW> or <CHG> in the dest string
|
||
destTmp += src.substr(nextPos + 1);
|
||
dest = destTmp;
|
||
}
|
||
else
|
||
{
|
||
dest = src;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
dest = src;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
dest = src;
|
||
}
|
||
return colorCode;
|
||
}
|
||
|
||
|
||
NLMISC::CRGBA stringToRGBA(const char *ptr)
|
||
{
|
||
if (!ptr) return NLMISC::CRGBA::White;
|
||
int r = 255, g = 255, b = 255, a = 255;
|
||
sscanf (ptr, "%d %d %d %d", &r, &g, &b, &a);
|
||
NLMISC::clamp (r, 0, 255);
|
||
NLMISC::clamp (g, 0, 255);
|
||
NLMISC::clamp (b, 0, 255);
|
||
NLMISC::clamp (a, 0, 255);
|
||
return CRGBA(r,g,b,a);
|
||
}
|
||
|
||
|
||
|
||
|
||
// ***************************************************************************
|
||
|
||
inline bool isSeparator (ucchar c)
|
||
{
|
||
return (c == ' ') || (c == '\t') || (c == '\n') || (c == '\r');
|
||
}
|
||
|
||
// ***************************************************************************
|
||
|
||
inline bool isEndSentence (ucstring& str, uint index)
|
||
{
|
||
// Ex: One sentence. Another sentence.
|
||
// ^
|
||
// Counterexample: nevrax.com
|
||
// ^
|
||
ucchar c = str[index];
|
||
if ((str[index] == ' ') || (str[index] == '\n'))
|
||
{
|
||
if (index < 1)
|
||
return false;
|
||
c = str[index-1];
|
||
return (c == '.') || (c == '!') || (c == '?');
|
||
}
|
||
return false;
|
||
}
|
||
|
||
|
||
void setCase (ucstring &str, TCaseMode mode)
|
||
{
|
||
const uint length = str.length();
|
||
uint i;
|
||
bool newString = true;
|
||
bool newSentence = true;
|
||
bool newWord = true;
|
||
switch (mode)
|
||
{
|
||
case CaseLower:
|
||
str = toLower (str);
|
||
break;
|
||
case CaseUpper:
|
||
str = toUpper (str);
|
||
break;
|
||
case CaseFirstStringLetterUp:
|
||
for (i=0; i<length; i++)
|
||
{
|
||
if (!isSeparator (str[i]))
|
||
{
|
||
if (newString)
|
||
str[i] = toUpper (str[i]);
|
||
else
|
||
str[i] = toLower (str[i]);
|
||
newString = false;
|
||
}
|
||
}
|
||
break;
|
||
case CaseFirstSentenceLetterUp:
|
||
for (i=0; i<length; i++)
|
||
{
|
||
if (isEndSentence (str, i))
|
||
newSentence = true;
|
||
else
|
||
{
|
||
if (newSentence)
|
||
str[i] = toUpper (str[i]);
|
||
else
|
||
str[i] = toLower (str[i]);
|
||
|
||
if (!isSeparator (str[i]))
|
||
newSentence = false;
|
||
}
|
||
}
|
||
break;
|
||
case CaseFirstWordLetterUp:
|
||
for (i=0; i<length; i++)
|
||
{
|
||
if (isSeparator (str[i]) || isEndSentence (str, i))
|
||
newWord = true;
|
||
else
|
||
{
|
||
if (newWord)
|
||
str[i] = toUpper (str[i]);
|
||
else
|
||
str[i] = toLower (str[i]);
|
||
|
||
newWord = false;
|
||
}
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
|
||
|
||
// ***************************************************************************
|
||
sint ucstrnicmp(const ucstring &s0, uint p0, uint n0, const ucstring &s1)
|
||
{
|
||
// start
|
||
const ucchar *start1= s1.c_str();
|
||
uint lenS1= s1.size();
|
||
const ucchar *start0= s0.c_str();
|
||
uint lenS0= s0.size();
|
||
if(p0!=0)
|
||
{
|
||
if(p0<lenS0)
|
||
{
|
||
start0+= p0;
|
||
lenS0-= p0;
|
||
}
|
||
else
|
||
{
|
||
start0+= lenS0; // points to '\0'
|
||
lenS0= 0;
|
||
}
|
||
}
|
||
lenS0= min(lenS0, n0);
|
||
|
||
// compare character to character
|
||
while(lenS0>0 && lenS1>0)
|
||
{
|
||
ucchar c0= toLower(*start0++);
|
||
ucchar c1= toLower(*start1++);
|
||
if(c0!=c1)
|
||
return c0<c1?-1:+1;
|
||
lenS0--;
|
||
lenS1--;
|
||
}
|
||
|
||
// return -1 if s1>s0, 1 if s0>s1, or 0 if equals
|
||
if(lenS1>0)
|
||
return -1;
|
||
else if(lenS0>0)
|
||
return 1;
|
||
else
|
||
return 0;
|
||
}
|
||
|
||
|
||
// *******************************************************************************************
|
||
float computeUniformNoise(const NLMISC::CNoiseValue &nv, const CVector &pos)
|
||
{
|
||
NLMISC::OptFastFloorBegin();
|
||
float value = nv.eval(pos);
|
||
value = 10.f * fmodf(value, 0.1f); // make repartition more uniform
|
||
NLMISC::OptFastFloorEnd();
|
||
return value;
|
||
}
|
||
|
||
|
||
// ***************************************************************************
|
||
void computeCurrentFovAspectRatio(float &fov, float &ar)
|
||
{
|
||
// compute the fov
|
||
fov = (float)(ClientCfg.FoV*Pi/180.0);
|
||
|
||
// get the screen aspect ratio from CFG
|
||
ar = ClientCfg.ScreenAspectRatio;
|
||
|
||
// if Driver is not created, we can't get current screen mode
|
||
if (!Driver) return;
|
||
|
||
// if windowed, must modulate aspect ratio by (WindowResolution / ScreenResolution)
|
||
if(ClientCfg.Windowed)
|
||
{
|
||
uint32 wndW, wndH;
|
||
Driver->getWindowSize(wndW, wndH);
|
||
UDriver::CMode mode;
|
||
Driver->getCurrentScreenMode(mode);
|
||
if(wndH)
|
||
{
|
||
// compute window aspect ratio
|
||
float arWnd= float(wndW) / float(wndH);
|
||
if (ar == 0.f)
|
||
{
|
||
// auto mode, we are using window aspect ratio
|
||
ar = arWnd;
|
||
}
|
||
else if (mode.Width && mode.Height)
|
||
{
|
||
// compute screen aspect ratio
|
||
float arScreen= float(mode.Width) / float(mode.Height);
|
||
ar *= arWnd / arScreen;
|
||
}
|
||
}
|
||
}
|
||
// if fullscreen, must modulate aspect ratio by ScreenResolution
|
||
else
|
||
{
|
||
if (ar == 0.f)
|
||
{
|
||
UDriver::CMode mode;
|
||
Driver->getCurrentScreenMode(mode);
|
||
if(mode.Height)
|
||
{
|
||
ar = float(mode.Width) / float(mode.Height);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// ***************************************************************************
|
||
void drawDisc(CBitmap &dest, float x, float y, float radius, const CRGBA &color, bool additif /*= false*/, uint numSegs /*= 127*/)
|
||
{
|
||
CPolygon2D poly;
|
||
poly.Vertices.resize(numSegs);
|
||
for(uint k = 0; k < numSegs; ++k)
|
||
{
|
||
poly.Vertices[k].set(x + radius * (float) cos(k / (float) numSegs * 2 * Pi), y + radius * (float) sin(k / (float) numSegs * 2 * Pi));
|
||
}
|
||
CPolygon2D::TRasterVect rasters;
|
||
sint minY;
|
||
poly.computeOuterBorders(rasters, minY);
|
||
sint maxY = std::min((sint) dest.getHeight(), (sint) rasters.size() + minY);
|
||
for (sint y = std::max((sint) 0, minY); y < maxY; ++y)
|
||
{
|
||
nlassert(y >= 0 && y < (sint) dest.getHeight());
|
||
sint minX = std::max((sint) 0, rasters[y - minY].first);
|
||
sint maxX = std::min((sint) dest.getWidth(), rasters[y - minY].second);
|
||
if (maxX > minX)
|
||
{
|
||
CRGBA *pt = (CRGBA *) &dest.getPixels(0)[0];
|
||
pt += y * dest.getWidth() + minX;
|
||
const CRGBA *endPt = pt + (maxX - minX);
|
||
while (pt != endPt)
|
||
{
|
||
if (additif)
|
||
{
|
||
pt->add(*pt, color);
|
||
}
|
||
else
|
||
{
|
||
*pt = color;
|
||
}
|
||
++ pt;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
// *************************************************************************************************
|
||
NL3D::UScene *getSkyScene()
|
||
{
|
||
if (ContinentMngr.cur() && !ContinentMngr.cur()->Indoor)
|
||
{
|
||
CSky &sky = ContinentMngr.cur()->CurrentSky;
|
||
if (sky.getScene())
|
||
{
|
||
return sky.getScene();
|
||
}
|
||
else
|
||
{
|
||
return SkyScene; // old sky rendering
|
||
}
|
||
}
|
||
return NULL;
|
||
}
|
||
|
||
// *************************************************************************************************
|
||
void setEmissive(NL3D::UInstance instance, const NLMISC::CRGBA &color)
|
||
{
|
||
if (instance.empty()) return;
|
||
for(uint k = 0; k < instance.getNumMaterials(); ++k)
|
||
{
|
||
NL3D::UInstanceMaterial mat = instance.getMaterial(k);
|
||
mat.setEmissive(color);
|
||
}
|
||
}
|
||
|
||
// *************************************************************************************************
|
||
void setDiffuse(NL3D::UInstance instance, bool onOff, const NLMISC::CRGBA &color)
|
||
{
|
||
if (instance.empty()) return;
|
||
for(uint k = 0; k < instance.getNumMaterials(); ++k)
|
||
{
|
||
NL3D::UInstanceMaterial mat = instance.getMaterial(k);
|
||
if (mat.isLighted())
|
||
{
|
||
CRGBA src;
|
||
if (onOff)
|
||
{
|
||
src = mat.getDiffuse();
|
||
src.R = color.R;
|
||
src.G = color.G;
|
||
src.B = color.B;
|
||
}
|
||
else
|
||
{
|
||
NL3D::UMaterial matShape = instance.getShape().getMaterial(k);
|
||
src = matShape.getDiffuse();
|
||
src.A = mat.getDiffuse().A;
|
||
}
|
||
mat.setDiffuse(src);
|
||
}
|
||
else
|
||
{
|
||
CRGBA src;
|
||
if (onOff)
|
||
{
|
||
src = mat.getColor();
|
||
src.R = color.R;
|
||
src.G = color.G;
|
||
src.B = color.B;
|
||
}
|
||
else
|
||
{
|
||
NL3D::UMaterial matShape = instance.getShape().getMaterial(k);
|
||
src = matShape.getColor();
|
||
src.A = mat.getColor().A;
|
||
}
|
||
mat.setColor(src);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
|
||
// *************************************************************************************************
|
||
void makeInstanceTransparent(UInstance &inst, uint8 opacity, bool disableZWrite)
|
||
{
|
||
UShape shape= inst.getShape();
|
||
if(shape.empty())
|
||
return;
|
||
uint numMats= shape.getNumMaterials();
|
||
if(numMats==0)
|
||
return;
|
||
if(numMats!=inst.getNumMaterials())
|
||
return;
|
||
|
||
// instance transparent or not?
|
||
if (opacity == 255)
|
||
{
|
||
// reset default shape opacity / transparency
|
||
inst.setOpacity(shape.getDefaultOpacity());
|
||
inst.setTransparency(shape.getDefaultTransparency());
|
||
inst.setBypassLODOpacityFlag(false);
|
||
}
|
||
else
|
||
{
|
||
// Will have some blend material => sure not to be rendered in Opaque pass
|
||
inst.setOpacity(false);
|
||
inst.setTransparency(true);
|
||
inst.setBypassLODOpacityFlag(true); // these flags prevails over the current lods flags for multi-lod objects
|
||
}
|
||
|
||
// set all materials
|
||
for (uint32 j = 0; j < numMats; ++j)
|
||
{
|
||
NL3D::UInstanceMaterial matInst = inst.getMaterial(j);
|
||
NL3D::UMaterial matShape= shape.getMaterial(j);
|
||
|
||
// disalbe zwrite?
|
||
if(disableZWrite)
|
||
matInst.setZWrite(false);
|
||
else
|
||
matInst.setZWrite(matShape.getZWrite());
|
||
|
||
// if no more transparent
|
||
if (opacity == 255)
|
||
{
|
||
// reset to default
|
||
matInst.setBlend(matShape.getBlend());
|
||
matInst.setBlendFunc((NL3D::UInstanceMaterial::TBlend)matShape.getSrcBlend(),
|
||
(NL3D::UInstanceMaterial::TBlend)matShape.getDstBlend());
|
||
// if orginal material is opaque or additif and has no alpha test, then ensure restore last tex env if needed
|
||
CMaterial *destInternalMat = matInst.getObjectPtr();
|
||
if (!matShape.getBlend() && !matShape.getAlphaTest())
|
||
{
|
||
if (destInternalMat->getShader() == CMaterial::Normal)
|
||
{
|
||
CMaterial *srcInternalMat = matShape.getObjectPtr();
|
||
uint numTex = 0;
|
||
for (;numTex < 4 && srcInternalMat->getTexture(numTex) != NULL; ++numTex) {}
|
||
if (numTex > 0)
|
||
{
|
||
if (srcInternalMat->getTexEnvMode(numTex - 1) != destInternalMat->getTexEnvMode(numTex - 1))
|
||
{
|
||
destInternalMat->setTexEnvMode(numTex - 1, srcInternalMat->getTexEnvMode(numTex - 1));
|
||
}
|
||
}
|
||
}
|
||
}
|
||
if (destInternalMat->getShader() == CMaterial::Normal)
|
||
{
|
||
// if !lighted, restore color
|
||
if (!destInternalMat->isLighted())
|
||
{
|
||
CMaterial *srcInternalMat = matShape.getObjectPtr();
|
||
// restore alpha in color
|
||
CRGBA color = destInternalMat->getColor();
|
||
color.A = srcInternalMat->getColor().A;
|
||
destInternalMat->setColor(color);
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
// Enable blend
|
||
matInst.setBlend(true);
|
||
// If default is ???/one or , then use a srcalpha/one (eg: for Diamond-like weapons)
|
||
if(matShape.getBlend() && (sint32)matShape.getDstBlend()==(sint32)NL3D::UInstanceMaterial::one)
|
||
matInst.setBlendFunc(NL3D::UInstanceMaterial::srcalpha, NL3D::UInstanceMaterial::one);
|
||
// else use a standard srcalpha/invsrcalpha
|
||
else
|
||
matInst.setBlendFunc(NL3D::UInstanceMaterial::srcalpha, NL3D::UInstanceMaterial::invsrcalpha);
|
||
// if orginal material is opaque or additif and has no alpha test, then ensure that the alpha output is 'diffuse'
|
||
CMaterial *internalMat = matInst.getObjectPtr();
|
||
if (!matShape.getBlend() && !matShape.getAlphaTest())
|
||
{
|
||
if (internalMat->getShader() == CMaterial::Normal)
|
||
{
|
||
uint numTex = 0;
|
||
for (;numTex < 4 && internalMat->getTexture(numTex) != NULL; ++numTex) {}
|
||
if (numTex > 0)
|
||
{
|
||
internalMat->texEnvOpAlpha(numTex - 1, CMaterial::Replace);
|
||
// if material is unlighted, then use the constant at this stage to set the alpha
|
||
internalMat->texEnvArg0Alpha(numTex - 1, CMaterial::Diffuse, CMaterial::SrcAlpha);
|
||
}
|
||
}
|
||
}
|
||
if (internalMat->getShader() == CMaterial::Normal)
|
||
{
|
||
if (!internalMat->isLighted())
|
||
{
|
||
// replace alpha in color
|
||
CRGBA color = internalMat->getColor();
|
||
color.A = opacity;
|
||
internalMat->setColor(color);
|
||
}
|
||
}
|
||
}
|
||
|
||
// suppose that default opacity is always 255
|
||
if (matInst.isLighted())
|
||
{
|
||
matInst.setOpacity(opacity);
|
||
}
|
||
|
||
matInst.setAlphaTestThreshold(matShape.getAlphaTestThreshold()*((float)opacity)/255.0f);
|
||
|
||
}
|
||
}
|
||
|
||
void setVideoMode(const UDriver::CMode &mode)
|
||
{
|
||
UDriver::CMode oldMode;
|
||
oldMode.Windowed = true; // getCurrentScreenMode may fail if first init ...
|
||
Driver->getCurrentScreenMode(oldMode);
|
||
Driver->setMode(mode);
|
||
if (oldMode.Windowed && !mode.Windowed) // going to fullscreen ?
|
||
{
|
||
/*CInterfaceManager *pIM = CInterfaceManager::getInstance();
|
||
pIM->movePointerAbs((sint32) mode.Width / 2, (sint32) mode.Height / 2);
|
||
Driver->setMousePos(0.5f, 0.5f);*/
|
||
}
|
||
}
|
||
|
||
uint getCurrentColorDepth()
|
||
{
|
||
if (Driver && Driver->isActive())
|
||
{
|
||
UDriver::CMode videoMode;
|
||
Driver->getCurrentScreenMode(videoMode);
|
||
if (!videoMode.Windowed)
|
||
{
|
||
return videoMode.Depth;
|
||
}
|
||
}
|
||
uint depth = 0;
|
||
#ifdef NL_OS_WINDOWS
|
||
HWND desktopWnd = GetDesktopWindow();
|
||
if (desktopWnd)
|
||
{
|
||
HDC desktopDC = GetWindowDC(desktopWnd);
|
||
if (desktopDC)
|
||
{
|
||
depth = (uint) GetDeviceCaps(desktopDC, BITSPIXEL);
|
||
ReleaseDC(desktopWnd, desktopDC);
|
||
}
|
||
}
|
||
#else
|
||
Display *display = XOpenDisplay(NULL);
|
||
if (display)
|
||
{
|
||
depth = (uint) DefaultDepth(display, DefaultScreen(display));
|
||
XCloseDisplay(display);
|
||
}
|
||
#endif
|
||
return depth;
|
||
}
|
||
|
||
|