// Ryzom - MMORPG Framework // 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 . #include "stdpch.h" #include "sbrick_manager.h" #include "interface_manager.h" #include "../sheet_manager.h" #include "dbgroup_build_phrase.h" #include "skill_manager.h" using namespace std; using namespace NLMISC; using namespace NLGEORGES; CSBrickManager* CSBrickManager::_Instance = NULL; // *************************************************************************** void CSBrickManager::releaseInstance() { if( _Instance ) { delete _Instance; _Instance = NULL; } } // *************************************************************************** CSBrickManager::CSBrickManager() : _NbFamily(0), _SabrinaCom(&_BrickContainer) { _BrickFamilyObs.Owner= this; } // *************************************************************************** void CSBrickManager::init() { // Read the Bricks from the SheetMngr. const CSheetManager::TEntitySheetMap &sheetMap = SheetMngr.getSheets(); _BrickVector.clear(); _BrickVector.reserve(16 * 1024); uint32 shtype = CSheetId::typeFromFileExtension("sbrick"); for (CSheetManager::TEntitySheetMap::const_iterator it(sheetMap.begin()), end(sheetMap.end()); it != end; ++it) { // it's a brick? CSBrickSheet *br = dynamic_cast(it->second.EntitySheet); // TODO: Avoid dynamic_cast, depend on getSheetType if (br) { // ok, add it uint32 shid = it->first.getShortId(); nlassert(shtype == it->first.getSheetType()); if (shid >= _BrickVector.size()) _BrickVector.resize(shid + 1); _BrickVector[shid] = br; } } // Process Bricks if (_BrickVector.empty()) return; // Build the vector of family bit fields, and the vector of existing bricks for (std::vector::iterator itb(_BrickVector.begin()), endb(_BrickVector.end()); itb != endb; ++itb) { CSBrickSheet *brickSheet = *itb; if (!brickSheet) continue; // Resize our vectors if necessary if (brickSheet->BrickFamily >= (sint32)_NbFamily) { _SheetsByFamilies.resize(brickSheet->BrickFamily + 1); _FamiliesBits.resize(brickSheet->BrickFamily + 1, 0); _NbBricksPerFamily.resize(brickSheet->BrickFamily + 1, 0); _NbFamily = brickSheet->BrickFamily + 1; } } // Since _SheetsByFamilies is a vector of vector, avoid long reallocation by building it in 2 pass sint32 shidc = -1; for (std::vector::iterator itb(_BrickVector.begin()), endb(_BrickVector.end()); itb != endb; ++itb) { ++shidc; CSBrickSheet *brickSheet = *itb; if (!brickSheet) continue; // Resize our vectors if necessary // The index in familly must be decremented because the values start at 1 in the sheets if (brickSheet->IndexInFamily < 1) { nlwarning("CSBrickManager::CSBrickManager(): Reading file: %s: IndexInFamily==0 but should be >=1 - entry ignored", brickSheet->Id.toString().c_str()); continue; } if (_NbBricksPerFamily[brickSheet->BrickFamily] < (uint)(brickSheet->IndexInFamily)) { _SheetsByFamilies[brickSheet->BrickFamily].resize(brickSheet->IndexInFamily); _NbBricksPerFamily[brickSheet->BrickFamily] = brickSheet->IndexInFamily; } _SheetsByFamilies[brickSheet->BrickFamily][brickSheet->IndexInFamily - 1] = brickSheet->Id; } // check brick content for client. checkBricks(); // make root list makeRoots(); // Make interface bricks makeVisualBrickForSkill(); // compile brick properties compileBrickProperties(); // see getInterfaceRemoveBrick _InterfaceRemoveBrick.buildSheetId("big_remove.sbrick"); } // *************************************************************************** void CSBrickManager::initInGame() { CInterfaceManager* pIM = CInterfaceManager::getInstance(); // shortcut to the node char buf[100]; for(uint i=0;i<_FamiliesBits.size();i++) { //get the known brick entries in the database sprintf(buf,"SERVER:BRICK_FAMILY:%d:BRICKS",i); _FamiliesBits[i] = NLGUI::CDBManager::getInstance()->getDbProp(buf); } // Add a branch observer on brick family NLGUI::CDBManager::getInstance()->addBranchObserver( "SERVER:BRICK_FAMILY", &_BrickFamilyObs); } // *************************************************************************** void CSBrickManager::uninitInGame() { CInterfaceManager* pIM = CInterfaceManager::getInstance(); // remove shortcuts to the node for(uint i=0;i<_FamiliesBits.size();i++) { _FamiliesBits[i] = NULL; } // remove branch observer on brick family CCDBNodeBranch *branch= NLGUI::CDBManager::getInstance()->getDbBranch("SERVER:BRICK_FAMILY"); if(branch) branch->removeBranchObserver(&_BrickFamilyObs); } // *************************************************************************** CSBrickManager::~CSBrickManager() { } // *************************************************************************** CSheetId CSBrickManager::getBrickSheet(uint family, uint index) const { if (family >= _NbFamily) return NLMISC::CSheetId(0); if ( index >= _NbBricksPerFamily[family] ) return NLMISC::CSheetId(0); return _SheetsByFamilies[family][index]; } // *************************************************************************** sint64 CSBrickManager::getKnownBrickBitField(uint family) const { if (family < _NbFamily && _FamiliesBits[family]) return _FamiliesBits[family]->getValue64(); return 0; } // *************************************************************************** CCDBNodeLeaf* CSBrickManager::getKnownBrickBitFieldDB(uint family) const { if (family < _NbFamily && _FamiliesBits[family]) return _FamiliesBits[family]; return NULL; } // *************************************************************************** void CSBrickManager::makeRoots() { _Roots.clear(); for (std::vector::size_type ib = 0; ib < _BrickVector.size(); ++ib) { const CSBrickSheet *brickSheet = _BrickVector[ib]; if (!brickSheet) continue; // List only the Roots if (!brickSheet->isRoot()) continue; _Roots.push_back(brickSheet->Id); } } // *************************************************************************** const std::vector &CSBrickManager::getFamilyBricks(uint family) const { static std::vector empty; if (family >= _NbFamily) return empty; return _SheetsByFamilies[family]; } // *************************************************************************** void CSBrickManager::checkBricks() { for (std::vector::size_type ib = 0; ib < _BrickVector.size(); ++ib) { CSBrickSheet *brickSheet = _BrickVector[ib]; if (!brickSheet) continue; if (brickSheet->ParameterFamilies.size() > CDBGroupBuildPhrase::MaxParam) { nlwarning("The Sheet %s has too many parameters for Client Composition: %d/%d", brickSheet->Id.toString().c_str(), brickSheet->ParameterFamilies.size(), CDBGroupBuildPhrase::MaxParam); // reset them... don't crahs client, but won't work. brickSheet->ParameterFamilies.clear(); } } } // *************************************************************************** sint32 CSBrickManager::CBrickContainer::getSabrinaCost(NLMISC::CSheetId id) const { // get the true container CSBrickManager *pBM= CSBrickManager::getInstance(); CSBrickSheet *brick= pBM->getBrick(id); if(brick) { return brick->SabrinaCost; } else { return 0; } } // *************************************************************************** float CSBrickManager::CBrickContainer::getSabrinaRelativeCost(NLMISC::CSheetId id) const { // get the true container CSBrickManager *pBM= CSBrickManager::getInstance(); CSBrickSheet *brick= pBM->getBrick(id); if(brick) { return brick->SabrinaRelativeCost; } else { return 0.f; } } // *************************************************************************** sint32 CSBrickManager::CBrickContainer::getNumParameters(NLMISC::CSheetId id) const { CSBrickManager *pBM= CSBrickManager::getInstance(); CSBrickSheet *brick= pBM->getBrick(id); if(brick) { return (sint32)brick->ParameterFamilies.size(); } else { return 0; } } // *************************************************************************** BRICK_FAMILIES::TBrickFamily CSBrickManager::CBrickContainer::getBrickFamily(NLMISC::CSheetId id, uint& indexInFamily ) const { CSBrickManager *pBM= CSBrickManager::getInstance(); CSBrickSheet *brick= pBM->getBrick(id); if(brick) { indexInFamily = brick->IndexInFamily; return brick->BrickFamily; } else { return BRICK_FAMILIES::Unknown; } } // *************************************************************************** BRICK_TYPE::EBrickType CSBrickManager::CBrickContainer::getBrickType(NLMISC::CSheetId id) const { CSBrickManager *pBM= CSBrickManager::getInstance(); CSBrickSheet *brick= pBM->getBrick(id); if(brick) { return BRICK_FAMILIES::brickType(brick->BrickFamily); } else { return BRICK_TYPE::UNKNOWN; } } // *************************************************************************** TOOL_TYPE::TCraftingToolType CSBrickManager::CBrickContainer::getFaberPlanToolType(NLMISC::CSheetId id) const { CSBrickManager *pBM= CSBrickManager::getInstance(); CSBrickSheet *brick= pBM->getBrick(id); if(brick) { return brick->FaberPlan.ToolType; } else { return TOOL_TYPE::Unknown; } } // *************************************************************************** void CSBrickManager::makeVisualBrickForSkill() { // clear for (uint i = 0; i < SKILLS::NUM_SKILLS; ++i) { _VisualBrickForSkill[i] = CSheetId(); } // fill with interface bricks for (std::vector::size_type ib = 0; ib < _BrickVector.size(); ++ib) { const CSBrickSheet *brickSheet = _BrickVector[ib]; if (!brickSheet) continue; // List only bricks with family == BIF if (brickSheet->BrickFamily == BRICK_FAMILIES::BIF) { if (brickSheet->getSkill() < SKILLS::NUM_SKILLS) { _VisualBrickForSkill[brickSheet->getSkill()] = brickSheet->Id; } } } } // *************************************************************************** CSheetId CSBrickManager::getVisualBrickForSkill(SKILLS::ESkills s) { if(s::size_type ib = 0; ib < _BrickVector.size(); ++ib) { CSBrickSheet *brickSheet = _BrickVector[ib]; if (!brickSheet) continue; // For all properties of this brick, compile for (uint i = 0; i < brickSheet->Properties.size(); ++i) { CSBrickSheet::CProperty &prop = brickSheet->Properties[i]; string::size_type pos = prop.Text.find(':'); if (pos != string::npos) { string key = prop.Text.substr(0, pos); strlwr(key); string value = prop.Text.substr(pos + 1); // get key id. if (_BrickPropIdMap.find(key) == _BrickPropIdMap.end()) { // Inc before to leave 0 as "undefined" _BrickPropIdMap[key] = ++NumIds; } prop.PropId = _BrickPropIdMap[key]; fromString(value, prop.Value); pos = value.find(':'); if (pos != string::npos) fromString(value.substr(pos + 1), prop.Value2); } } } // Get usual PropIds HpPropId= getBrickPropId("hp"); SapPropId= getBrickPropId("sap"); StaPropId= getBrickPropId("sta"); StaWeightFactorId = getBrickPropId("sta_weight_factor"); FocusPropId= getBrickPropId("focus"); CastTimePropId= getBrickPropId("ma_casting_time"); RangePropId= getBrickPropId("ma_range"); // **** for all bricks, recompute localized text with formated version ucstring textTemp; textTemp.reserve(1000); for (std::vector::size_type ib = 0; ib < _BrickVector.size(); ++ib) { CSBrickSheet *brickSheet = _BrickVector[ib]; if (!brickSheet) continue; // Get the Brick texts ucstring texts[3]; texts[0]= STRING_MANAGER::CStringManagerClient::getSBrickLocalizedName(brickSheet->Id); texts[1]= STRING_MANAGER::CStringManagerClient::getSBrickLocalizedDescription(brickSheet->Id); texts[2]= STRING_MANAGER::CStringManagerClient::getSBrickLocalizedCompositionDescription(brickSheet->Id); // For alls texts, parse format for(uint i=0;i<3;i++) { ucstring &text= texts[i]; textTemp.erase(); // Parse the text uint textSize= (uint)text.size(); for(uint j=0;jProperties.size();p++) { const CSBrickSheet::CProperty &prop= brickSheet->Properties[p]; if(prop.PropId==propId) { if(paramId==0) value= brickSheet->Properties[p].Value; else { // must parse the initial text/ skip the identifier string::size_type pos= prop.Text.find(':'); uint id= 0; while(pos!=string::npos && idId, texts[0], texts[1], texts[2]); } } // *************************************************************************** uint CSBrickManager::getBrickPropId(const std::string &name) { std::map::const_iterator it; it= _BrickPropIdMap.find(name); if(it!=_BrickPropIdMap.end()) return it->second; return 0; } // *************************************************************************** NLMISC::CSheetId CSBrickManager::getInterfaceRemoveBrick() { return _InterfaceRemoveBrick; } // *************************************************************************** void CSBrickManager::filterKnownBricks(std::vector &bricks) { std::vector res; res.reserve(bricks.size()); // keep only known ones for(uint i=0;i_BrickLearnedCallbackSet.begin();it!=Owner->_BrickLearnedCallbackSet.end();it++) { (*it)->onBrickLearned(); } }