Added spawnCrystalItem
[ryzomcore.git] / ryzom / client / src / continent.cpp
blobac57607086bea74c44ca4274857046c0496a4af3
1 // Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
2 // Copyright (C) 2010-2019 Winch Gate Property Limited
3 //
4 // This source file has been modified by the following contributors:
5 // Copyright (C) 2020 Jan BOON (Kaetemi) <jan.boon@kaetemi.be>
6 //
7 // This program is free software: you can redistribute it and/or modify
8 // it under the terms of the GNU Affero General Public License as
9 // published by the Free Software Foundation, either version 3 of the
10 // License, or (at your option) any later version.
12 // This program is distributed in the hope that it will be useful,
13 // but WITHOUT ANY WARRANTY; without even the implied warranty of
14 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 // GNU Affero General Public License for more details.
17 // You should have received a copy of the GNU Affero General Public License
18 // along with this program. If not, see <http://www.gnu.org/licenses/>.
23 /////////////
24 // INCLUDE //
25 /////////////
26 #include "stdpch.h"
27 // Misc
28 #include "nel/misc/vectord.h"
29 #include "nel/misc/path.h"
30 #include "nel/misc/smart_ptr.h"
31 #include "nel/misc/progress_callback.h"
32 #include "nel/misc/async_file_manager.h"
33 #include "nel/misc/bitmap.h"
34 #include "nel/misc/polygon.h"
35 // 3D Interface.
36 #include "nel/3d/u_landscape.h"
37 #include "nel/3d/u_scene.h"
38 #include "nel/3d/u_driver.h"
39 #include "nel/3d/u_texture.h"
40 // Georges
41 #include "nel/georges/u_form.h"
42 #include "nel/georges/u_form_elm.h"
43 #include "nel/georges/u_form_loader.h"
44 // Gui
45 #include "nel/gui/lua_manager.h"
46 // Client
47 #include "global.h"
48 #include "continent.h"
49 #include "light_cycle_manager.h" // \todo GUIGUI : maybe change this when the time will be better managed.
50 #include "door_manager.h"
51 #include "client_cfg.h"
52 #include "ig_client.h"
53 #include "pacs_client.h"
54 #include "village.h"
55 #include "ig_callback.h"
56 #include "streamable_ig.h"
57 #include "weather_manager_client.h"
58 #include "weather.h"
59 #include "entities.h"
60 #include "fog_map.h"
61 #include "demo.h"
62 #include "ingame_database_manager.h"
63 #include "sky_render.h"
64 #include "fix_season_data.h"
65 #include "ig_season_callback.h"
66 #include "user_entity.h"
67 #include "micro_life_manager.h"
68 #include "zone_util.h"
69 #include "mesh_camera_col_manager.h"
70 #include "misc.h"
71 #include "sheet_manager.h"
72 #include "continent_manager.h"
73 #include "connection.h"
74 #include "water_env_map_rdr.h"
75 #include "input.h"
76 #include "bg_downloader_access.h"
78 // Game Share
79 #include "game_share/georges_helper.h"
80 #include "game_share/season_file_ext.h"
82 // R2 Share
83 #include "game_share/scenario_entry_points.h"
86 #include "r2/editor.h"
88 // std
89 #include <vector>
90 #include <string>
91 #include <memory>
92 #include <algorithm>
95 ///////////
96 // USING //
97 ///////////
98 using namespace NLMISC;
99 using namespace NL3D;
100 using namespace std;
101 using namespace NLGEORGES;
104 ////////////
105 // EXTERN //
106 ////////////
107 extern ULandscape *Landscape;
108 extern UScene *Scene;
109 extern UScene *SceneRoot;
110 extern UInstanceGroup *BackgroundIG;
111 extern UDriver *Driver;
112 extern UVisualCollisionManager *CollisionManager;
113 extern bool InitCloudScape;
114 extern bool FirstFrame;
115 extern CContinentManager ContinentMngr;
116 extern CEntityManager EntitiesMngr;
118 //-----------------------------------------------
119 // CUserLandMark
120 //-----------------------------------------------
122 NLMISC::CRGBA CUserLandMark::getColor () const
124 return _LandMarksColor[Type];
128 //-----------------------------------------------
129 // CFogOfWar
130 //-----------------------------------------------
132 //===================================================================================
133 CFogOfWar::~CFogOfWar()
135 if (Tx != NULL)
136 Driver->deleteTextureMem(Tx);
139 //===================================================================================
140 uint8 *CFogOfWar::getData()
142 return (Tx != NULL) ? Tx->getPointer() : NULL;
145 //===================================================================================
146 sint16 CFogOfWar::getRealWidth()
148 return (Tx != NULL) ? (sint16)Tx->getImageWidth() : 0;
151 //===================================================================================
152 sint16 CFogOfWar::getRealHeight()
154 return (Tx != NULL) ? (sint16)Tx->getImageHeight() : 0;
157 //===================================================================================
158 bool CFogOfWar::createData(sint16 w, sint16 h)
160 if (Tx != NULL)
161 Driver->deleteTextureMem(Tx);
163 MapWidth = w;
164 MapHeight = h;
165 uint32 WReal = NLMISC::raiseToNextPowerOf2(MapWidth);
166 uint32 HReal = NLMISC::raiseToNextPowerOf2(MapHeight);
168 Tx = Driver->createTextureMem(WReal, HReal, CBitmap::Alpha);
169 if (Tx == NULL)
170 return false;
172 Tx->setWrapS(NL3D::UTexture::Clamp);
173 Tx->setWrapT(NL3D::UTexture::Clamp);
175 uint8 *pData = Tx->getPointer();
176 // Set the texture to black (not discovered)
177 for (uint32 j = 0; j < WReal*HReal; ++j)
178 pData[j] = 0;
180 // Upload it
181 Tx->touch();
182 return true;
185 //===================================================================================
186 void CFogOfWar::explored(sint16 /* mapPosX */, sint16 /* mapPosY */)
188 // TODO trap : do something better than upload the whole texture
189 if (Tx != NULL)
190 Tx->touch();
194 //===================================================================================
195 void CFogOfWar::load(const std::string &/* contName */)
197 /* // Temporary comment : see this when server ready
198 string fileName = "save/fow_" + contName + "_" + PlayerSelectedFileName + ".ibmp";
199 // Try to load the file corresponding
200 CIFile inFile;
201 if (inFile.open(fileName))
203 serial(inFile);
205 else
207 // If the file do not exist create a new texture
208 // Look in all the maps to know the texture map size and apply a ratio to this size
209 CWorldSheet *pWS = dynamic_cast<CWorldSheet*>(SheetMngr.get(CSheetId("ryzom.world")));
210 if (pWS == NULL) return;
211 for (uint32 i = 0; i < pWS->Maps.size(); ++i)
213 SMap &rM = pWS->Maps[i];
214 if (strncmp(rM.Name.c_str(), "continent", 9) == 0)
216 if (stricmp(rM.ContinentName.c_str(), contName.c_str()) == 0)
218 // get the size of the bitmap
219 uint32 nWidth = 0, nHeight = 0;
220 string path = CPath::lookup(rM.BitmapName, false);
221 if (!path.empty())
222 CBitmap::loadSize(path, nWidth, nHeight);
223 // create the texture fow
224 if ((nWidth != 0) && (nHeight != 0))
226 MapWidth = (sint16)nWidth / 4;
227 MapHeight = (sint16)nHeight / 4;
228 createData(MapWidth, MapHeight);
230 MinX = rM.MinX;
231 MinY = rM.MinY;
232 MaxX = rM.MaxX;
233 MaxY = rM.MaxY;
234 return;
242 //===================================================================================
243 void CFogOfWar::save(const std::string &/* contName */)
245 /* // Temporary comment : see this when server ready
246 string fileName = "save/fow_" + contName + "_" + PlayerSelectedFileName + ".ibmp";
247 // Try to load the file corresponding
248 COFile inFile;
249 if (inFile.open(fileName))
250 serial(inFile);
255 /////////////
256 // METHODS //
257 /////////////
258 //-----------------------------------------------
259 // CContinent :
260 // Constructor.
261 //-----------------------------------------------
262 CContinent::CContinent()
264 // Newbie = false;
265 }// CContinent //
268 class CChangeTaskPriority : public CTaskManager::IChangeTaskPriority
270 virtual float getTaskPriority(const IRunnable &runnable)
272 const NLMISC::IRunnablePos *_3dRunnable = dynamic_cast<const NLMISC::IRunnablePos*>(&runnable);
273 if (_3dRunnable)
275 return (_3dRunnable->Position - UserEntity->pos()).norm();
277 return 0;
279 } ChangeTaskPriority;
284 //===================================================================================
285 void CContinent::setup()
287 if (SheetName.empty())
289 nlwarning("SheetName not set : cannot setup continent");
290 return;
293 CEntitySheet *pES = SheetMngr.get(CSheetId (SheetName));
294 CContinentSheet *pCS = NULL;
295 if (pES->type() == CEntitySheet::CONTINENT)
296 pCS = (CContinentSheet*)pES;
298 if (pCS == NULL)
300 nlwarning("Bad type for continent form %s.", SheetName.c_str());
301 return;
304 // copy basics parameters of sheet
305 CContinentParameters::operator = (pCS->Continent);
307 // setup weather functions
308 for(uint k = 0; k < EGSPD::CSeason::Invalid; ++k)
310 WeatherFunction[k].buildFromSheet(pCS->WeatherFunction[k], WeatherManager);
314 nlinfo("Seasons for continent %s", SheetName.c_str());
315 nlinfo("==================================");
316 for(uint k = 0; k < EGSPD::CSeason::Invalid; ++k)
318 nlinfo("Season = %s, num setups = %d", EGSPD::CSeason::toString((EGSPD::CSeason::TSeason) k).c_str(), (int) WeatherFunction[k].getNumWeatherSetups());
319 for(uint l = 0; l < WeatherFunction[k].getNumWeatherSetups(); ++l)
321 if (WeatherFunction[k].getWeatherSetup(l))
323 nlinfo("Setup %d for season %s is %s", (int) l, EGSPD::CSeason::toString((EGSPD::CSeason::TSeason) k).c_str(), NLMISC::CStringMapper::unmap(WeatherFunction[k].getWeatherSetup(l)->SetupName).c_str());
325 else
327 nlinfo("Setup %d not found", (int) l);
333 uint k;
334 // setup villages
335 for(k = 0; k < pCS->Villages.size(); ++k)
337 CVillage *village = new CVillage;
338 if (village->setupFromSheet(Scene, pCS->Villages[k], &IGLoaded))
339 _Villages.add(village);
340 else
341 delete village;
345 // Setup outpost
346 _Outposts.clear();
347 for(k = 0; k < pCS->Continent.ZCList.size(); ++k)
349 // must add the ruins in 3D?
350 bool mustAddRuins= pCS->Continent.ZCList[k].EnableRuins;
352 // add a village for ruins display?
353 CVillage *village= NULL;
354 if(mustAddRuins)
356 village = new CVillage;
357 if (village->setupOutpost(Scene, pCS->Continent.ZCList[k], k, &IGLoaded))
358 _Villages.add(village);
359 else
361 delete village;
362 village= NULL;
366 // add an outpost
367 COutpost outpost;
368 if (outpost.setupOutpost(pCS->Continent.ZCList[k], k, village))
369 _Outposts.push_back(outpost);
372 // setup fog of war
373 FoW.load(Name);
378 //-----------------------------------------------
379 class CPCBank : public IProgressCallback
381 virtual void progress (float /* progressValue */) {}
383 CPCBank PCBank;
385 static uint getNumZones()
387 if (Landscape == NULL) return 0;
388 std::vector<std::string> zoneLoaded;
389 Landscape->getAllZoneLoaded(zoneLoaded);
390 return (uint)zoneLoaded.size();
393 static uint16 getZoneIdFromName(const string &zoneName)
395 uint16 zoneId = 0;
396 CVector2f pos;
397 if (getPosFromZoneName(zoneName, pos))
399 uint x = (uint)pos.x / 160;
400 uint y = -(uint)pos.y / 160;
401 zoneId = (x&255) + (y<<8);
403 else
405 nlinfo("no zone...");
407 return zoneId;
410 //-----------------------------------------------
411 // select :
412 // Update global parameters like the texture for the micro veget.
413 //-----------------------------------------------
414 void CContinent::select(const CVectorD &pos, NLMISC::IProgressCallback &progress, bool complete, bool unhibernate, EGSPD::CSeason::TSeason season)
416 #ifdef RYZOM_BG_DOWNLOADER
417 pauseBGDownloader();
418 #endif
419 CNiceInputAuto niceInputs;
420 // Season has changed ?
421 Season = season;
423 if (Landscape) Landscape->setTileColor(TileColorMono[Season], TileColorFactor[Season]);
425 // First frame
426 FirstFrame = true;
429 H_AUTO(InitRZWorldLanscapeShow)
430 // If continent is an indoor, hde the landscape
431 if (Landscape)
433 if (Indoor)
434 Landscape->hide();
435 else
436 Landscape->show();
440 bool toto = false;
441 if(complete)
443 // Progress bar
444 progress.progress (0);
445 progress.pushCropedValues (0, 1.f/3.f);
448 H_AUTO(InitRZWorldIndoor)
449 if (!ClientCfg.Light)
451 if (Landscape)
453 Landscape->removeAllZones();
454 toto= true;
455 // If continent is an indoor, hde the landscape
456 if (!Indoor)
458 // Change the MicroVeget Texture.
459 string microVegetSeason = MicroVeget;
460 SeasonPostfixTextureFilename (microVegetSeason, Season);
461 Landscape->loadVegetableTexture (microVegetSeason);
463 // Change the Material for static and dynamic PointLights
464 Landscape->setPointLightDiffuseMaterial(LandscapePointLightMaterial);
469 // Initialize Landscape IG Callbacks.
471 H_AUTO(InitRZWorldInitCallbacks)
472 initLandscapeIGCallbacks();
473 nlassert(IGCallbacks); // forgot to call ::initLandscapeIGCallbacks()
474 if (!ClientCfg.Light)
476 // Add weather as an observer (to know when an ig is added, and to set its lightmaps color accordingly)
477 registerObserver(&LightCycleManager);
478 IGCallbacks->registerObserver(&LightCycleManager);
480 // register the LightCycleManager as an observer to Objects that may create IGs
481 IGSeasonCallback.Season = Season;
482 registerObserver(&IGSeasonCallback);
483 IGCallbacks->registerObserver(&IGSeasonCallback);
485 // register the Camera collision manager
486 registerObserver(&MeshCameraColManager);
488 if (!Indoor)
490 // Change LOD texture
493 // Get the CoarseMeshMap for this season
494 string seasonname = CoarseMeshMap;
495 SeasonPostfixTextureFilename (seasonname, Season);
497 // Set the texture for the coarse mesh manager
498 Scene->setCoarseMeshManagerTexture (CPath::lookup(seasonname).c_str());
500 catch (const Exception &e)
502 nlwarning (e.what());
506 // Register door handling (when an ig is added check all shapes for doors and add to the door manager)
507 registerObserver(&IGDoorCallback);
511 H_AUTO(InitRZWorldPacs)
513 releasePACS();
514 // Init PACS
515 std::string pacsRBankPath = CPath::lookup(PacsRBank, false);
516 std::string pacsGRPath = CPath::lookup(PacsGR, false);
517 if(!pacsRBankPath.empty() && !pacsGRPath.empty())
518 initPACS(pacsRBankPath.c_str(), pacsGRPath.c_str(), progress);
519 else
520 initPACS(0, 0, progress);
521 // Set the move container
522 if (IGCallbacks) IGCallbacks->setMoveContainer(PACS);
526 H_AUTO(InitRZWorldBuildings)
527 // No landscape IG in indoors
528 if (!Indoor)
530 if (!ClientCfg.Light && Landscape)
532 // Init ig
535 LandscapeIGManager.initIG (Scene, CPath::lookup(LandscapeIG).c_str(), Driver, Season, &progress);
537 catch (const Exception &e)
539 nlwarning (e.what());
543 // Get IGs for the continent with their name.
544 vector<pair<UInstanceGroup *, string> > igsWithNames;
545 LandscapeIGManager.getAllIGWithNames(igsWithNames);
547 // Associate IGs with the ZC number or -1 if there is no ZC.
548 for(uint i = 0; i<igsWithNames.size(); ++i)
550 string igZone = toLowerAscii(CFile::getFilenameWithoutExtension(igsWithNames[i].second));
552 // Search for the IG name in the ZC list to associate.
553 for(uint j = 0; j<ZCList.size(); ++j)
555 // Get the outpost ZC
556 COutpost *outpost = getOutpost (j);
557 if (outpost)
559 // If name matching -> this zone should be a ZC.
560 string outpostZone = toLowerAscii(CFile::getFilenameWithoutExtension(ZCList[j].Name));
561 if(igZone == outpostZone)
563 nlctassert(RZ_MAX_BUILDING_PER_OUTPOST==4);
564 // Check there is all information in the IG for a ZC and initialize.
565 bool zcOK[RZ_MAX_BUILDING_PER_OUTPOST+1] = {false, false, false, false, false};
566 UInstanceGroup *ig = igsWithNames[i].first;
567 uint k;
568 for(k=0; k<ig->getNumInstance(); ++k)
570 if(ig->getInstanceName(k) == "flag_zc")
572 // todo hulud handle outpost flag
573 zcOK[0] = true;
576 // For each building
577 uint b;
578 for (b=0; b<RZ_MAX_BUILDING_PER_OUTPOST; b++)
580 // Good one ?
581 if (ig->getInstanceName(k) == ("bat_zc_0"+toString(b+1)))
583 // Set the position
584 CVector pos = ig->getInstancePos(k);
585 CQuat rot = ig->getInstanceRot(k);
586 outpost->setBuildingPosition (b, rot, pos);
587 zcOK[b+1] = true;
592 // There are not enough information in the IG for ZC.
593 for (k=0; k<RZ_MAX_BUILDING_PER_OUTPOST+1; k++)
594 if (!zcOK[k])
595 break;
597 // Failed ?
598 if (k != RZ_MAX_BUILDING_PER_OUTPOST+1)
599 nlwarning("CContinent::select : Zone '%s' should be a ZC but some doomy are missing.", igsWithNames[i].second.c_str());
600 // Outpost found, and setuped. stop
601 break;
605 // Add IGs with the Num ZC associated in the callback system.
606 if (IGCallbacks) IGCallbacks->addIG(igsWithNames[i].first);
609 // Register outpost collisions and Display
610 initOutpost();
614 if (!ClientCfg.Light)
616 // Progress bar
617 progress.popCropedValues ();
618 progress.progress (1.f/3.f);
619 progress.pushCropedValues (1.f/3.f, 2.f/3.f);
622 H_AUTO(InitRZWorldTileBank)
623 // Select the tile bank
624 if (Landscape)
626 if (!Indoor)
628 string farBankSeason = FarBank;
629 SeasonPostfixTextureFilename (farBankSeason, Season);
630 if (!SmallBank.empty() && !farBankSeason.empty())
632 std::string smallBankPath = CPath::lookup(SmallBank, false);
633 std::string farBankSeasonPath = CPath::lookup(farBankSeason, false);
634 if (!smallBankPath.empty() && !farBankSeasonPath.empty())
636 Landscape->loadBankFiles (smallBankPath, farBankSeasonPath);
638 // Postfix tiles and vegetset by the season string
639 Landscape->postfixTileFilename (CSeasonFileExt::getExtension (Season));
640 Landscape->postfixTileVegetableDesc (CSeasonFileExt::getExtension (Season));
642 // Flush the tiles
643 Landscape->flushTiles (progress);
649 // Not an indoor ?
650 if (!Indoor)
653 H_AUTO(InitRZWorldSky)
654 // load the sky if any
655 initSky();
656 // build the canopee
659 nlassert(CurrSeason < EGSPD::CSeason::Invalid);
660 if (!CanopyIGfileName[CurrSeason].empty() && CurrSeason != EGSPD::CSeason::Invalid)
662 BackgroundIG = UInstanceGroup::createInstanceGroup(CanopyIGfileName[CurrSeason]);
664 else
666 BackgroundIG = UInstanceGroup::createInstanceGroup(BackgroundIGName);
669 catch (const Exception &e)
671 nlwarning (e.what());
673 if(BackgroundIG)
675 if (SceneRoot)
677 // Add the IG to the scene
678 BackgroundIG->addToScene(*SceneRoot, Driver);
680 // See for which objects distance should be overriden (object which use a .PLANT sheet)
681 uint numInstances = BackgroundIG->getNumInstance();
682 uint k;
683 for(k = 0; k < numInstances; ++k)
685 // Select the texture function of the season
686 UInstance instance = BackgroundIG->getInstance (k);
687 if (!instance.empty())
689 instance.selectTextureSet (Season);
698 // Progress bar
699 progress.popCropedValues ();
700 progress.progress (2.f/3.f);
701 progress.pushCropedValues (2.f/3.f, 1);
702 }// complete
704 if (complete || unhibernate)
706 H_AUTO(InitRZWorldLightCycleManager)
707 // Touch light setup
708 LightCycleManager.touch();
711 if (complete)
713 H_AUTO(InitRZWorldWaterMap)
714 // init water map
715 initWaterMap();
718 const R2::CScenarioEntryPoints::CCompleteIsland *completeIsland = NULL;
720 progress.pushCropedValues (0, 3.f/4.f);
722 if (!ClientCfg.Light)
724 if (!Indoor)
726 if (Landscape)
728 H_AUTO(InitRZWorldLandscapeIGManager)
729 // If any zone already loaded into the landscape, must load them to the IGManager.
730 // Yoyo: this is important for teleportation near a current position, since not so many landscape zones
731 // will be removed/added.
732 // Hence, a solution is to reload all igs.
733 vector<string> zonesLoaded;
734 Landscape->getAllZoneLoaded(zonesLoaded);
735 // and try to load the ones which fit the IGManager
736 vector<NL3D::UInstanceGroup *> igAdded;
737 LandscapeIGManager.loadArrayZoneIG(zonesLoaded, &igAdded);
739 // Refresh zone to load/remove.
740 vector<string> zonesAdded;
741 vector<string> zonesRemoved;
742 completeIsland = R2::CScenarioEntryPoints::getInstance().getCompleteIslandFromCoords(CVector2f((float) UserEntity->pos().x, (float) UserEntity->pos().y));
743 Landscape->refreshAllZonesAround(pos, ClientCfg.Vision + ExtraZoneLoadingVision, zonesAdded, zonesRemoved, progress, completeIsland ? &(completeIsland->ZoneIDs) : NULL);
745 for (uint i = 0; i < zonesRemoved.size(); i++)
746 EntitiesMngr.removeInstancesInIgZone(getZoneIdFromName(zonesRemoved[i]));
748 for (uint i = 0; i < zonesAdded.size(); i++)
750 string luaScriptName = CPath::lookup(zonesAdded[i]+".lua", false);
752 if (!luaScriptName.empty())
754 CIFile in;
755 if (in.open(luaScriptName))
758 string luaScript;
759 if (in.readAll(luaScript))
761 CLuaManager::getInstance().executeLuaScript(luaScript, true);
762 nlinfo("loading %s", luaScriptName.c_str());
766 else
768 nlinfo("file not found %s", luaScriptName.c_str());
772 LandscapeIGManager.unloadArrayZoneIG(zonesRemoved);
773 LandscapeIGManager.loadArrayZoneIG(zonesAdded, &igAdded);
778 progress.popCropedValues ();
779 progress.progress (3.f/4.f);
780 progress.pushCropedValues (3.f/4.f, 1);
782 // Refresh PACS
783 if(GR)
785 H_AUTO(InitRZWorldRefreshPacs)
786 GR->refreshLrAroundNow(pos, LRRefeshRadius);
789 // Refresh streamable object (villages ..)
790 if (!ClientCfg.Light)
792 if (ClientCfg.VillagesEnabled)
794 H_AUTO(InitRZWorldForceUpdateStreamable)
795 forceUpdateStreamable(pos, progress);
799 progress.popCropedValues ();
802 // Execute only when you change the targeted continent is not the current one.
803 if(complete || unhibernate)
805 // Continent changed -> update entities.
806 EntitiesMngr.changeContinent();
807 R2::getEditor().onContinentChanged();
809 // Ask to reinit cloudscape
810 InitCloudScape = true;
812 // Set the entity sun contribution parameters
813 CollisionManager->setSunContributionPower (EntitySunContributionPower, EntitySunContributionMaxThreshold);
816 if(complete)
818 reloadFogMap();
819 // temp for debug : create visualisation of current pacs primitives
820 // createInstancesFromMoveContainer(Scene, IGCallbacks->getMoveContainer(), NULL);
822 // Progress bar
823 progress.popCropedValues ();
826 if (complete || unhibernate)
828 if (!ClientCfg.Light && ClientCfg.LandscapeEnabled && ClientCfg.MicroLifeEnabled)
830 H_AUTO(InitRZWorldLoadMicroLife)
831 loadMicroLife();
834 // Register callback on streaming
835 CAsyncFileManager::getInstance().registerTaskPriorityCallback(&ChangeTaskPriority);
836 }// select //
838 //=========================================================================
839 void CContinent::reloadFogMap()
841 const R2::CScenarioEntryPoints::CCompleteIsland *completeIsland = R2::CScenarioEntryPoints::getInstance().getCompleteIslandFromCoords(CVector2f((float) UserEntity->pos().x, (float) UserEntity->pos().y));
842 if (completeIsland)
844 std::string islandName = toLowerAscii(completeIsland->Island);
845 std::string::size_type lastPos = islandName.find_last_of("_");
846 if (lastPos != std::string::npos)
848 islandName = islandName.substr(lastPos + 1);
850 // special fogmaps are use for islands
851 CFogMapBuild fmb;
852 enum TMapType { Day = 0, Night, Dusk, Distance, Depth, NoPrecipitation, NumMap };
853 fmb.Map[CFogMapBuild::Day] = "fog_" + islandName + "_day.tga";
854 fmb.Map[CFogMapBuild::Night] = "fog_" + islandName + "_night.tga";
855 fmb.Map[CFogMapBuild::Dusk] = "fog_" + islandName + "_day.tga";
856 fmb.Map[CFogMapBuild::Distance] = "fog_" + islandName + "_dist.tga";
857 fmb.Map[CFogMapBuild::Depth] = "fog_" + islandName + "_depth.tga";
858 fmb.Map[CFogMapBuild::NoPrecipitation] = islandName + "_no_rain.tga";
859 // there is an outtline of 50%, take it in account
860 float deltaX = 0.25f * (completeIsland->XMax - completeIsland->XMin);
861 float deltaY = 0.25f * (completeIsland->XMax - completeIsland->XMin);
862 FogMap.init(fmb,
863 CVector((float) completeIsland->XMin + deltaX, (float) completeIsland->YMin + deltaY, 0.f),
864 CVector((float) completeIsland->XMax - deltaX, (float) completeIsland->YMax - deltaY, 0.f)
866 bool found = false;
867 for(uint k = 0; k < CFogMapBuild::NumMap && !found; ++k)
869 if (FogMap.getMap((CFogMap::TMapType) k).getWidth() != 0) found = true;
871 // as a fallback, assume that new fog maps have not been installed, so fallback on the old ones ...
872 if (!found)
874 FogMap.init(FogMapBuild);
877 else
879 FogMap.init(FogMapBuild);
881 #ifdef NL_DEBUG
882 nlinfo("fog map : (minX, minY) = (%.1f, %.1f), (maxX, maxY) = (%.1f, %.1f)", FogMap.getMinCoord().x, FogMap.getMinCoord().y, FogMap.getMaxCoord().x, FogMap.getMaxCoord().y);
883 #endif
886 //=========================================================================
887 bool CContinent::getCorners(NLMISC::CVector2f &cornerMin, NLMISC::CVector2f &cornerMax) const
889 if (!getPosFromZoneName(ZoneMin, cornerMin))
891 nlwarning("Can't retrieve continent min corner");
892 return false;
894 if (!getPosFromZoneName(ZoneMax, cornerMax))
896 nlwarning("Can't retrieve continent max corner");
897 return false;
899 if(cornerMin.x > cornerMax.x) std::swap(cornerMin.x, cornerMax.x);
900 if(cornerMin.y > cornerMax.y) std::swap(cornerMin.y, cornerMax.y);
901 cornerMax.x += 160.f;
902 cornerMax.y += 160.f;
903 return true;
906 //=========================================================================
907 void CContinent::initWaterMap()
909 NLMISC::CVector2f cornerMin, cornerMax;
910 if (!getCorners(cornerMin, cornerMax)) return;
911 WaterMap.init(cornerMin, cornerMax);
914 //=========================================================================
915 void CContinent::loadMicroLife()
917 CMicroLifeManager::getInstance().release();
918 NLMISC::CVector2f cornerMin, cornerMax;
919 if (!getCorners(cornerMin, cornerMax)) return;
920 CMicroLifeManager::getInstance().init(cornerMin, cornerMax);
921 // register all micro life primitives
922 CMicroLifeManager::getInstance().build(MicroLifeZones);
925 //=========================================================================
926 bool CContinent::isLoadingforced(const NLMISC::CVector &pos) const
928 return _Villages.needCompleteLoading(pos);
932 //=========================================================================
933 void CContinent::updateStreamable(const NLMISC::CVector &pos)
935 _Villages.update(pos);
938 //=========================================================================
939 void CContinent::forceUpdateStreamable(const NLMISC::CVector &playerPos, NLMISC::IProgressCallback &progress)
941 _Villages.forceUpdate(playerPos, progress);
944 //=========================================================================
945 void CContinent::removeVillages()
947 _Villages.removeAll();
950 //=========================================================================
951 void CContinent::unselect()
953 // remove envmap
954 if (SceneRoot && !WaterEnvMapCanopyCam.empty())
956 SceneRoot->deleteCamera(WaterEnvMapCanopyCam);
958 WaterEnvMapCanopyCam.detach();
959 if (CurrentSky.getScene() && !WaterEnvMapSkyCam.empty())
961 CurrentSky.getScene()->deleteCamera(WaterEnvMapSkyCam);
963 WaterEnvMapSkyCam.detach();
965 if (!Indoor)
967 releaseSky();
969 // Setup the Root scene.
970 if (BackgroundIG)
972 nlassert(SceneRoot);
973 BackgroundIG->removeFromScene (*SceneRoot);
974 SceneRoot->deleteInstanceGroup (BackgroundIG);
975 BackgroundIG = NULL;
979 // Remove outpost (collisions etc...)
980 removeOutpost();
982 // Unregister callback on streaming
983 CAsyncFileManager::getInstance().registerTaskPriorityCallback(NULL);
985 // Remove the primitive for all entitites (new PACS coming soon and need new primitives).
986 EntitiesMngr.removeCollision();
988 // Remove the instances (shapes).
989 EntitiesMngr.removeInstances();
991 // release collision primitives
992 if (IGCallbacks)
994 IGCallbacks->resetContainer();
997 // Unload villages
998 _Villages.forceUnload();
1000 // Release PACS
1001 releasePACS ();
1003 // Initialize Landscape IG Callbacks.
1004 releaseLandscapeIGCallbacks();
1006 // Release the IG manager only if where are not in indoor
1007 if (!Indoor)
1009 // Release the IG manager
1010 if (Landscape) LandscapeIGManager.reset ();
1013 // release fog maps
1014 FogMap.release();
1016 // Remove weather as an observer
1017 removeObserver(&LightCycleManager);
1019 // Remove season observer
1020 removeObserver(&IGSeasonCallback);
1022 // Remove Camera collision observer
1023 removeObserver(&MeshCameraColManager);
1025 // Remove door management observers
1026 removeObserver(&IGDoorCallback);
1028 // Release water map
1029 WaterMap.release();
1031 // Release water envmap
1032 #ifdef USE_WATER_ENV_MAP
1033 Driver->deleteWaterEnvMap(WaterEnvMap);
1034 WaterEnvMap = NULL;
1035 #endif
1038 //=========================================================================
1039 /** Override current fog by the fog in the weather system. This is used when there is bad weather or precipitations
1040 * \param dest the fog state to modify
1041 * \param fogIndex the fog to modify (0 for main, 1 for canopy)
1043 static void overrideFog(CFogState &dest, TFogType fogType, float dayNight, float duskRatio, CLightCycleManager::TLightState lightState, bool /* overrideDist */, bool overideByWeatherFogDist = true)
1045 nlassert(fogType < NumFogType); // For now, 2 kinds of fogs : main & canopy
1046 if (!dest.FogEnabled) return;
1047 const CWeatherState &ws = WeatherManager.getCurrWeatherState();
1048 if (ws.FogRatio == 0.f) return; // no need to override fog
1049 // override the fog by the fog indicated in the WeatherManager
1050 NLMISC::CRGBA fogColor;
1051 // if there's a new style sky with a fog bitmap *(giving fog depending on hour & weather, use it)
1052 if (ContinentMngr.cur() && ContinentMngr.cur()->CurrentSky.getScene() != NULL && ContinentMngr.cur()->CurrentSky.hasFogColor())
1054 fogColor = ContinentMngr.cur()->CurrentSky.computeFogColor(CClientDate(RT.getRyzomDay(), (float) DayNightCycleHour), WeatherManager.getWeatherValue());
1056 else
1058 switch(lightState)
1060 case CLightCycleManager::DayToNight:
1062 if (dayNight < duskRatio)
1064 float blendFactor = duskRatio != 0 ? dayNight / duskRatio : 0.f;
1065 fogColor.blendFromui(ws.FogColorDay, ws.FogColorDusk, (uint) (blendFactor * 256.f));
1067 else
1069 float blendFactor = duskRatio != 1 ? (dayNight - duskRatio) / (1.f - duskRatio) : 0.f;
1070 fogColor.blendFromui(ws.FogColorDusk, ws.FogColorNight, (uint) (blendFactor * 256.f));
1073 break;
1074 default: // no dusk transition
1075 fogColor.blendFromui(ws.FogColorDay, ws.FogColorNight, (uint) (dayNight * 256.f));
1076 break;
1079 float finalFogAlpha = (float)dest.FogColor.A * 256.f / 255.f;
1080 dest.FogColor.blendFromui(dest.FogColor, fogColor, (uint) ((256.f-finalFogAlpha) * ws.FogRatio));
1081 if (overideByWeatherFogDist)
1083 dest.FogEndDist = std::min (ws.FogFar[fogType], dest.FogEndDist);
1084 dest.FogStartDist = std::min (ws.FogNear[fogType], dest.FogStartDist);
1088 //=========================================================================
1089 /** Adapt fog dist to the viewport
1091 static inline void makeRangedFog(CFogState &result, UScene &scene)
1093 if (result.FogEndDist != 0.f)
1095 float farPlaneDist = scene.getCam().getFrustum().Far;
1096 float newEndFogDist = std::min(result.FogEndDist, farPlaneDist);
1097 result.FogStartDist = result.FogStartDist * newEndFogDist / result.FogEndDist;
1098 result.FogEndDist = newEndFogDist;
1099 result.FogEnabled = true;
1101 else
1103 result.FogEnabled = false;
1107 //=========================================================================
1108 void CContinent::getFogState(TFogType fogType, float dayNight, float duskRatio, CLightCycleManager::TLightState lightState, const NLMISC::CVectorD &pos, CFogState &result, bool overideByWeatherFogDist /*= true*/)
1110 if (FogStart == FogEnd)
1112 result.FogEnabled = false;
1113 return;
1115 switch(fogType)
1117 case MainFog: FogMap.getFogParams(FogStart, FogEnd, (float) pos.x, (float) pos.y, dayNight, duskRatio, lightState, result); break;
1119 case CanopyFog: FogMap.getFogParams(RootFogStart, RootFogEnd, (float) pos.x, (float) pos.y, dayNight, duskRatio, lightState, result); break;
1120 default:
1121 nlstop;
1122 break;
1124 // Override fog because of weather.
1125 overrideFog(result, fogType, dayNight, duskRatio, lightState, overideByWeatherFogDist);
1126 // normalize fog dist to view frustum
1128 switch(fogType)
1130 case MainFog: if (Scene) makeRangedFog(result, *Scene); break;
1131 case CanopyFog: if (SceneRoot) makeRangedFog(result, *SceneRoot); break;
1135 if (fogType == MainFog)
1137 result.FogStartDist *= (ClientCfg.Vision / ClientCfg.Vision_max);
1138 result.FogEndDist *= (ClientCfg.Vision / ClientCfg.Vision_max);
1144 //=========================================================================
1145 bool CContinent::enumIGs(IIGEnum *callback)
1147 uint numEntities = _Villages.getNumEntities();
1148 bool continueEnum = true;
1149 for(uint k = 0; k < numEntities && continueEnum; ++k)
1151 CVillage *v = dynamic_cast<CVillage *>(_Villages.getEntity(k));
1152 if (v) // well this should always be the case, but it may change in the future
1154 continueEnum = v->enumIGs(callback);
1157 return continueEnum;
1160 //=========================================================================
1161 void CContinent::registerObserver(IIGObserver *obs)
1163 uint numEntities = _Villages.getNumEntities();
1164 for(uint k = 0; k < numEntities; ++k)
1166 CVillage *v = dynamic_cast<CVillage *>(_Villages.getEntity(k));
1167 if (v) // well this should always be the case, but it may change in the future
1169 v->registerObserver(obs);
1175 //=========================================================================
1176 void CContinent::removeObserver(IIGObserver *obs)
1178 uint numEntities = _Villages.getNumEntities();
1179 for(uint k = 0; k < numEntities; ++k)
1181 CVillage *v = dynamic_cast<CVillage *>(_Villages.getEntity(k));
1182 if (v) // well this should always be the case, but it may change in the future
1184 v->removeObserver(obs);
1190 //=========================================================================
1191 bool CContinent::isObserver(IIGObserver *obs) const
1193 uint numEntities = _Villages.getNumEntities();
1194 for(uint k = 0; k < numEntities; ++k)
1196 CVillage *v = dynamic_cast<CVillage *>(_Villages.getEntity(k));
1197 if (v) // well this should always be the case, but it may change in the future
1199 if (v->isObserver(obs))
1200 return true;
1203 return false;
1207 //=========================================================================
1208 COutpost *CContinent::getOutpost (uint outpostId)
1210 for (uint i=0; i<_Outposts.size(); i++)
1212 // Outpost ?
1213 if(_Outposts[i].getOutpostId()==(sint)outpostId)
1214 return &_Outposts[i];
1216 return NULL;
1219 //=========================================================================
1220 void CContinent::initOutpost()
1222 for (uint i=0; i<_Outposts.size(); i++)
1224 _Outposts[i].initOutpost();
1228 //=========================================================================
1229 void CContinent::removeOutpost()
1231 for (uint i=0; i<_Outposts.size(); i++)
1233 _Outposts[i].removeOutpost();
1238 //=========================================================================
1239 void CContinent::dumpVillagesLoadingZones(const std::string &filename)
1241 const uint NUM_METER_PER_PIX = 4;
1242 CBitmap outBitmap;
1243 CVector2f minPos, maxPos;
1244 getPosFromZoneName(ZoneMin, minPos);
1245 getPosFromZoneName(ZoneMax, maxPos);
1246 if (minPos.x > maxPos.x) std::swap(minPos.x, maxPos.x);
1247 if (minPos.y > maxPos.y) std::swap(minPos.y, maxPos.y);
1248 uint width = (uint) ((maxPos.x - minPos.x) / NUM_METER_PER_PIX);
1249 uint height = (uint) ((maxPos.y - minPos.y) / NUM_METER_PER_PIX);
1250 outBitmap.resize(width, height);
1251 for(uint k = 0; k < _Villages.getNumEntities(); ++k)
1253 CVillage &v = *safe_cast<CVillage *>(_Villages.getEntity(k));
1254 if(!v.isOutpost())
1256 const CStreamableIG &sig = v.getIG();
1257 drawDisc(outBitmap, (sig.getPos().x - minPos.x) / NUM_METER_PER_PIX, (sig.getPos().y - minPos.y) / NUM_METER_PER_PIX, sig.getUnloadRadius() / NUM_METER_PER_PIX, CRGBA(0, 0, 100), true);
1258 drawDisc(outBitmap, (sig.getPos().x - minPos.x) / NUM_METER_PER_PIX, (sig.getPos().y - minPos.y) / NUM_METER_PER_PIX, sig.getLoadRadius() / NUM_METER_PER_PIX, CRGBA(0, 100, 0), true);
1259 drawDisc(outBitmap, (sig.getPos().x - minPos.x) / NUM_METER_PER_PIX, (sig.getPos().y - minPos.y) / NUM_METER_PER_PIX, sig.getForceLoadRadius() / NUM_METER_PER_PIX, CRGBA(100, 0, 0), true);
1262 // load the map and do a lerp between the 2 bitmaps
1263 if (!WorldMap.empty())
1265 std::string path = CPath::lookup(WorldMap, false);
1266 if (!path.empty())
1268 CIFile stream;
1269 if (stream.open(path))
1271 CBitmap worldMap;
1272 if (worldMap.load(stream))
1274 worldMap.convertToType(CBitmap::RGBA);
1275 worldMap.flipV();
1276 worldMap.resample(outBitmap.getWidth(), outBitmap.getHeight());
1277 outBitmap.blend(outBitmap, worldMap, 127, true);
1282 drawDisc(outBitmap, ((float) UserEntity->pos().x - minPos.x) / (float) NUM_METER_PER_PIX, ((float) UserEntity->pos().y - minPos.y) / (float) NUM_METER_PER_PIX, 2, CRGBA::Magenta);
1284 // draw grid
1285 const uint subdiv= 500;
1286 const CRGBA lineCol(CRGBA::Yellow);
1287 sint minY= (sint)floor(minPos.y/subdiv);
1288 sint maxY= (sint)ceil (maxPos.y/subdiv);
1289 sint minX= (sint)floor(minPos.x/subdiv);
1290 sint maxX= (sint)ceil (maxPos.x/subdiv);
1291 // draw HLines
1292 for(sint y= minY;y<maxY;y++)
1294 sint ypix= (sint)((y*subdiv-minPos.y)/NUM_METER_PER_PIX);
1295 if(ypix>=0 && ypix<(sint)height)
1297 for(uint x=0;x<width;x++)
1299 CRGBA *pCol= (CRGBA*)(&outBitmap.getPixels()[0]);
1300 pCol[ypix*width+x]= lineCol;
1304 // draw VLines
1305 for(sint x= minX;x<maxX;x++)
1307 sint xpix= (sint)((x*subdiv-minPos.x)/NUM_METER_PER_PIX);
1308 if(xpix>=0 && xpix<(sint)width)
1310 for(uint y=0;y<height;y++)
1312 CRGBA *pCol= (CRGBA*)(&outBitmap.getPixels()[0]);
1313 pCol[y*width+xpix]= lineCol;
1319 COFile outFile;
1320 if (outFile.open(filename))
1324 outBitmap.writeTGA(outFile, 24, true);
1326 catch(const EStream &)
1332 //=========================================================================
1333 void CContinent::dumpFogMap(CFogMapBuild::TMapType mapType, const std::string &filename, TChannel channel /*= ChannelRGBA*/, const CRGBA channelLookup[256] /* = NULL */)
1335 CBitmap outBitmap = FogMap.getMap(mapType);
1336 outBitmap.convertToType(CBitmap::RGBA);
1337 if (channel != ChannelRGBA)
1340 CRGBA *pix = (CRGBA *) &outBitmap.getPixels(0)[0];
1341 const CRGBA *endPix = pix + outBitmap.getWidth() * outBitmap.getHeight();
1342 while (pix != endPix)
1344 uint8 intensity = ((uint8 *) pix)[channel];
1345 if (!channelLookup)
1347 *pix = CRGBA(intensity, intensity, intensity, intensity);
1349 else
1351 *pix = channelLookup[intensity];
1353 ++pix;
1356 if (outBitmap.getWidth() == 0 || outBitmap.getHeight() == 0) return;
1357 CVector pos = MainCam.getMatrix().getPos();
1358 float mx, my;
1359 FogMap.worldPosToMapPos(pos.x, pos.y, mx, my);
1360 // load the map and do a lerp between the 2 bitmaps
1361 if (!WorldMap.empty())
1363 std::string path = CPath::lookup(WorldMap, false);
1364 if (!path.empty())
1366 CIFile stream;
1367 if (stream.open(path))
1369 CBitmap worldMap;
1370 if (worldMap.load(stream))
1372 worldMap.convertToType(CBitmap::RGBA);
1373 worldMap.resample(outBitmap.getWidth(), outBitmap.getHeight());
1374 outBitmap.blend(outBitmap, worldMap, 127, true);
1379 drawDisc(outBitmap, mx * outBitmap.getWidth(), my * outBitmap.getHeight(), 2.f, CRGBA::Magenta);
1380 COFile outFile;
1381 if (outFile.open(filename))
1385 outBitmap.writeTGA(outFile, 24);
1387 catch(const EStream &)
1393 //=========================================================================
1394 void CContinent::initSky()
1396 if (!SkySheet[CurrSeason].empty())
1398 // get the sky sheet
1399 CEntitySheet *skySheet = SheetMngr.get(NLMISC::CSheetId(SkySheet[CurrSeason]));
1400 if (skySheet && skySheet->Type == CEntitySheet::SKY)
1402 // new-style sky
1403 CurrentSky.init(Driver, *static_cast<CSkySheet *>(skySheet), false, WorldLightCycle.NumHours);
1405 #ifdef USE_WATER_ENV_MAP
1406 WaterEnvMapRdr.Sky = &CurrentSky;
1407 #endif
1409 else
1411 createSkyScene();
1412 // fallback to previous sky version
1413 Sky = SkyScene->createInstance(SkyDay);
1414 Sky2ndPass = SkyScene->createInstance(SkyDay); // Sky shape used for second pass. We use it to keep pointers on textures
1415 SkyFogPart = SkyScene->createInstance(SkyFogPartName);
1416 // Setup the 2nd
1417 if (!Sky.empty())
1419 DaySkySetup.buildFromInstance(Sky, 0); // day textures are at stage 0
1420 NightSkySetup.buildFromInstance(Sky, 1); // night textures are at stage 1
1425 //=========================================================================
1426 void CContinent::releaseSky()
1428 // Remove the (old-style) sky
1429 if (!Sky.empty())
1431 SkyScene->deleteInstance (Sky);
1432 SkyScene->deleteInstance (Sky2ndPass);
1433 SkyScene->deleteInstance(SkyFogPart);
1434 Sky = NULL;
1435 Sky2ndPass = NULL;
1436 SkyFogPart = NULL;
1438 deleteSkyScene();
1440 // Release the (new-style) sky
1441 CurrentSky.release();