2#include "ViewContext.h"
3#include "Resources/DataFetcherManager.h"
4#include "Resources/MaterialManager.h"
5#include "ExtensionRegistry.h"
6#include "Services/DPIService.h"
8#include "Services/Services.h"
9#include "Generators/TextureGenerator.h"
11#include "Systems/Core/TransformSystem.h"
12#include "Systems/Core/ClipShapeSystem.h"
14#include "PotreeSystem.h"
15#include "PotreeRenderer.h"
17#include "Rendering/IGraphicsDevice.h"
18#include "Rendering/ICapabilities.h"
19#include "Foundation/Logging/Logger.h"
20#include "Foundation/Platform/IO.h"
24 const Cogs::StringView spacingGlobalScaleName =
"potree.spacing.globalScale";
25 const Cogs::StringView pointSizeGlobalMinScaleName =
"potree.pointSize.globalMinScale";
26 const Cogs::StringView pointSizeGlobalMaxScaleName =
"potree.pointSize.globalMaxScale";
27 const Cogs::StringView pointSizeGlobalScaleName =
"potree.pointSize.globalScale";
41 static uint32_t idCounter = 1;
42 if (idCounter == 0) idCounter++;
50 unsigned popcount(uint8_t x)
52 x = (x & 0x55u) + ((x >> 1) & 0x55u);
53 x = (x & 0x33u) + ((x >> 2) & 0x33u);
54 x = (x & 0x0Fu) + ((x >> 4) & 0x0Fu);
66 poSystem->requestsInFlight++;
67 poData->fetch_id = DataFetcherManager::fetchAsync(context, poData->
jsonPath, [context, instanceId=poData->instanceId](std::unique_ptr<Cogs::FileContents> data)
69 auto task = [context, instanceId, dataPtr = data.release()]()
71 std::unique_ptr<Cogs::FileContents> data(dataPtr);
72 PotreeSystem* poSystem = nullptr;
73 PotreeComponent* poComp = nullptr;
74 PotreeData* poData = nullptr;
75 bool ret = PotreeSystem::lookupAndCheckForStaleness(context, poSystem, poData, poComp, instanceId);
76 poSystem->requestsInFlight--;
77 PotreeSystem::endFetch(context, poComp, poData);
81 LOG_TRACE(logger,
"[instance=%u] Received stale metadata file, ignoring: %.*s", instanceId, StringViewFormat(data->origin()));
85 else if (poData->state != PotreeState::RequestedCloudJS) {
87 LOG_ERROR(logger,
"[instance=%u] Unexpected metadata file, ignoring: %.*s", instanceId, StringViewFormat(data->origin()));
90 LOG_ERROR(logger,
"[instance=%u] Unexpected metadata file", instanceId);
94 LOG_ERROR(logger,
"[instance=%u] Failed to get metadata file: %s", instanceId, poData ? poData->jsonPath.c_str() :
"");
95 poData->state = PotreeState::Error;
97 else if (!parseCloudJs(context, poData, data.get())) {
98 LOG_ERROR(logger,
"[instance=%u] Error while parsing metadata file", instanceId);
99 poData->state = PotreeState::Error;
102 LOG_TRACE(logger,
"[instance=%u] metadata parsed successfully", instanceId);
103 poData->state = PotreeState::ParsedCloudJs;
106 poData->fetch_id = DataFetcherManager::NoFetchId;
107 context->engine->setDirty();
110 context->
engine->runTaskInMainThread(std::move(task));
111 }, 0, 0,
true, Cogs::FileContentsHints::None);
117 auto * PM = poData->pointMaterial.instance->
material;
118 poData->pointMaterial.pointSize = PM->getFloatKey(
"pointSize");
119 poData->pointMaterial.shadingIntensity = PM->getFloatKey(
"shadingIntensity");
120 poData->pointMaterial.octTexScale = PM->getFloatKey(
"octTexScale");
121 poData->pointMaterial.minPointSize = PM->getFloatKey(
"minPointSize");
122 poData->pointMaterial.maxPointSize = PM->getFloatKey(
"maxPointSize");
123 poData->pointMaterial.outlineSize = PM->getFloatKey(
"outlineSize");
124 poData->pointMaterial.outlineColor = PM->getVec3Key(
"outlineColor");
125 poData->pointMaterial.baseColor = PM->getVec4Key(
"baseColor");
126 poData->pointMaterial.invShift = PM->getVec3Key(
"invShift");
127 poData->pointMaterial.invScale = PM->getVec3Key(
"invScale");
128 poData->pointMaterial.gradient = PM->getTextureKey(
"Gradient");
130 auto* BM = poData->boxMaterial.instance->
material;
131 poData->boxMaterial.scale = BM->getVec3Key(
"scale");
132 poData->boxMaterial.offset = BM->getVec3Key(
"offset");
137 float minPointSize = poData->pointSize.
min;
138 float maxPointSize = poData->pointSize.
max;
139 float pointSize = poData->pointSize.
val;
143 float rawPointSize = pointSize / context->getDefaultView()->
dpiService->getScaleFactor();
144 float pointSizeScale = glm::clamp(rawPointSize / 10.f, 1e-2f, 1e2f);
145 poData->modelSpaceRadius = 0.5f * pointSizeScale * poData->spacing;
147 auto& PM = poData->pointMaterial;
148 bool hasFragDepth = context->device->getCapabilities()->getDeviceCapabilities().FragDepth;
150 case PotreePointShape::Square:
151 PM.instance->setVariant(
"Shape",
"Square");
152 hasFragDepth =
false;
154 case PotreePointShape::Disc:
155 PM.instance->setVariant(
"Shape",
"Disc");
156 hasFragDepth =
false;
158 case PotreePointShape::Paraboloid:
159 PM.instance->setVariant(
"Shape", hasFragDepth ?
"Paraboloid" :
"Square");
161 case PotreePointShape::Sphere:
162 PM.instance->setVariant(
"Shape", hasFragDepth ?
"Sphere" :
"Disc");
165 LOG_FATAL(logger,
"Component member pointShape has illegal enum value: %u",
unsigned(poComp->
pointShape));
166 poComp->
pointShape = PotreePointShape::Square;
167 PM.instance->setVariant(
"Shape",
"Square");
168 hasFragDepth =
false;
173 PM.instance->setVariant(
"CustomDepth", hasFragDepth);
177 case PotreePointSizing::FixedSize:
178 PM.instance->setVariant(
"PointSizing",
"FixedSize");
180 case PotreePointSizing::DistanceScaled:
181 PM.instance->setVariant(
"PointSizing",
"DistanceScaled");
183 case PotreePointSizing::DistanceAndDensityScaled:
184 PM.instance->setVariant(
"PointSizing",
"DistanceAndDensityScaled");
187 LOG_FATAL(logger,
"Component member pointSizing has illegal enum value: %u",
unsigned(poComp->
pointSizing));
188 poComp->
pointSizing = PotreePointSizing::FixedSize;
189 PM.instance->setVariant(
"PointSizing",
"FixedSize");
195 case PotreeShading::Flat:
196 PM.instance->setVariant(
"BasicShading",
false);
198 case PotreeShading::Basic:
199 PM.instance->setVariant(
"BasicShading",
true);
202 LOG_FATAL(logger,
"Component member shading has illegal enum value: %u",
unsigned(poComp->
shading));
203 poComp->
shading = PotreeShading::Flat;
204 PM.instance->setVariant(
"BasicShading",
false);
209 case PotreeColoring::Base:
210 PM.instance->setVariant(
"HasColor",
false);
211 PM.instance->setVariant(
"HasIntensity",
false);
212 PM.instance->setVariant(
"Coloring",
"Base");
214 case PotreeColoring::Color:
215 PM.instance->setVariant(
"HasColor", poData->hasColor);
216 PM.instance->setVariant(
"HasIntensity",
false);
217 PM.instance->setVariant(
"Coloring", poData->hasColor ?
"Color" :
"Base");
219 case PotreeColoring::Intensity:
220 PM.instance->setVariant(
"HasColor",
false);
221 PM.instance->setVariant(
"HasIntensity", poData->hasIntensity);
222 PM.instance->setVariant(
"Coloring", poData->hasIntensity ?
"Intensity" :
"Base");
224 case PotreeColoring::Level:
225 PM.instance->setVariant(
"HasColor",
false);
226 PM.instance->setVariant(
"HasIntensity",
false);
227 PM.instance->setVariant(
"Coloring",
"Level");
230 LOG_FATAL(logger,
"Component member coloring has illegal enum value: %u",
unsigned(poComp->
coloring));
231 poComp->
coloring = PotreeColoring::Base;
232 PM.instance->setVariant(
"Coloring",
"Base");
235 PM.instance->setVariant(
"Outline", poComp->
outlineSize != 0.f);
236 PM.instance->setVariant(
"ClipPlanes",
static_cast<int>(poData->clipPlaneCount));
238 if (PM.pointSize != NoProperty) PM.instance->setFloatProperty(PM.pointSize, pointSize);
239 if (PM.shadingIntensity != NoProperty) PM.instance->setFloatProperty(PM.shadingIntensity, poComp->
shadingIntensity);
240 if (PM.octTexScale != NoProperty) PM.instance->setFloatProperty(PM.octTexScale, 1.f / poData->octtreeTextureData.size());
241 if (PM.minPointSize != NoProperty) PM.instance->setFloatProperty(PM.minPointSize, minPointSize);
242 if (PM.maxPointSize != NoProperty) PM.instance->setFloatProperty(PM.maxPointSize, maxPointSize);
243 if (PM.baseColor != NoProperty) PM.instance->setVec4Property(PM.baseColor, poComp->
baseColor);
244 if (PM.outlineColor != NoProperty) PM.instance->setVec3Property(PM.outlineColor, poComp->
outlineColor);
245 if (PM.outlineSize != NoProperty) PM.instance->setFloatProperty(PM.outlineSize, poComp->
outlineSize);
246 if (PM.gradient != NoProperty) PM.instance->setTextureProperty(PM.gradient,
HandleIsValid(poComp->
gradient) ? poComp->
gradient : system->defaultGradient);
248 if (PM.invShift != NoProperty) PM.instance->setVec3Property(PM.invShift, 0.5f * (poData->octtreeFrame.tightBBoxMax - poData->octtreeFrame.tightBBoxMin));
249 if (PM.invScale != NoProperty) PM.instance->setVec3Property(PM.invScale, glm::vec3(1.f) / (poData->octtreeFrame.fullBBoxMax - poData->octtreeFrame.fullBBoxMin));
251 auto& BM = poData->boxMaterial;
252 if (BM.scale != NoProperty) BM.instance->
setVec3Property(BM.scale, poData->octtreeFrame.fullBBoxSize);
253 if (BM.offset != NoProperty) BM.instance->setVec3Property(BM.offset,-poData->octtreeFrame.currentPositionInEntityFrame);
258 poData->instanceId = createId();
259 poData->pointMaterial.instance = context->materialInstanceManager->createMaterialInstance(system->defaultPointMaterial);
260 poData->boxMaterial.instance = context->materialInstanceManager->createMaterialInstance(system->defaultBoxMaterial);
261 setUpMaterial(context, poComp, poData);
266 LOG_DEBUG(logger,
"[instance=%u] Releasing resources (%s)", poData->instanceId, poData->
jsonPath.c_str());
267 while (!poData->subtrees.empty()) {
268 auto* subtree = poData->subtrees.shift();
269 subtree->release(context, poData);
270 poData->subtreeStore.destroy(subtree);
272 DataFetcherManager::cancelAsyncFetch(context, poData->fetch_id);
273 poData->fetch_id = DataFetcherManager::NoFetchId;
274 assert(poData->subtrees.empty());
275 assert(poData->cellStore.size() == 0);
276 assert(poData->subtreeStore.size() == 0);
277 poData->~PotreeData();
279 initComponentData(context, system, poComp, poData);
284PotreeVertexLayoutInfo Cogs::Core::PotreeSystem::vertexLayoutInfo[
static_cast<size_t>(PotreeVertexLayout::COUNT)] =
287 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
289 { 0, 3 * 4, ~0u, ~0u, ~0u, 4 * 4, {
294 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
296 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
298 { 0, ~0u, ~0u, 3 * 4, ~0u, 4 * 4, {
303 { 0, 4 * 4, ~0u, 3 * 4, ~0u, 5 * 4, {
309 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
311 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
313 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
315 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
317 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
319 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
321 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
323 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
325 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
327 { ~0u, ~0u, ~0u, ~0u, ~0u, ~0u, {}},
335 if (parent ==
nullptr && !poData->subtrees.empty()) {
336 LOG_FATAL(logger,
"[instanceId=%u] Fetching root node but subtree array is not empty.", poData->instanceId);
337 assert(parent !=
nullptr || poData->subtrees.empty());
340 subtree = poData->subtrees.push(poData->subtreeStore.create());
341 subtree->subtreeId = createId();
342 subtree->fetchLayoutAndInitialize(
context,
this, poComp, poData, parent);
347 potreeSystem = ExtensionRegistry::getExtensionSystem<PotreeSystem>(context);
348 for (
auto& poComp_ : potreeSystem->
pool) {
351 PotreeData* poData_ = poDataHolder.poData.get();
354 if (poData_->instanceId == instanceId) {
360 LOG_TRACE(logger,
"[instance=%u] Failed to lookup instance, doesn't exist anymore.", instanceId);
366 if (lookupAndCheckForStaleness(context, potreeSystem, poData, poComp, instanceId)) {
367 for (
auto* item : poData->subtrees) {
368 if (item->subtreeId == subtreeId) {
374 LOG_TRACE(logger,
"[instance=%u,tree=%u] Failed to lookup subtree, doesn't exist anymore.", instanceId, subtreeId);
386 defaultGradient = texGen->getTexture(Cogs::Core::ImageType::GradientHeat);
388 if (!context->
variables->exist(spacingGlobalScaleName)) {
389 context->
variables->set(spacingGlobalScaleName, 1.f);
392 if (!context->
variables->exist(pointSizeGlobalScaleName)) {
393 context->
variables->set(pointSizeGlobalScaleName, 1.f);
396 if (!context->
variables->exist(pointSizeGlobalMinScaleName)) {
397 context->
variables->set(pointSizeGlobalMinScaleName, 1.f);
400 if (!context->
variables->exist(pointSizeGlobalMaxScaleName)) {
401 context->
variables->set(pointSizeGlobalMaxScaleName, 1.f);
414 if (!materialLoaded) {
415 context->materialManager->loadMaterial(
"Materials/PotreeBoxCore.material");
416 context->materialManager->loadMaterial(
"Materials/PotreePointCore.material");
417 defaultBoxMaterial = context->materialManager->loadMaterial(
"Materials/PotreeBoxDefault.material");
418 defaultPointMaterial = context->materialManager->loadMaterial(
"Materials/PotreePointDefault.material");
419 context->materialManager->processLoading();
421 materialLoaded =
true;
426 auto& poDataHolder = getData(poComp);
427 poDataHolder.poData = std::make_unique<PotreeData>();
428 initComponentData(context,
this, poComp, poDataHolder.poData.get());
430 if(pool.size() == 1) {
431 assert(bounds ==
nullptr);
433 context->
bounds->addBoundsExtension(bounds);
435 assert(picker ==
nullptr);
439 LOG_DEBUG(logger,
"Registered potree bounds and picker extensions");
450 if (pool.size() == 0) {
452 context->
bounds->removeBoundsExtension(bounds);
461 LOG_DEBUG(logger,
"Last component destroyed, unregistered potree bounds and picker extensions");
468 CpuInstrumentationScope(SCOPE_POTREE,
"PotreeSystem::preUpdate");
470 for (
auto& poComp : pool) {
475 PotreeData* poData = poDataHolder.poData.get();
479 poData->octtreeFrame.currentPositionInEntityFrame = poData->octtreeFrame.positionInEntityFrame;
482 poData->octtreeFrame.currentPositionInEntityFrame = glm::vec3(0.f);
485 poData->octtreeFrame.currentPositionInEntityFrame = glm::vec3(0.f);
486 if (trComp->
coordinates != poData->octtreeFrame.positionInEntityFrame) {
487 trComp->
coordinates = poData->octtreeFrame.positionInEntityFrame;
500 const glm::vec3& o = poData.octtreeFrame.currentPositionInEntityFrame;
502 poData.worldFromOcttreeFrame = context->transformSystem->getLocalToWorld(transformComp) * glm::mat4(1.f, 0.f, 0.f, 0.f,
512 CpuInstrumentationScope(SCOPE_POTREE,
"PotreeSystem::update");
514 const float spacingGlobalScale = context->
variables->get(spacingGlobalScaleName, 1.f);
515 const float pointSizeGlobalMinScale = context->
variables->get(pointSizeGlobalMinScaleName, 1.f);
516 const float pointSizeGlobalMaxScale = context->
variables->get(pointSizeGlobalMaxScaleName, 1.f);
517 const float pointSizeGlobalScale = context->
variables->get(pointSizeGlobalScaleName, 1.f);
519 for (
auto& poComp : pool) {
522 PotreeData* poData = poDataHolder.poData.get();
525 if (poData->state != PotreeState::Uninitialized) {
526 resetComponentData(context,
this, &poComp, poData);
527 assert(poData->state == PotreeState::Uninitialized);
533 if (poData->state != PotreeState::Uninitialized &&
537 resetComponentData(context,
this, &poComp, poData);
538 assert(poData->state == PotreeState::Uninitialized);
541 if (poData->state == PotreeState::Uninitialized) {
545 if (
size_t o = poData->
jsonPath.find_last_of(
"\\/"); o != std::string::npos) {
552 poData->instanceId = createId();
553 poData->state = PotreeState::RequestedCloudJS;
554 issueCloudJSFetch(context,
this, &poComp, poData);
559 if (poData->state == PotreeState::ParsedCloudJs) {
560 poData->state = PotreeState::Running;
561 issueSubtreeFetch(context, &poComp, poData,
nullptr);
565 bool materialChanged =
false;
568 materialChanged = materialChanged || (poData->clipPlaneCount != 0);
572 if (
EntityPtr clipCompEntity = clipRefComp->clipShape.lock(); clipCompEntity) {
586 for (
size_t i = 0; i < clipPlaneCount; i++) {
590 materialChanged = materialChanged || (poData->clipPlaneCount != clipPlaneCount);
591 poData->clipPlaneCount = clipPlaneCount;
596 if (poData->pointMaterial.instance != poComp.
pointMaterial) {
597 LOG_DEBUG(logger,
"Using point material instance from component, state=%u", uint32_t(poData->state));
599 materialChanged =
true;
603 if (!poData->pointMaterial.instance || poData->pointMaterial.instance->
material != defaultPointMaterial.resolve()) {
604 LOG_DEBUG(logger,
"Creating default point material instance, state=%u", uint32_t(poData->state));
605 poData->pointMaterial.instance = context->materialInstanceManager->createMaterialInstance(defaultPointMaterial);
606 materialChanged =
true;
611 if (poData->boxMaterial.instance != poComp.
boxMaterial) {
612 LOG_DEBUG(logger,
"Using box material instance from component, state=%u", uint32_t(poData->state));
614 materialChanged =
true;
618 if (!poData->boxMaterial.instance || poData->boxMaterial.instance->
material != defaultBoxMaterial.resolve()) {
619 LOG_DEBUG(logger,
"Creating default box material instance, state=%u", uint32_t(poData->state));
620 poData->boxMaterial.instance = context->materialInstanceManager->createMaterialInstance(defaultBoxMaterial);
621 materialChanged =
true;
624 if (materialChanged) setUpMaterial(context, &poComp, poData);
627 updateOcttreeFrame(context, *poData, poComp);
629 float gridSize = std::max(1.f, poData->octtreeFrame.fullBBoxSize.x / poData->spacing);
630 float dpiScale = context->getDefaultView()->
dpiService->getScaleFactor();
631 poData->tolerance = std::max(1.f, dpiScale * gridSize * spacingGlobalScale * poComp.
pointScreenSpacing);
638 poData->pointSize.
min = pointSizeGlobalMinScale * dpiScale * poComp.
minPointSize;
639 poData->pointSize.
max = pointSizeGlobalMaxScale * dpiScale * poComp.
maxPointSize;
640 poData->pointSize.
val = glm::clamp(pointSizeGlobalScale * dpiScale * poComp.
pointSize, poData->pointSize.
min, poData->pointSize.
max);
647 refreshMaterialInstances(context,
this, &poComp, poData);
649 updateLodLevels(context);
669 if (poComp && poData) {
674 LOG_ERROR(logger,
"endFetch: Too many endFetch received");
void setChanged()
Sets the component to the ComponentFlags::Changed state with carry.
ComponentType * getComponent() const
Sets up a clipping shape that can be used by multiple entities.
Component that attaches a ClipShape to an entity.
virtual ComponentHandle createComponent()
Create a new component instance.
Context * context
Pointer to the Context instance the system lives in.
virtual void initialize(Context *context)
Initialize the system.
virtual void destroyComponent(ComponentHandle)
Destroy the component held by the given handle.
void update()
Updates the system state to that of the current frame.
void preUpdate()
Run the pre-update method of the system.
ComponentPool< ComponentType > pool
Pool of components managed by the system.
A Context instance contains all the services, systems and runtime components needed to use Cogs.
std::unique_ptr< class Services > services
Services.
class IRenderer * renderer
Renderer.
std::unique_ptr< class Bounds > bounds
Bounds service instance.
std::unique_ptr< class RayPicking > rayPicking
RayPicking service instance.
std::unique_ptr< class Variables > variables
Variables service instance.
std::unique_ptr< class Engine > engine
Engine instance.
virtual void registerExtension(IRendererExtension *extension)=0
Register an extension with the renderer.
virtual void unregisterExtension(IRendererExtension *extension)=0
Unregister an extension with the renderer.
ComponentModel::ComponentHandle clipShapeComponent
Handle to the currently active clip component, if any.
constexpr bool isVisible() const
Check if the entity is visible or not.
std::unique_ptr< class DPIService > dpiService
DPI service instance.
Log implementation class.
Provides a weakly referenced view over the contents of a string.
Contains the Engine, Renderer, resource managers and other systems needed to run Cogs....
std::shared_ptr< ComponentModel::Entity > EntityPtr
Smart pointer for Entity access.
@ FetchingBegin
Loading queue transitions from empty to non-empty, that is, instance needs data.
@ FetchingEnd
Loading queue transitions from non-empty to empty, that is, instance has all data it currently needs.
bool HandleIsValid(const ResourceHandle_t< T > &handle)
Check if the given resource is valid, that is not equal to NoHandle or InvalidHandle.
@ CenterOnOrigin
Adjust offset s.t. point cloud is centered around entity origin.
@ None
Just use scale and offset as is.
@ CenterOnOriginAdjustCoordinate
Adjust offset s.t. point cloud is centered around enttiy origin and adjust transformcomponent coordin...
constexpr Log getLogger(const char(&name)[LEN]) noexcept
@ OpenGLES30
Graphics device using the OpenGLES 3.0 API.
@ VertexData
Per vertex data.
@ Position
Position semantic.
@ TextureCoordinate
Texture coordinate semantic.
Handle to a Component instance.
COGSFOUNDATION_API class Component * resolve() const
Resolve the handle, returning a pointer to the held Component instance.
static ComponentHandle Empty()
Returns an empty, invalid handle. Will evaluate to false if tested against using operator bool().
void setVec3Property(const VariableKey key, glm::vec3 value)
Set the vec3 property with the given key to value.
Material * material
Material resource this MaterialInstance is created from.
Component for Point Cloud Display.
float maxPointSize
Maximum point size in pixels regardless of point size strategy, gets DPI scaled.
std::vector< glm::vec4 > clipPlanes
Deprecated, use clip shapes instead.
MaterialInstanceHandle boxMaterial
Optional debug material override, material should inherit from PotreeBoxCore.material.
float shadingIntensity
Intensity of point shading effect, if enabled.
float pointSize
Base point size in pixels, gets DPI scaled.
float outlineSize
Size of point outlines, set to 0.0 to disable point outlines.
PotreeColoring coloring
Specify coloring strategy.
TextureHandle gradient
Optional gradient texture to colorize scalar values.
std::string source
URL of path where metadata.js(OLD format: cloud.js) resides.
MaterialInstanceHandle pointMaterial
Optional point material override, material should inherit from PotreePointCore.material.
glm::vec3 outlineColor
Color of point outlines if enabled.
float rootNodeMinScreenSize
Minimal screen-space size root node can have before being eligble for distance-based culling.
PotreeShading shading
Specify optional shading.
PotreePointShape pointShape
Specify point shape.
glm::vec4 baseColor
Base color to use when Base coloring strategy is used.
float densityBias
DistanceAndDensityScaled: Offset where depth and density starts to affectg point size calc.
PotreePointSizing pointSizing
Specify point sizing strategy.
bool disableCustomClipping
Debug switch to disable clipping of points against custom clipping planes.
float densityLevelScale
DistanceAndDensityScaled: Strength of level depth in point size calc.
bool supportPicking
If true, an extra CPU side copy of the point positions are maintained to enable CPU picking.
float pointScreenSpacing
Maximal screen space distance between points before refining, gets adjusted by DPI scaling.
float densityScale
DistanceAndDensityScaled: Strength of local point density in point size calc.
PotreeOriginPolicy originPolicy
Specify origin policy, see enum for details.
PotreeDebugBoxes debugBoxes
Specify optional debug graphics.
float minPointSize
Minimum point size in pixels regardless of point size strategy, gets DPI scaled.
uint32_t requestsInFlight
Number of outstanding data fetch requests.
float val
Current dpi-scaled and min-max clamped point-size.
float min
Current dpi-scaled minimum point-size.
float max
Current dpi-scaled maximum point-size.
static constexpr size_t MaxClipPlanes
Max available in shader.
std::string jsonPath
URL to metadata file.
PotreeNotification potreeNotificationState
Current notification state for this PoTree Component.
std::string rootPath
URL to folder containing metadata file.
ComponentHandle createComponent() override
static void startFetch(Context *context, const PotreeComponent *poComp, PotreeData *poData)
Update Component request count and notify client when starting data fetch.
void destroyComponent(ComponentHandle component) override
void cleanup(Context *context) override
Provided for custom cleanup logic in derived systems.
void initialize(Context *context) override
Initialize the system.
static void endFetch(Context *context, const PotreeComponent *poComp, PotreeData *poData)
Abstract base class storing data read from a file.
@ Clamp
Texture coordinates are clamped to the [0, 1] range.