cesium-native 0.46.0
Loading...
Searching...
No Matches
PropertyTableView.h
1#pragma once
2
3#include <CesiumGltf/ExtensionModelExtStructuralMetadata.h>
4#include <CesiumGltf/Model.h>
5#include <CesiumGltf/PropertyTablePropertyView.h>
6#include <CesiumGltf/PropertyType.h>
7#include <CesiumGltf/PropertyTypeTraits.h>
8
9#include <glm/common.hpp>
10
11#include <optional>
12
13namespace CesiumGltf {
14
47
56public:
63 PropertyTableView(const Model& model, const PropertyTable& propertyTable);
64
73 PropertyTableViewStatus status() const noexcept { return _status; }
74
79 const std::optional<std::string>& name() const noexcept {
80 return _pPropertyTable->name;
81 }
82
89 int64_t size() const noexcept {
90 return _status == PropertyTableViewStatus::Valid ? _pPropertyTable->count
91 : 0;
92 }
93
100 const Class* getClass() const noexcept { return _pClass; }
101
109 const ClassProperty* getClassProperty(const std::string& propertyId) const;
110
137 template <typename T, bool Normalized = false>
139 getPropertyView(const std::string& propertyId) const {
140 if (this->size() <= 0) {
143 }
144
145 const ClassProperty* pClassProperty = getClassProperty(propertyId);
146 if (!pClassProperty) {
149 }
150
151 return getPropertyViewImpl<T, Normalized>(propertyId, *pClassProperty);
152 }
153
176 template <typename Callback>
177 void
178 getPropertyView(const std::string& propertyId, Callback&& callback) const {
179 if (this->size() <= 0) {
180 callback(
181 propertyId,
184 return;
185 }
186
187 const ClassProperty* pClassProperty = getClassProperty(propertyId);
188 if (!pClassProperty) {
189 callback(
190 propertyId,
193 return;
194 }
195
196 PropertyType type = convertStringToPropertyType(pClassProperty->type);
198
199 // Handle enum component type indirection
200 if (type == PropertyType::Enum && pClassProperty->enumType) {
201 const auto& enumDefinitionIt =
202 this->_pEnumDefinitions->find(*pClassProperty->enumType);
203 if (enumDefinitionIt == this->_pEnumDefinitions->end()) {
204 callback(
205 propertyId,
208 return;
209 }
210
212 enumDefinitionIt->second.valueType);
213
214 if (componentType == PropertyComponentType::Float32 ||
215 componentType == PropertyComponentType::Float64) {
216 callback(
217 propertyId,
220 return;
221 }
222
223 } else if (type == PropertyType::Enum) {
224 callback(
225 propertyId,
228 return;
229 } else if (pClassProperty->componentType) {
230 componentType =
232 }
233
234 bool normalized = pClassProperty->normalized;
235 if (normalized && (!isPropertyComponentTypeInteger(componentType) ||
236 type == PropertyType::Enum)) {
237 // Only integer components may be normalized.
238 callback(
239 propertyId,
242 return;
243 }
244
245 if (pClassProperty->array) {
246 if (normalized) {
247 getArrayPropertyViewImpl<Callback, true>(
248 propertyId,
249 *pClassProperty,
250 type,
251 componentType,
252 std::forward<Callback>(callback));
253 } else {
254 getArrayPropertyViewImpl<Callback, false>(
255 propertyId,
256 *pClassProperty,
257 type,
258 componentType,
259 std::forward<Callback>(callback));
260 }
261 return;
262 }
263
264 if (type == PropertyType::Scalar || type == PropertyType::Enum) {
265 if (normalized) {
266 getScalarPropertyViewImpl<Callback, true>(
267 propertyId,
268 *pClassProperty,
269 componentType,
270 std::forward<Callback>(callback));
271 } else {
272 getScalarPropertyViewImpl<Callback, false>(
273 propertyId,
274 *pClassProperty,
275 componentType,
276 std::forward<Callback>(callback));
277 }
278 return;
279 }
280
281 if (isPropertyTypeVecN(type)) {
282 if (normalized) {
283 getVecNPropertyViewImpl<Callback, true>(
284 propertyId,
285 *pClassProperty,
286 type,
287 componentType,
288 std::forward<Callback>(callback));
289 } else {
290 getVecNPropertyViewImpl<Callback, false>(
291 propertyId,
292 *pClassProperty,
293 type,
294 componentType,
295 std::forward<Callback>(callback));
296 }
297 return;
298 }
299
300 if (isPropertyTypeMatN(type)) {
301 if (normalized) {
302 getMatNPropertyViewImpl<Callback, true>(
303 propertyId,
304 *pClassProperty,
305 type,
306 componentType,
307 std::forward<Callback>(callback));
308 } else {
309 getMatNPropertyViewImpl<Callback, false>(
310 propertyId,
311 *pClassProperty,
312 type,
313 componentType,
314 std::forward<Callback>(callback));
315 }
316 return;
317 }
318
319 if (type == PropertyType::String) {
320 callback(
321 propertyId,
322 getPropertyViewImpl<std::string_view, false>(
323 propertyId,
324 *pClassProperty));
325 return;
326 }
327
328 if (type == PropertyType::Boolean) {
329 callback(
330 propertyId,
331 getPropertyViewImpl<bool, false>(propertyId, *pClassProperty));
332 return;
333 }
334
335 callback(
336 propertyId,
339 }
340
362 template <typename Callback> void forEachProperty(Callback&& callback) const {
363 for (const auto& property : this->_pClass->properties) {
364 getPropertyView(property.first, std::forward<Callback>(callback));
365 }
366 }
367
368private:
369 template <typename Callback, bool Normalized>
370 void getScalarArrayPropertyViewImpl(
371 const std::string& propertyId,
372 const ClassProperty& classProperty,
373 PropertyComponentType componentType,
374 Callback&& callback) const {
375 switch (componentType) {
377 callback(
378 propertyId,
379 getPropertyViewImpl<PropertyArrayView<int8_t>, Normalized>(
380 propertyId,
381 classProperty));
382 break;
384 callback(
385 propertyId,
386 getPropertyViewImpl<PropertyArrayView<uint8_t>, Normalized>(
387 propertyId,
388 classProperty));
389 break;
391 callback(
392 propertyId,
393 getPropertyViewImpl<PropertyArrayView<int16_t>, Normalized>(
394 propertyId,
395 classProperty));
396 break;
398 callback(
399 propertyId,
400 getPropertyViewImpl<PropertyArrayView<uint16_t>, Normalized>(
401 propertyId,
402 classProperty));
403 break;
405 callback(
406 propertyId,
407 getPropertyViewImpl<PropertyArrayView<int32_t>, Normalized>(
408 propertyId,
409 classProperty));
410 break;
412 callback(
413 propertyId,
414 getPropertyViewImpl<PropertyArrayView<uint32_t>, Normalized>(
415 propertyId,
416 classProperty));
417 break;
419 callback(
420 propertyId,
421 getPropertyViewImpl<PropertyArrayView<int64_t>, Normalized>(
422 propertyId,
423 classProperty));
424 break;
426 callback(
427 propertyId,
428 getPropertyViewImpl<PropertyArrayView<uint64_t>, Normalized>(
429 propertyId,
430 classProperty));
431 break;
433 callback(
434 propertyId,
435 getPropertyViewImpl<PropertyArrayView<float>, false>(
436 propertyId,
437 classProperty));
438 break;
440 callback(
441 propertyId,
442 getPropertyViewImpl<PropertyArrayView<double>, false>(
443 propertyId,
444 classProperty));
445 break;
446 default:
447 callback(
448 propertyId,
451 break;
452 }
453 }
454
455 template <typename Callback, glm::length_t N, bool Normalized>
456 void getVecNArrayPropertyViewImpl(
457 const std::string& propertyId,
458 const ClassProperty& classProperty,
459 PropertyComponentType componentType,
460 Callback&& callback) const {
461 switch (componentType) {
463 callback(
464 propertyId,
465 getPropertyViewImpl<
466 PropertyArrayView<glm::vec<N, int8_t>>,
467 Normalized>(propertyId, classProperty));
468 break;
470 callback(
471 propertyId,
472 getPropertyViewImpl<
473 PropertyArrayView<glm::vec<N, uint8_t>>,
474 Normalized>(propertyId, classProperty));
475 break;
477 callback(
478 propertyId,
479 getPropertyViewImpl<
480 PropertyArrayView<glm::vec<N, int16_t>>,
481 Normalized>(propertyId, classProperty));
482 break;
484 callback(
485 propertyId,
486 getPropertyViewImpl<
487 PropertyArrayView<glm::vec<N, uint16_t>>,
488 Normalized>(propertyId, classProperty));
489 break;
491 callback(
492 propertyId,
493 getPropertyViewImpl<
494 PropertyArrayView<glm::vec<N, int32_t>>,
495 Normalized>(propertyId, classProperty));
496 break;
498 callback(
499 propertyId,
500 getPropertyViewImpl<
501 PropertyArrayView<glm::vec<N, uint32_t>>,
502 Normalized>(propertyId, classProperty));
503 break;
505 callback(
506 propertyId,
507 getPropertyViewImpl<
508 PropertyArrayView<glm::vec<N, int64_t>>,
509 Normalized>(propertyId, classProperty));
510 break;
512 callback(
513 propertyId,
514 getPropertyViewImpl<
515 PropertyArrayView<glm::vec<N, uint64_t>>,
516 Normalized>(propertyId, classProperty));
517 break;
519 callback(
520 propertyId,
521 getPropertyViewImpl<PropertyArrayView<glm::vec<N, float>>, false>(
522 propertyId,
523 classProperty));
524 break;
526 callback(
527 propertyId,
528 getPropertyViewImpl<PropertyArrayView<glm::vec<N, double>>, false>(
529 propertyId,
530 classProperty));
531 break;
532 default:
533 callback(
534 propertyId,
535 PropertyTablePropertyView<uint8_t>(
537 break;
538 }
539 }
540
541 template <typename Callback, bool Normalized>
542 void getVecNArrayPropertyViewImpl(
543 const std::string& propertyId,
544 const ClassProperty& classProperty,
545 PropertyType type,
546 PropertyComponentType componentType,
547 Callback&& callback) const {
548 glm::length_t N = getDimensionsFromPropertyType(type);
549 switch (N) {
550 case 2:
551 getVecNArrayPropertyViewImpl<Callback, 2, Normalized>(
552 propertyId,
553 classProperty,
554 componentType,
555 std::forward<Callback>(callback));
556 break;
557 case 3:
558 getVecNArrayPropertyViewImpl<Callback, 3, Normalized>(
559 propertyId,
560 classProperty,
561 componentType,
562 std::forward<Callback>(callback));
563 break;
564 case 4:
565 getVecNArrayPropertyViewImpl<Callback, 4, Normalized>(
566 propertyId,
567 classProperty,
568 componentType,
569 std::forward<Callback>(callback));
570 break;
571 default:
572 callback(
573 propertyId,
574 PropertyTablePropertyView<uint8_t>(
576 break;
577 }
578 }
579
580 template <typename Callback, glm::length_t N, bool Normalized>
581 void getMatNArrayPropertyViewImpl(
582 const std::string& propertyId,
583 const ClassProperty& classProperty,
584 PropertyComponentType componentType,
585 Callback&& callback) const {
586 switch (componentType) {
588 callback(
589 propertyId,
590 getPropertyViewImpl<
591 PropertyArrayView<glm::mat<N, N, int8_t>>,
592 Normalized>(propertyId, classProperty));
593 break;
595 callback(
596 propertyId,
597 getPropertyViewImpl<
598 PropertyArrayView<glm::mat<N, N, uint8_t>>,
599 Normalized>(propertyId, classProperty));
600 break;
602 callback(
603 propertyId,
604 getPropertyViewImpl<
605 PropertyArrayView<glm::mat<N, N, int16_t>>,
606 Normalized>(propertyId, classProperty));
607 break;
609 callback(
610 propertyId,
611 getPropertyViewImpl<
612 PropertyArrayView<glm::mat<N, N, uint16_t>>,
613 Normalized>(propertyId, classProperty));
614 break;
616 callback(
617 propertyId,
618 getPropertyViewImpl<
619 PropertyArrayView<glm::mat<N, N, int32_t>>,
620 Normalized>(propertyId, classProperty));
621 break;
623 callback(
624 propertyId,
625 getPropertyViewImpl<
626 PropertyArrayView<glm::mat<N, N, uint32_t>>,
627 Normalized>(propertyId, classProperty));
628 break;
630 callback(
631 propertyId,
632 getPropertyViewImpl<
633 PropertyArrayView<glm::mat<N, N, int64_t>>,
634 Normalized>(propertyId, classProperty));
635 break;
637 callback(
638 propertyId,
639 getPropertyViewImpl<
640 PropertyArrayView<glm::mat<N, N, uint64_t>>,
641 Normalized>(propertyId, classProperty));
642 break;
644 callback(
645 propertyId,
646 getPropertyViewImpl<PropertyArrayView<glm::mat<N, N, float>>, false>(
647 propertyId,
648 classProperty));
649 break;
651 callback(
652 propertyId,
653 getPropertyViewImpl<PropertyArrayView<glm::mat<N, N, double>>, false>(
654 propertyId,
655 classProperty));
656 break;
657 default:
658 callback(
659 propertyId,
660 PropertyTablePropertyView<uint8_t>(
662 break;
663 }
664 }
665
666 template <typename Callback, bool Normalized>
667 void getMatNArrayPropertyViewImpl(
668 const std::string& propertyId,
669 const ClassProperty& classProperty,
670 PropertyType type,
671 PropertyComponentType componentType,
672 Callback&& callback) const {
673 const glm::length_t N = getDimensionsFromPropertyType(type);
674 switch (N) {
675 case 2:
676 getMatNArrayPropertyViewImpl<Callback, 2, Normalized>(
677 propertyId,
678 classProperty,
679 componentType,
680 std::forward<Callback>(callback));
681 break;
682 case 3:
683 getMatNArrayPropertyViewImpl<Callback, 3, Normalized>(
684 propertyId,
685 classProperty,
686 componentType,
687 std::forward<Callback>(callback));
688 break;
689 case 4:
690 getMatNArrayPropertyViewImpl<Callback, 4, Normalized>(
691 propertyId,
692 classProperty,
693 componentType,
694 std::forward<Callback>(callback));
695 break;
696 default:
697 callback(
698 propertyId,
699 PropertyTablePropertyView<uint8_t>(
701 break;
702 }
703 }
704
705 template <typename Callback, bool Normalized>
706 void getArrayPropertyViewImpl(
707 const std::string& propertyId,
708 const ClassProperty& classProperty,
709 PropertyType type,
710 PropertyComponentType componentType,
711 Callback&& callback) const {
712 if (type == PropertyType::Scalar || type == PropertyType::Enum) {
713 getScalarArrayPropertyViewImpl<Callback, Normalized>(
714 propertyId,
715 classProperty,
716 componentType,
717 std::forward<Callback>(callback));
718 } else if (isPropertyTypeVecN(type)) {
719 getVecNArrayPropertyViewImpl<Callback, Normalized>(
720 propertyId,
721 classProperty,
722 type,
723 componentType,
724 std::forward<Callback>(callback));
725 } else if (isPropertyTypeMatN(type)) {
726 getMatNArrayPropertyViewImpl<Callback, Normalized>(
727 propertyId,
728 classProperty,
729 type,
730 componentType,
731 std::forward<Callback>(callback));
732 } else if (type == PropertyType::Boolean) {
733 callback(
734 propertyId,
735 getPropertyViewImpl<PropertyArrayView<bool>, false>(
736 propertyId,
737 classProperty));
738
739 } else if (type == PropertyType::String) {
740 callback(
741 propertyId,
742 getPropertyViewImpl<PropertyArrayView<std::string_view>, false>(
743 propertyId,
744 classProperty));
745 } else {
746 callback(
747 propertyId,
748 PropertyTablePropertyView<uint8_t>(
750 }
751 }
752
753 template <typename Callback, glm::length_t N, bool Normalized>
754 void getVecNPropertyViewImpl(
755 const std::string& propertyId,
756 const ClassProperty& classProperty,
757 PropertyComponentType componentType,
758 Callback&& callback) const {
759
760 switch (componentType) {
762 callback(
763 propertyId,
764 getPropertyViewImpl<glm::vec<N, int8_t>, Normalized>(
765 propertyId,
766 classProperty));
767 break;
769 callback(
770 propertyId,
771 getPropertyViewImpl<glm::vec<N, uint8_t>, Normalized>(
772 propertyId,
773 classProperty));
774 break;
776 callback(
777 propertyId,
778 getPropertyViewImpl<glm::vec<N, int16_t>, Normalized>(
779 propertyId,
780 classProperty));
781 break;
783 callback(
784 propertyId,
785 getPropertyViewImpl<glm::vec<N, uint16_t>, Normalized>(
786 propertyId,
787 classProperty));
788 break;
790 callback(
791 propertyId,
792 getPropertyViewImpl<glm::vec<N, int32_t>, Normalized>(
793 propertyId,
794 classProperty));
795 break;
797 callback(
798 propertyId,
799 getPropertyViewImpl<glm::vec<N, uint32_t>, Normalized>(
800 propertyId,
801 classProperty));
802 break;
804 callback(
805 propertyId,
806 getPropertyViewImpl<glm::vec<N, int64_t>, Normalized>(
807 propertyId,
808 classProperty));
809 break;
811 callback(
812 propertyId,
813 getPropertyViewImpl<glm::vec<N, uint64_t>, Normalized>(
814 propertyId,
815 classProperty));
816 break;
818 callback(
819 propertyId,
820 getPropertyViewImpl<glm::vec<N, float>, false>(
821 propertyId,
822 classProperty));
823 break;
825 callback(
826 propertyId,
827 getPropertyViewImpl<glm::vec<N, double>, false>(
828 propertyId,
829 classProperty));
830 break;
831 default:
832 callback(
833 propertyId,
834 PropertyTablePropertyView<uint8_t>(
836 break;
837 }
838 }
839
840 template <typename Callback, bool Normalized>
841 void getVecNPropertyViewImpl(
842 const std::string& propertyId,
843 const ClassProperty& classProperty,
844 PropertyType type,
845 PropertyComponentType componentType,
846 Callback&& callback) const {
847 const glm::length_t N = getDimensionsFromPropertyType(type);
848 switch (N) {
849 case 2:
850 getVecNPropertyViewImpl<Callback, 2, Normalized>(
851 propertyId,
852 classProperty,
853 componentType,
854 std::forward<Callback>(callback));
855 break;
856 case 3:
857 getVecNPropertyViewImpl<Callback, 3, Normalized>(
858 propertyId,
859 classProperty,
860 componentType,
861 std::forward<Callback>(callback));
862 break;
863 case 4:
864 getVecNPropertyViewImpl<Callback, 4, Normalized>(
865 propertyId,
866 classProperty,
867 componentType,
868 std::forward<Callback>(callback));
869 break;
870 default:
871 callback(
872 propertyId,
873 PropertyTablePropertyView<uint8_t>(
875 break;
876 }
877 }
878
879 template <typename Callback, glm::length_t N, bool Normalized>
880 void getMatNPropertyViewImpl(
881 const std::string& propertyId,
882 const ClassProperty& classProperty,
883 PropertyComponentType componentType,
884 Callback&& callback) const {
885 switch (componentType) {
887 callback(
888 propertyId,
889 getPropertyViewImpl<glm::mat<N, N, int8_t>, Normalized>(
890 propertyId,
891 classProperty));
892 break;
894 callback(
895 propertyId,
896 getPropertyViewImpl<glm::mat<N, N, uint8_t>, Normalized>(
897 propertyId,
898 classProperty));
899 break;
901 callback(
902 propertyId,
903 getPropertyViewImpl<glm::mat<N, N, int16_t>, Normalized>(
904 propertyId,
905 classProperty));
906 break;
908 callback(
909 propertyId,
910 getPropertyViewImpl<glm::mat<N, N, uint16_t>, Normalized>(
911 propertyId,
912 classProperty));
913 break;
915 callback(
916 propertyId,
917 getPropertyViewImpl<glm::mat<N, N, int32_t>, Normalized>(
918 propertyId,
919 classProperty));
920 break;
922 callback(
923 propertyId,
924 getPropertyViewImpl<glm::mat<N, N, uint32_t>, Normalized>(
925 propertyId,
926 classProperty));
927 break;
929 callback(
930 propertyId,
931 getPropertyViewImpl<glm::mat<N, N, int64_t>, Normalized>(
932 propertyId,
933 classProperty));
934 break;
936 callback(
937 propertyId,
938 getPropertyViewImpl<glm::mat<N, N, uint64_t>, Normalized>(
939 propertyId,
940 classProperty));
941 break;
943 callback(
944 propertyId,
945 getPropertyViewImpl<glm::mat<N, N, float>, false>(
946 propertyId,
947 classProperty));
948 break;
950 callback(
951 propertyId,
952 getPropertyViewImpl<glm::mat<N, N, double>, false>(
953 propertyId,
954 classProperty));
955 break;
956 default:
957 callback(
958 propertyId,
959 PropertyTablePropertyView<uint8_t>(
961 break;
962 }
963 }
964
965 template <typename Callback, bool Normalized>
966 void getMatNPropertyViewImpl(
967 const std::string& propertyId,
968 const ClassProperty& classProperty,
969 PropertyType type,
970 PropertyComponentType componentType,
971 Callback&& callback) const {
972 glm::length_t N = getDimensionsFromPropertyType(type);
973 switch (N) {
974 case 2:
975 getMatNPropertyViewImpl<Callback, 2, Normalized>(
976 propertyId,
977 classProperty,
978 componentType,
979 std::forward<Callback>(callback));
980 break;
981 case 3:
982 getMatNPropertyViewImpl<Callback, 3, Normalized>(
983 propertyId,
984 classProperty,
985 componentType,
986 std::forward<Callback>(callback));
987 break;
988 case 4:
989 getMatNPropertyViewImpl<Callback, 4, Normalized>(
990 propertyId,
991 classProperty,
992 componentType,
993 std::forward<Callback>(callback));
994 break;
995 default:
996 callback(
997 propertyId,
998 PropertyTablePropertyView<uint8_t>(
1000 break;
1001 }
1002 }
1003
1004 template <typename Callback, bool Normalized>
1005 void getScalarPropertyViewImpl(
1006 const std::string& propertyId,
1007 const ClassProperty& classProperty,
1008 PropertyComponentType componentType,
1009 Callback&& callback) const {
1010 switch (componentType) {
1012 callback(
1013 propertyId,
1014 getPropertyViewImpl<int8_t, Normalized>(propertyId, classProperty));
1015 return;
1017 callback(
1018 propertyId,
1019 getPropertyViewImpl<uint8_t, Normalized>(propertyId, classProperty));
1020 return;
1022 callback(
1023 propertyId,
1024 getPropertyViewImpl<int16_t, Normalized>(propertyId, classProperty));
1025 return;
1027 callback(
1028 propertyId,
1029 getPropertyViewImpl<uint16_t, Normalized>(propertyId, classProperty));
1030 break;
1032 callback(
1033 propertyId,
1034 getPropertyViewImpl<int32_t, Normalized>(propertyId, classProperty));
1035 break;
1037 callback(
1038 propertyId,
1039 getPropertyViewImpl<uint32_t, Normalized>(propertyId, classProperty));
1040 break;
1042 callback(
1043 propertyId,
1044 getPropertyViewImpl<int64_t, Normalized>(propertyId, classProperty));
1045 break;
1047 callback(
1048 propertyId,
1049 getPropertyViewImpl<uint64_t, Normalized>(propertyId, classProperty));
1050 break;
1052 callback(
1053 propertyId,
1054 getPropertyViewImpl<float, false>(propertyId, classProperty));
1055 break;
1057 callback(
1058 propertyId,
1059 getPropertyViewImpl<double, false>(propertyId, classProperty));
1060 break;
1061 default:
1062 callback(
1063 propertyId,
1064 PropertyTablePropertyView<uint8_t>(
1066 break;
1067 }
1068 }
1069
1070 template <typename T, bool Normalized>
1071 PropertyTablePropertyView<T, Normalized> getPropertyViewImpl(
1072 const std::string& propertyId,
1073 const ClassProperty& classProperty) const {
1074 auto propertyTablePropertyIter =
1075 _pPropertyTable->properties.find(propertyId);
1076 if (propertyTablePropertyIter == _pPropertyTable->properties.end()) {
1077 if (!classProperty.required && classProperty.defaultProperty) {
1078 // If the property was omitted from the property table, it is still
1079 // technically valid if it specifies a default value. Create a view that
1080 // just returns the default value.
1081 return PropertyTablePropertyView<T, Normalized>(
1082 classProperty,
1083 _pPropertyTable->count);
1084 }
1085
1086 // Otherwise, the property is erroneously nonexistent.
1087 return PropertyTablePropertyView<T, Normalized>(
1089 }
1090
1091 const PropertyTableProperty& propertyTableProperty =
1092 propertyTablePropertyIter->second;
1093
1094 if constexpr (IsMetadataNumeric<T>::value || IsMetadataBoolean<T>::value) {
1095 return getNumericOrBooleanPropertyValues<T, Normalized>(
1096 classProperty,
1097 propertyTableProperty);
1098 }
1099
1100 if constexpr (IsMetadataString<T>::value) {
1101 return getStringPropertyValues(classProperty, propertyTableProperty);
1102 }
1103
1104 if constexpr (IsMetadataBooleanArray<T>::value) {
1105 return getBooleanArrayPropertyValues(
1106 classProperty,
1107 propertyTableProperty);
1108 }
1109
1110 if constexpr (IsMetadataNumericArray<T>::value) {
1111 return getNumericArrayPropertyValues<
1113 Normalized>(classProperty, propertyTableProperty);
1114 }
1115
1116 if constexpr (IsMetadataStringArray<T>::value) {
1117 return getStringArrayPropertyValues(classProperty, propertyTableProperty);
1118 }
1119 }
1120
1121 template <typename T, bool Normalized>
1122 PropertyTablePropertyView<T, Normalized> getNumericOrBooleanPropertyValues(
1123 const ClassProperty& classProperty,
1124 const PropertyTableProperty& propertyTableProperty) const {
1125 if (classProperty.array) {
1126 return PropertyTablePropertyView<T, Normalized>(
1128 }
1129
1130 const PropertyType type = convertStringToPropertyType(classProperty.type);
1131 if (!canRepresentPropertyType<T>(type)) {
1132 return PropertyTablePropertyView<T, Normalized>(
1134 }
1135
1137 classProperty.componentType.value_or(""));
1138 const CesiumGltf::Enum* pEnumDefinition = nullptr;
1139 if (type == PropertyType::Enum && classProperty.enumType) {
1140 const auto& enumDefinitionIt =
1141 this->_pEnumDefinitions->find(*classProperty.enumType);
1142 if (enumDefinitionIt == this->_pEnumDefinitions->end()) {
1143 return PropertyTablePropertyView<T, Normalized>(
1145 }
1146
1148 enumDefinitionIt->second.valueType);
1149 if (componentType == PropertyComponentType::Float32 ||
1150 componentType == PropertyComponentType::Float64) {
1151 return PropertyTablePropertyView<T, Normalized>(
1153 }
1154 pEnumDefinition = &enumDefinitionIt->second;
1155 } else if (type == PropertyType::Enum) {
1156 return PropertyTablePropertyView<T, Normalized>(
1158 }
1159
1160 if (TypeToPropertyType<T>::component != componentType) {
1161 return PropertyTablePropertyView<T, Normalized>(
1163 }
1164
1165 if (classProperty.normalized != Normalized ||
1166 (classProperty.normalized && type == PropertyType::Enum)) {
1167 return PropertyTablePropertyView<T, Normalized>(
1169 }
1170
1171 std::span<const std::byte> values;
1172 const auto status = getBufferSafe(propertyTableProperty.values, values);
1174 return PropertyTablePropertyView<T, Normalized>(status);
1175 }
1176
1177 if (values.size() % sizeof(T) != 0) {
1178 return PropertyTablePropertyView<T, Normalized>(
1179 PropertyTablePropertyViewStatus::
1180 ErrorBufferViewSizeNotDivisibleByTypeSize);
1181 }
1182
1183 size_t maxRequiredBytes = 0;
1184 if (IsMetadataBoolean<T>::value) {
1185 maxRequiredBytes = static_cast<size_t>(
1186 glm::ceil(static_cast<double>(_pPropertyTable->count) / 8.0));
1187 } else {
1188 maxRequiredBytes =
1189 static_cast<size_t>(_pPropertyTable->count) * sizeof(T);
1190 }
1191
1192 if (values.size() < maxRequiredBytes) {
1193 return PropertyTablePropertyView<T, Normalized>(
1194 PropertyTablePropertyViewStatus::
1195 ErrorBufferViewSizeDoesNotMatchPropertyTableCount);
1196 }
1197
1198 if constexpr (!Normalized) {
1199 return PropertyTablePropertyView<T, Normalized>(
1200 propertyTableProperty,
1201 classProperty,
1202 pEnumDefinition,
1203 _pPropertyTable->count,
1204 values);
1205 } else {
1206 return PropertyTablePropertyView<T, Normalized>(
1207 propertyTableProperty,
1208 classProperty,
1209 _pPropertyTable->count,
1210 values);
1211 }
1212 }
1213
1214 PropertyTablePropertyView<std::string_view> getStringPropertyValues(
1215 const ClassProperty& classProperty,
1216 const PropertyTableProperty& propertyTableProperty) const;
1217
1218 PropertyTablePropertyView<PropertyArrayView<bool>>
1219 getBooleanArrayPropertyValues(
1220 const ClassProperty& classProperty,
1221 const PropertyTableProperty& propertyTableProperty) const;
1222
1223 template <typename T, bool Normalized>
1224 PropertyTablePropertyView<PropertyArrayView<T>, Normalized>
1225 getNumericArrayPropertyValues(
1226 const ClassProperty& classProperty,
1227 const PropertyTableProperty& propertyTableProperty) const {
1228 if (!classProperty.array) {
1229 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1231 }
1232
1233 const PropertyType type = convertStringToPropertyType(classProperty.type);
1234 if (!canRepresentPropertyType<T>(type)) {
1235 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1237 }
1238
1240 classProperty.componentType.value_or(""));
1241 const CesiumGltf::Enum* pEnumDefinition = nullptr;
1242 if (type == PropertyType::Enum && classProperty.enumType) {
1243 const auto& enumDefinitionIt =
1244 this->_pEnumDefinitions->find(*classProperty.enumType);
1245 if (enumDefinitionIt == this->_pEnumDefinitions->end()) {
1246 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1248 }
1249
1251 enumDefinitionIt->second.valueType);
1252
1253 if (componentType == PropertyComponentType::Float32 ||
1254 componentType == PropertyComponentType::Float64) {
1255 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1257 }
1258 pEnumDefinition = &enumDefinitionIt->second;
1259 } else if (type == PropertyType::Enum) {
1260 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1262 }
1263
1264 if (TypeToPropertyType<T>::component != componentType) {
1265 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1267 }
1268
1269 if (classProperty.normalized != Normalized) {
1270 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1272 }
1273
1274 std::span<const std::byte> values;
1275 auto status = getBufferSafe(propertyTableProperty.values, values);
1277 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1278 status);
1279 }
1280
1281 if (values.size() % sizeof(T) != 0) {
1282 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1285 }
1286
1287 const int64_t fixedLengthArrayCount = classProperty.count.value_or(0);
1288 if (fixedLengthArrayCount > 0 && propertyTableProperty.arrayOffsets >= 0) {
1289 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1292 }
1293
1294 if (fixedLengthArrayCount <= 0 && propertyTableProperty.arrayOffsets < 0) {
1295 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1298 }
1299
1300 // Handle fixed-length arrays
1301 if (fixedLengthArrayCount > 0) {
1302 size_t maxRequiredBytes =
1303 static_cast<size_t>(_pPropertyTable->count * fixedLengthArrayCount) *
1304 sizeof(T);
1305
1306 if (values.size() < maxRequiredBytes) {
1307 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1310 }
1311
1312 if constexpr (!Normalized) {
1313 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1314 propertyTableProperty,
1315 classProperty,
1316 pEnumDefinition,
1317 _pPropertyTable->count,
1318 values);
1319 } else {
1320 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1321 propertyTableProperty,
1322 classProperty,
1323 _pPropertyTable->count,
1324 values);
1325 }
1326 }
1327
1328 // Handle variable-length arrays
1329 const PropertyComponentType arrayOffsetType =
1331 propertyTableProperty.arrayOffsetType);
1332 if (arrayOffsetType == PropertyComponentType::None) {
1333 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1335 }
1336
1337 constexpr bool checkBitsSize = false;
1338 std::span<const std::byte> arrayOffsets;
1339 status = getArrayOffsetsBufferSafe(
1340 propertyTableProperty.arrayOffsets,
1341 arrayOffsetType,
1342 values.size(),
1343 static_cast<size_t>(_pPropertyTable->count),
1344 checkBitsSize,
1345 arrayOffsets);
1347 return PropertyTablePropertyView<PropertyArrayView<T>, Normalized>(
1348 status);
1349 }
1350
1351 if constexpr (Normalized) {
1352 return PropertyTablePropertyView<PropertyArrayView<T>, true>(
1353 propertyTableProperty,
1354 classProperty,
1355 _pPropertyTable->count,
1356 values,
1357 arrayOffsets,
1358 arrayOffsetType);
1359 } else {
1360 return PropertyTablePropertyView<PropertyArrayView<T>, false>(
1361 propertyTableProperty,
1362 classProperty,
1363 pEnumDefinition,
1364 _pPropertyTable->count,
1365 values,
1366 arrayOffsets,
1367 std::span<const std::byte>(),
1368 arrayOffsetType,
1370 }
1371 }
1372
1373 PropertyTablePropertyView<PropertyArrayView<std::string_view>>
1374 getStringArrayPropertyValues(
1375 const ClassProperty& classProperty,
1376 const PropertyTableProperty& propertyTableProperty) const;
1377
1378 PropertyViewStatusType getBufferSafe(
1379 int32_t bufferView,
1380 std::span<const std::byte>& buffer) const noexcept;
1381
1382 PropertyViewStatusType getArrayOffsetsBufferSafe(
1383 int32_t arrayOffsetsBufferView,
1384 PropertyComponentType arrayOffsetType,
1385 size_t valuesBufferSize,
1386 size_t propertyTableCount,
1387 bool checkBitsSize,
1388 std::span<const std::byte>& arrayOffsetsBuffer) const noexcept;
1389
1390 PropertyViewStatusType getStringOffsetsBufferSafe(
1391 int32_t stringOffsetsBufferView,
1392 PropertyComponentType stringOffsetType,
1393 size_t valuesBufferSize,
1394 size_t propertyTableCount,
1395 std::span<const std::byte>& stringOffsetsBuffer) const noexcept;
1396
1397 const Model* _pModel;
1398 const PropertyTable* _pPropertyTable;
1399 const Class* _pClass;
1400 const std::unordered_map<std::string, CesiumGltf::Enum>* _pEnumDefinitions;
1402};
1403} // namespace CesiumGltf
A view on an array element of a PropertyTableProperty or PropertyTextureProperty.
static const PropertyViewStatusType ErrorBufferViewSizeDoesNotMatchPropertyTableCount
This property view has an invalid buffer view; its length does not match the size of the property tab...
static const PropertyViewStatusType ErrorArrayCountAndOffsetBufferCoexist
This array property view has both a fixed length and an offset buffer view defined.
static const PropertyViewStatusType ErrorArrayCountAndOffsetBufferDontExist
This array property view has neither a fixed length nor an offset buffer view defined.
static const PropertyViewStatusType ErrorInvalidPropertyTable
This property view was initialized from an invalid PropertyTable.
static const PropertyViewStatusType ErrorInvalidArrayOffsetType
This property view has an unknown array offset type.
static const PropertyViewStatusType ErrorBufferViewSizeNotDivisibleByTypeSize
This property view has an invalid buffer view; its length is not a multiple of the size of its type /...
A view on the data of the PropertyTableProperty that is created by a PropertyTableView.
Utility to retrieve the data of PropertyTable.
PropertyTableViewStatus status() const noexcept
Gets the status of this property table view.
void getPropertyView(const std::string &propertyId, Callback &&callback) const
Gets a PropertyTablePropertyView through a callback that accepts a property id and a PropertyTablePro...
const Class * getClass() const noexcept
Gets the Class that this property table conforms to.
PropertyTablePropertyView< T, Normalized > getPropertyView(const std::string &propertyId) const
Gets a PropertyTablePropertyView that views the data of a property stored in the PropertyTable.
int64_t size() const noexcept
Get the number of elements in this PropertyTableView. If the view is valid, this returns PropertyTabl...
const std::optional< std::string > & name() const noexcept
Gets the name of the property table being viewed. Returns std::nullopt if no name was specified.
void forEachProperty(Callback &&callback) const
Iterates over each property in the PropertyTable with a callback that accepts a property id and a Pro...
const ClassProperty * getClassProperty(const std::string &propertyId) const
Finds the ClassProperty that describes the type information of the property with the specified id.
PropertyTableView(const Model &model, const PropertyTable &propertyTable)
Creates an instance of PropertyTableView.
static const PropertyViewStatusType ErrorTypeMismatch
This property view's type does not match what is specified in ClassProperty::type.
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 ErrorInvalidNormalization
This property says it is normalized, but it does not have an integer component type.
static const PropertyViewStatusType ErrorArrayTypeMismatch
This property view differs from what is specified in ClassProperty::array.
static const PropertyViewStatusType ErrorNormalizationMismatch
This property view's normalization differs from what is specified in ClassProperty::normalized.
static const PropertyViewStatusType ErrorInvalidEnum
The property provided an invalid enum value.
static const PropertyViewStatusType ErrorComponentTypeMismatch
This property view's component type does not match what is specified in ClassProperty::componentType.
Classes for working with glTF models.
PropertyComponentType
The possible types of a property component.
@ Float32
A property component equivalent to a float.
@ Uint32
A property component equivalent to a uint32_t.
@ Uint16
A property component equivalent to a uint16_t.
@ Int16
A property component equivalent to an int16_t.
@ Uint8
A property component equivalent to a uint8_t.
@ Int8
A property component equivalent to an int8_t.
@ Int32
A property component equivalent to an int32_t.
@ Uint64
A property component equivalent to a uint32_t.
@ Float64
A property component equivalent to a double.
@ Int64
A property component equivalent to an int64_t.
PropertyComponentType convertArrayOffsetTypeStringToPropertyComponentType(const std::string &str)
Converts a string listed in PropertyTableProperty::ArrayOffsetType to its corresponding PropertyCompo...
bool isPropertyTypeVecN(PropertyType type)
Checks if the given PropertyType represents a vector with any number of components.
bool isPropertyComponentTypeInteger(PropertyComponentType componentType)
Checks if the given PropertyComponentType represents an integer value.
PropertyComponentType convertStringToPropertyComponentType(const std::string &str)
Converts a string into a PropertyComponentType.
@ ErrorClassNotFound
The property attribute's specified class could not be found in the extension.
@ ErrorMissingMetadataExtension
The glTF is missing the EXT_structural_metadata extension.
@ ErrorMissingSchema
The glTF EXT_structural_metadata extension doesn't contain a schema.
bool isPropertyTypeMatN(PropertyType type)
Checks if the given PropertyType represents a matrix with any number of components.
bool canRepresentPropertyType(PropertyType type)
Returns whether the type T can represent the given PropertyType.
@ Valid
This accessor is valid and ready to use.
PropertyType convertStringToPropertyType(const std::string &str)
Converts a string into a PropertyType.
PropertyType
The possible types of a property in a PropertyTableView.
@ Scalar
A scalar property, i.e. an integer or floating point value.
glm::length_t getDimensionsFromPropertyType(PropertyType type)
Obtains the number of dimensions in the given PropertyType.
PropertyTableViewStatus
Indicates the status of a property table view.
@ Valid
This property table view is valid and ready to use.
int32_t PropertyViewStatusType
The type used for fields of PropertyViewStatus.
bool normalized
Specifies whether integer values are normalized. Only applicable to SCALAR, VECN, and MATN types with...
std::optional< std::string > componentType
The datatype of the element's components. Only applicable to SCALAR, VECN, and MATN types.
bool array
Whether the property is an array. When count is defined the property is a fixed-length array....
std::string type
The element type.
std::optional< std::string > enumType
Enum ID as declared in the enums dictionary. Required when type is ENUM.
A class containing a set of properties.
Definition Class.h:17
std::unordered_map< std::string, CesiumGltf::ClassProperty > properties
A dictionary, where each key is a property ID and each value is an object defining the property....
Definition Class.h:38
This class is not meant to be instantiated directly. Use Enum instead.
Definition Enum.h:12
void type
The component type of this metadata array.
This class is not meant to be instantiated directly. Use Model instead.
Definition Model.h:14
Properties conforming to a class, organized as property values stored in binary columnar arrays.
std::unordered_map< std::string, CesiumGltf::PropertyTableProperty > properties
A dictionary, where each key corresponds to a property ID in the class' properties dictionary and eac...
std::optional< std::string > name
The name of the property table, e.g. for display purposes.
int64_t count
The number of elements in each property array.