cesium-native 0.46.0
Loading...
Searching...
No Matches
PropertyTexturePropertyView.h
1#pragma once
2
3#include <CesiumGltf/ImageAsset.h>
4#include <CesiumGltf/KhrTextureTransform.h>
5#include <CesiumGltf/PropertyTextureProperty.h>
6#include <CesiumGltf/PropertyTransformations.h>
7#include <CesiumGltf/PropertyTypeTraits.h>
8#include <CesiumGltf/PropertyView.h>
9#include <CesiumGltf/Sampler.h>
10#include <CesiumGltf/TextureView.h>
11#include <CesiumUtility/Assert.h>
12
13#include <array>
14#include <cmath>
15#include <cstdint>
16#include <optional>
17
18namespace CesiumGltf {
28public:
33 static const int ErrorInvalidPropertyTexture = 15;
34
39 static const int ErrorUnsupportedProperty = 16;
40
44 static const int ErrorInvalidTexture = 17;
45
49 static const int ErrorInvalidSampler = 18;
50
54 static const int ErrorInvalidImage = 19;
55
59 static const int ErrorEmptyImage = 20;
60
65 static const int ErrorInvalidBytesPerChannel = 21;
66
74 static const int ErrorInvalidChannels = 22;
75
82 static const int ErrorChannelsAndTypeMismatch = 23;
83};
84
92template <typename ElementType>
93ElementType assembleScalarValue(const std::span<uint8_t> bytes) noexcept {
94 if constexpr (std::is_same_v<ElementType, float>) {
95 CESIUM_ASSERT(
96 bytes.size() == sizeof(float) &&
97 "Not enough channel inputs to construct a float.");
98 uint32_t resultAsUint = 0;
99 for (size_t i = 0; i < bytes.size(); i++) {
100 resultAsUint |= static_cast<uint32_t>(bytes[i]) << i * 8;
101 }
102
103 // Reinterpret the bits as a float.
104 // We need to memcpy to avoid a "dereferencing type-punned pointer will
105 // break strict-aliasing rules" error on GCC. See:
106 // https://gist.github.com/shafik/848ae25ee209f698763cffee272a58f8#how-do-we-type-pun-correctly
107 float resultAsFloat;
108 std::memcpy(&resultAsFloat, &resultAsUint, sizeof(float));
109 return resultAsFloat;
110 }
111
113 using UintType = std::make_unsigned_t<ElementType>;
114 UintType resultAsUint = 0;
115 for (size_t i = 0; i < bytes.size(); i++) {
116 resultAsUint |=
117 static_cast<UintType>(static_cast<UintType>(bytes[i]) << i * 8);
118 }
119
120 // Reinterpret the bits with the correct signedness.
121 return *reinterpret_cast<ElementType*>(&resultAsUint);
122 }
123}
124
132template <typename ElementType>
133ElementType assembleVecNValue(const std::span<uint8_t> bytes) noexcept {
134 ElementType result = ElementType();
135
136 [[maybe_unused]] constexpr glm::length_t N =
138 using T = typename ElementType::value_type;
139
140 CESIUM_ASSERT(
141 sizeof(T) <= 2 && "Components cannot be larger than two bytes in size.");
142
143 if constexpr (std::is_same_v<T, int16_t>) {
144 CESIUM_ASSERT(
145 N == 2 && "Only vec2s can contain two-byte integer components.");
146 uint16_t x = static_cast<uint16_t>(bytes[0]) |
147 static_cast<uint16_t>(static_cast<uint16_t>(bytes[1]) << 8);
148 uint16_t y = static_cast<uint16_t>(bytes[2]) |
149 static_cast<uint16_t>(static_cast<uint16_t>(bytes[3]) << 8);
150
151 result[0] = *reinterpret_cast<int16_t*>(&x);
152 result[1] = *reinterpret_cast<int16_t*>(&y);
153 }
154
155 if constexpr (std::is_same_v<T, uint16_t>) {
156 CESIUM_ASSERT(
157 N == 2 && "Only vec2s can contain two-byte integer components.");
158 result[0] = static_cast<uint16_t>(bytes[0]) |
159 static_cast<uint16_t>(static_cast<uint16_t>(bytes[1]) << 8);
160 result[1] = static_cast<uint16_t>(bytes[2]) |
161 static_cast<uint16_t>(static_cast<uint16_t>(bytes[3]) << 8);
162 }
163
164 if constexpr (std::is_same_v<T, int8_t>) {
165 for (size_t i = 0; i < bytes.size(); i++) {
166 result[int32_t(i)] = *reinterpret_cast<const int8_t*>(&bytes[i]);
167 }
168 }
169
170 if constexpr (std::is_same_v<T, uint8_t>) {
171 for (size_t i = 0; i < bytes.size(); i++) {
172 result[int32_t(i)] = bytes[i];
173 }
174 }
175
176 return result;
177}
178
186template <typename T>
187PropertyArrayCopy<T>
188assembleArrayValue(const std::span<uint8_t> bytes) noexcept {
189 std::vector<T> result(bytes.size() / sizeof(T));
190
191 if constexpr (sizeof(T) == 2) {
192 for (size_t i = 0, b = 0; i < result.size(); i++, b += 2) {
193 using UintType = std::make_unsigned_t<T>;
194 UintType resultAsUint =
195 static_cast<UintType>(bytes[b]) |
196 static_cast<UintType>(static_cast<UintType>(bytes[b + 1]) << 8);
197 result[i] = *reinterpret_cast<T*>(&resultAsUint);
198 }
199 } else {
200 for (size_t i = 0; i < bytes.size(); i++) {
201 result[i] = *reinterpret_cast<const T*>(&bytes[i]);
202 }
203 }
204
205 return PropertyArrayCopy<T>(std::move(result));
206}
207
217template <typename ElementType>
219assembleValueFromChannels(const std::span<uint8_t> bytes) noexcept {
220 CESIUM_ASSERT(
221 bytes.size() > 0 && "Channel input must have at least one value.");
222
225 }
226
228 return assembleVecNValue<ElementType>(bytes);
229 }
230
232 return assembleArrayValue<typename MetadataArrayType<ElementType>::type>(
233 bytes);
234 }
235}
236
237#pragma region Non - normalized property
238
252template <typename ElementType, bool Normalized = false>
254
264template <typename ElementType>
265class PropertyTexturePropertyView<ElementType, false>
266 : public PropertyView<ElementType, false>, public TextureView {
267public:
272 : PropertyView<ElementType, false>(),
273 TextureView(),
274 _channels(),
275 _swizzle() {}
276
284 TextureView(),
285 _channels(),
286 _swizzle() {
287 CESIUM_ASSERT(
289 "An empty property view should not be constructed with a valid status");
290 }
291
299 PropertyTexturePropertyView(const ClassProperty& classProperty) noexcept
300 : PropertyView<ElementType, false>(classProperty),
301 TextureView(),
302 _channels(),
303 _swizzle() {
304 if (this->_status != PropertyTexturePropertyViewStatus::Valid) {
305 // Don't override the status / size if something is wrong with the class
306 // property's definition.
307 return;
308 }
309
310 if (!classProperty.defaultProperty) {
311 // This constructor should only be called if the class property *has* a
312 // default value. But in the case that it does not, this property view
313 // becomes invalid.
314 this->_status =
316 return;
317 }
318
320 }
321
332 const PropertyTextureProperty& property,
333 const ClassProperty& classProperty,
334 const Sampler& sampler,
335 const ImageAsset& image,
336 const TextureViewOptions& options = TextureViewOptions()) noexcept
337 : PropertyView<ElementType, false>(classProperty, property),
339 sampler,
340 image,
341 property.texCoord,
342 property.getExtension<ExtensionKhrTextureTransform>(),
343 options),
344 _channels(property.channels),
345 _swizzle() {
346 if (this->_status != PropertyTexturePropertyViewStatus::Valid) {
347 return;
348 }
349
350 switch (this->getTextureViewStatus()) {
352 break;
355 return;
358 return;
361 return;
363 this->_status =
365 return;
368 default:
370 return;
371 }
372
373 _swizzle.reserve(_channels.size());
374
375 for (size_t i = 0; i < _channels.size(); ++i) {
376 switch (_channels[i]) {
377 case 0:
378 _swizzle += "r";
379 break;
380 case 1:
381 _swizzle += "g";
382 break;
383 case 2:
384 _swizzle += "b";
385 break;
386 case 3:
387 _swizzle += "a";
388 break;
389 default:
390 CESIUM_ASSERT(
391 false && "A valid channels vector must be passed to the view.");
392 }
393 }
394 }
395
407 const PropertyTextureProperty& property,
408 const ClassProperty& classProperty,
409 const CesiumGltf::Enum* pEnumDefinition,
410 const Sampler& sampler,
411 const ImageAsset& image,
412 const TextureViewOptions& options = TextureViewOptions()) noexcept
413 : PropertyView<ElementType, false>(
414 classProperty,
415 property,
416 pEnumDefinition),
418 sampler,
419 image,
420 property.texCoord,
421 property.getExtension<ExtensionKhrTextureTransform>(),
422 options),
423 _channels(property.channels),
424 _swizzle() {
425 if (this->_status != PropertyTexturePropertyViewStatus::Valid) {
426 return;
427 }
428
429 switch (this->getTextureViewStatus()) {
431 break;
434 return;
437 return;
440 return;
442 this->_status =
444 return;
447 default:
449 return;
450 }
451
452 _swizzle.reserve(_channels.size());
453
454 for (size_t i = 0; i < _channels.size(); ++i) {
455 switch (_channels[i]) {
456 case 0:
457 _swizzle += "r";
458 break;
459 case 1:
460 _swizzle += "g";
461 break;
462 case 2:
463 _swizzle += "b";
464 break;
465 case 3:
466 _swizzle += "a";
467 break;
468 default:
469 CESIUM_ASSERT(
470 false && "A valid channels vector must be passed to the view.");
471 }
472 }
473 }
474
492 std::optional<PropertyValueViewToCopy<ElementType>>
493 get(double u, double v) const noexcept {
494 if (this->_status ==
496 return propertyValueViewToCopy(this->defaultValue());
497 }
498
499 PropertyValueViewToCopy<ElementType> value = getRaw(u, v);
500
501 if (value == this->noData()) {
502 return propertyValueViewToCopy(this->defaultValue());
503 } else if constexpr (IsMetadataNumeric<ElementType>::value) {
504 return transformValue(value, this->offset(), this->scale());
505 } else if constexpr (IsMetadataNumericArray<ElementType>::value) {
506 return transformArray(
507 propertyValueCopyToView(value),
508 this->offset(),
509 this->scale());
510 } else {
511 return value;
512 }
513 }
514
530 getRaw(double u, double v) const noexcept {
531 CESIUM_ASSERT(
533 "Check the status() first to make sure view is valid");
534
535 std::vector<uint8_t> sample =
536 this->sampleNearestPixel(u, v, this->_channels);
537
539 std::span(sample.data(), this->_channels.size()));
540 }
541
545 const std::vector<int64_t>& getChannels() const noexcept {
546 return this->_channels;
547 }
548
552 const std::string& getSwizzle() const noexcept { return this->_swizzle; }
553
554private:
555 std::vector<int64_t> _channels;
556 std::string _swizzle;
557};
558
559#pragma endregion
560
561#pragma region Normalized property
562
570template <typename ElementType>
571class PropertyTexturePropertyView<ElementType, true>
572 : public PropertyView<ElementType, true>, public TextureView {
573private:
574 using NormalizedType = typename TypeToNormalizedType<ElementType>::type;
575
576public:
581 : PropertyView<ElementType, true>(),
582 TextureView(),
583 _channels(),
584 _swizzle() {}
585
593 TextureView(),
594 _channels(),
595 _swizzle() {
596 CESIUM_ASSERT(
598 "An empty property view should not be constructed with a valid "
599 "status");
600 }
601
610 PropertyTexturePropertyView(const ClassProperty& classProperty) noexcept
611 : PropertyView<ElementType, true>(classProperty),
612 TextureView(),
613 _channels(),
614 _swizzle() {
615 if (this->_status != PropertyTexturePropertyViewStatus::Valid) {
616 // Don't override the status / size if something is wrong with the class
617 // property's definition.
618 return;
619 }
620
621 if (!classProperty.defaultProperty) {
622 // This constructor should only be called if the class property *has* a
623 // default value. But in the case that it does not, this property view
624 // becomes invalid.
625 this->_status =
627 return;
628 }
629
631 }
632
643 const PropertyTextureProperty& property,
644 const ClassProperty& classProperty,
645 const Sampler& sampler,
646 const ImageAsset& image,
647 const TextureViewOptions& options = TextureViewOptions()) noexcept
648 : PropertyView<ElementType, true>(classProperty, property),
650 sampler,
651 image,
652 property.texCoord,
653 property.getExtension<ExtensionKhrTextureTransform>(),
654 options),
655 _channels(property.channels),
656 _swizzle() {
657 if (this->_status != PropertyTexturePropertyViewStatus::Valid) {
658 return;
659 }
660
661 switch (this->getTextureViewStatus()) {
663 break;
666 return;
669 return;
672 return;
674 this->_status =
676 return;
679 default:
681 return;
682 }
683
684 _swizzle.reserve(_channels.size());
685 for (size_t i = 0; i < _channels.size(); ++i) {
686 switch (_channels[i]) {
687 case 0:
688 _swizzle += "r";
689 break;
690 case 1:
691 _swizzle += "g";
692 break;
693 case 2:
694 _swizzle += "b";
695 break;
696 case 3:
697 _swizzle += "a";
698 break;
699 default:
700 CESIUM_ASSERT(
701 false && "A valid channels vector must be passed to the view.");
702 }
703 }
704 }
705
724 std::optional<PropertyValueViewToCopy<NormalizedType>>
725 get(double u, double v) const noexcept {
726 if (this->_status ==
728 return propertyValueViewToCopy(this->defaultValue());
729 }
730
731 PropertyValueViewToCopy<ElementType> value = getRaw(u, v);
732
733 if (value == this->noData()) {
734 return propertyValueViewToCopy(this->defaultValue());
735 } else if constexpr (IsMetadataScalar<ElementType>::value) {
738 this->offset(),
739 this->scale());
740 } else if constexpr (IsMetadataVecN<ElementType>::value) {
741 constexpr glm::length_t N = ElementType::length();
742 using T = typename ElementType::value_type;
743 using NormalizedT = typename NormalizedType::value_type;
745 normalize<N, T>(value),
746 this->offset(),
747 this->scale());
748 } else if constexpr (IsMetadataArray<ElementType>::value) {
749 using ArrayElementType = typename MetadataArrayType<ElementType>::type;
752 propertyValueCopyToView(value),
753 this->offset(),
754 this->scale());
755 } else if constexpr (IsMetadataVecN<ArrayElementType>::value) {
756 constexpr glm::length_t N = ArrayElementType::length();
757 using T = typename ArrayElementType::value_type;
759 propertyValueCopyToView(value),
760 this->offset(),
761 this->scale());
762 }
763 }
764 }
765
781 getRaw(double u, double v) const noexcept {
782 CESIUM_ASSERT(
784 "Check the status() first to make sure view is valid");
785
786 std::vector<uint8_t> sample =
787 this->sampleNearestPixel(u, v, this->_channels);
788
790 std::span(sample.data(), this->_channels.size()));
791 }
792
796 const std::vector<int64_t>& getChannels() const noexcept {
797 return this->_channels;
798 }
799
803 const std::string& getSwizzle() const noexcept { return this->_swizzle; }
804
805private:
806 std::vector<int64_t> _channels;
807 std::string _swizzle;
808};
809#pragma endregion
810
811} // namespace CesiumGltf
A copy of an array element of a PropertyTableProperty or PropertyTextureProperty.
Indicates the status of a property texture property view.
static const int ErrorUnsupportedProperty
This property view is associated with a ClassProperty of an unsupported type.
static const int ErrorChannelsAndTypeMismatch
The channels of this property texture property do not provide the exact number of bytes required by t...
static const int ErrorInvalidImage
This property view does not have a valid image index.
static const int ErrorInvalidChannels
The channels of this property texture property are invalid. Channels must be in the range 0-N,...
static const int ErrorInvalidBytesPerChannel
This property uses an image with multi-byte channels. Only single-byte channels are supported.
static const int ErrorInvalidSampler
This property view does not have a valid sampler index.
static const int ErrorInvalidPropertyTexture
This property view was initialized from an invalid PropertyTexture.
static const int ErrorInvalidTexture
This property view does not have a valid texture index.
static const int ErrorEmptyImage
This property is viewing an empty image.
const std::string & getSwizzle() const noexcept
Gets this property's channels as a swizzle string.
std::optional< PropertyValueViewToCopy< ElementType > > get(double u, double v) const noexcept
Gets the value of the property for the given texture coordinates with all value transforms applied....
PropertyTexturePropertyView(PropertyViewStatusType status) noexcept
Constructs an invalid instance for an erroneous property.
PropertyTexturePropertyView(const PropertyTextureProperty &property, const ClassProperty &classProperty, const CesiumGltf::Enum *pEnumDefinition, const Sampler &sampler, const ImageAsset &image, const TextureViewOptions &options=TextureViewOptions()) noexcept
Construct a view of the data specified by a PropertyTextureProperty.
PropertyValueViewToCopy< ElementType > getRaw(double u, double v) const noexcept
Gets the raw value of the property for the given texture coordinates. The sampler's wrapping mode wil...
PropertyTexturePropertyView(const ClassProperty &classProperty) noexcept
Constructs an instance of an empty property that specifies a default value. Although this property ha...
PropertyTexturePropertyView() noexcept
Constructs an invalid instance for a non-existent property.
PropertyTexturePropertyView(const PropertyTextureProperty &property, const ClassProperty &classProperty, const Sampler &sampler, const ImageAsset &image, const TextureViewOptions &options=TextureViewOptions()) noexcept
Construct a view of the data specified by a PropertyTextureProperty.
const std::vector< int64_t > & getChannels() const noexcept
Gets the channels of this property texture property.
PropertyValueViewToCopy< ElementType > getRaw(double u, double v) const noexcept
Gets the raw value of the property for the given texture coordinates. The sampler's wrapping mode wil...
PropertyTexturePropertyView(const ClassProperty &classProperty) noexcept
Constructs an instance of an empty property that specifies a default value. Although this property ha...
const std::string & getSwizzle() const noexcept
Gets this property's channels as a swizzle string.
PropertyTexturePropertyView() noexcept
Constructs an invalid instance for a non-existent property.
PropertyTexturePropertyView(PropertyViewStatusType status) noexcept
Constructs an invalid instance for an erroneous property.
std::optional< PropertyValueViewToCopy< NormalizedType > > get(double u, double v) const noexcept
Gets the value of the property for the given texture coordinates with all value transforms applied....
PropertyTexturePropertyView(const PropertyTextureProperty &property, const ClassProperty &classProperty, const Sampler &sampler, const ImageAsset &image, const TextureViewOptions &options=TextureViewOptions()) noexcept
Construct a view of the data specified by a PropertyTextureProperty.
const std::vector< int64_t > & getChannels() const noexcept
Gets the channels of this property texture property.
A view of the data specified by a PropertyTextureProperty.
Indicates the status of a property view.
static const PropertyViewStatusType Valid
This property view is valid and ready to use.
static const PropertyViewStatusType ErrorNonexistentProperty
This property view is trying to view a property that does not exist.
static const PropertyViewStatusType EmptyPropertyWithDefault
This property view does not contain any data, but specifies a default value. This happens when a clas...
Represents a metadata property in EXT_structural_metadata.
A view into the texture data of a single texture from a Model.
Definition TextureView.h:98
Classes for working with glTF models.
PropertyArrayCopy< glm::vec< N, double > > transformNormalizedVecNArray(const PropertyArrayView< glm::vec< N, T > > &value, const std::optional< PropertyArrayView< glm::vec< N, double > > > &offset, const std::optional< PropertyArrayView< glm::vec< N, double > > > &scale)
Normalizes each element of an array of vectors and transforms them by optional offset and scale facto...
ElementType assembleVecNValue(const std::span< uint8_t > bytes) noexcept
Attempts to obtain a vector value from the given span of bytes.
PropertyArrayCopy< NormalizedType > transformNormalizedArray(const PropertyArrayView< T > &value, const std::optional< PropertyArrayView< NormalizedType > > &offset, const std::optional< PropertyArrayView< NormalizedType > > &scale)
Normalizes each element of an array of values and transforms them by optional offset and scale factor...
T transformValue(const T &value, const std::optional< T > &offset, const std::optional< T > &scale)
Transforms the value by optional offset and scale factors.
double normalize(T value)
Normalizes the given value between [0, 1] if unsigned or [-1, 1] if signed, based on the type's maxim...
@ ErrorEmptyImage
This texture is viewing an empty image.
@ ErrorUninitialized
This texture view has not yet been initialized.
@ Valid
This texture view is valid and ready to use.
@ ErrorInvalidSampler
This texture view does not have a valid sampler index.
@ ErrorInvalidImage
This texture view does not have a valid image index.
@ ErrorInvalidBytesPerChannel
The image for this texture has channels that take up more than a byte. Only single-byte channels are ...
@ ErrorInvalidTexture
This texture view does not have a valid texture index.
PropertyArrayCopy< T > transformArray(const PropertyArrayView< T > &value, const std::optional< PropertyArrayView< T > > &offset, const std::optional< PropertyArrayView< T > > &scale)
Transforms each element of an array of values by optional offset and scale factors....
std::conditional_t< IsMetadataNumericArray< T >::value, PropertyArrayCopy< typename MetadataArrayType< T >::type >, T > PropertyValueViewToCopy
Transforms a property value type from a view to an equivalent type that owns the data it is viewing....
ElementType assembleScalarValue(const std::span< uint8_t > bytes) noexcept
Attempts to obtain a scalar value from the given span of bytes.
PropertyArrayCopy< T > assembleArrayValue(const std::span< uint8_t > bytes) noexcept
Attempts to obtain an array value from the given span of bytes.
PropertyValueViewToCopy< ElementType > assembleValueFromChannels(const std::span< uint8_t > bytes) noexcept
Assembles the given type from the provided channel values of sampling a texture.
int32_t PropertyViewStatusType
The type used for fields of PropertyViewStatus.
std::optional< CesiumUtility::JsonValue > defaultProperty
A default value to use when encountering a noData value or an omitted property. The value is given in...
This class is not meant to be instantiated directly. Use Enum instead.
Definition Enum.h:12
glTF extension that enables shifting and scaling UV coordinates on a per-texture basis
A 2D image asset, including its pixel data. The image may have mipmaps, and it may be encoded in a GP...
Definition ImageAsset.h:34
Check if a C++ type can be represented as an array.
Check if a C++ type can be represented as an integer property type.
Check if a C++ type can be represented as an array of numeric elements property type.
Check if a C++ type can be represented as a numeric property, i.e. a scalar / vecN / matN type.
Check if a C++ type can be represented as a scalar property type.
Check if a C++ type can be represented as a vecN type.
void type
The component type of this metadata array.
A texture containing property values.
Texture sampler properties for filtering and wrapping modes.
Definition Sampler.h:14
Describes options for constructing a view on a glTF texture.
Definition TextureView.h:18
The number of dimensions that this type contains.
Convert an integer numeric type to the corresponding representation as a double type....