727{
729 if (!currentNode) {
730 static G4bool first =
true;
731 if (first) {
732 first = false;
733 G4Exception(
"G4Qt3DSceneHandler::AddPrimitive(const G4Polyhedron&)",
735 "No available node!");
736 }
737 return;
738 }
739
741
743
744
745
746
747
748
749
750
751 std::vector<G4Point3D> vertices;
752 std::vector<G4Normal3D> normals;
753
754
755
756 typedef std::pair<G4Point3D,G4Point3D> Line;
757 std::vector<Line> lines;
758 auto insertIfNew = [&lines](const Line& newLine) {
759
760
761
762
763
764
765
766
767 lines.push_back(newLine);
768 };
769
770 G4bool isAuxilaryEdgeVisible =
fpViewer->GetViewParameters().IsAuxEdgeVisible();
772 do {
777 notLastFace = polyhedron.
GetNextFacet(nEdges, vertex, edgeFlag, normal);
778 vertices.push_back(vertex[0]);
779 vertices.push_back(vertex[1]);
780 vertices.push_back(vertex[2]);
781 normals.push_back(normal[0]);
782 normals.push_back(normal[1]);
783 normals.push_back(normal[2]);
784 if(isAuxilaryEdgeVisible||edgeFlag[0]>0)insertIfNew(Line(vertex[0],vertex[1]));
785 if(isAuxilaryEdgeVisible||edgeFlag[1]>0)insertIfNew(Line(vertex[1],vertex[2]));
786 if (nEdges == 3) {
787
788
789 if(isAuxilaryEdgeVisible||edgeFlag[2]>0)insertIfNew(Line(vertex[2],vertex[0]));
790 } else if (nEdges == 4) {
791
792
793 vertices.push_back(vertex[2]);
794 vertices.push_back(vertex[3]);
795 vertices.push_back(vertex[0]);
796 normals.push_back(normal[2]);
797 normals.push_back(normal[3]);
798 normals.push_back(normal[0]);
799 if(isAuxilaryEdgeVisible||edgeFlag[2]>0)insertIfNew(Line(vertex[2],vertex[3]));
800 if(isAuxilaryEdgeVisible||edgeFlag[3]>0)insertIfNew(Line(vertex[3],vertex[0]));
801 } else {
803 << "ERROR: polyhedron face with unexpected number of edges (" << nEdges << ')'
804 <<
"\n Tag: " <<
fpModel->GetCurrentTag()
806 return;
807 }
808 } while (notLastFace);
809 const auto nVerts = vertices.size();
810 const auto nLines = lines.size();
811
812
813
815 transform->setObjectName("transform");
816
817 Qt3DCore::QEntity* wireframeEntity = nullptr;
818 Qt3DCore::QEntity* surfaceEntity = nullptr;
819 static G4int errorCount = 0;
821 switch (drawing_style) {
823 wireframeEntity = new Qt3DCore::QEntity(currentNode);
824 wireframeEntity->addComponent(transform);
825 break;
827 wireframeEntity = new Qt3DCore::QEntity(currentNode);
828 wireframeEntity->addComponent(transform);
829 surfaceEntity = new Qt3DCore::QEntity(currentNode);
830 surfaceEntity->addComponent(transform);
831 break;
833 surfaceEntity = new Qt3DCore::QEntity(currentNode);
834 surfaceEntity->addComponent(transform);
835 break;
837 wireframeEntity = new Qt3DCore::QEntity(currentNode);
838 wireframeEntity->addComponent(transform);
839 surfaceEntity = new Qt3DCore::QEntity(currentNode);
840 surfaceEntity->addComponent(transform);
841 break;
843
844 if (errorCount == 0) {
845 ++errorCount;
846 G4warn <<
"WARNING: Qt3D: cloud drawing not implemented" <<
G4endl;
847 }
848 return;
849 break;
850 }
851
852 const auto vertexByteSize = 3*
sizeof(
PRECISION);
853
854 G4Qt3DCompat::QGeometry* vertexGeometry = nullptr;
855 G4Qt3DCompat::QGeometry* lineGeometry = nullptr;
856
857 G4Qt3DCompat::QAttribute* positionAtt = nullptr;
858 G4Qt3DCompat::QAttribute* normalAtt = nullptr;
859 G4Qt3DCompat::QAttribute* lineAtt = nullptr;
860 G4Qt3DCompat::QAttribute* dummyNormalLineAtt = nullptr;
861
862 G4Qt3DCompat::QBuffer* vertexBuffer = nullptr;
866
867
868
869 QByteArray vertexByteArray;
870 const auto vertexBufferByteSize = 2*nVerts*vertexByteSize;
871 vertexByteArray.resize((
G4int)vertexBufferByteSize);
872 auto vertexBufferArray =
reinterpret_cast<PRECISION*
>(vertexByteArray.data());
874 for (std::size_t i = 0; i < nVerts; ++i) {
875 vertexBufferArray[i1++] = vertices[i].x();
876 vertexBufferArray[i1++] = vertices[i].y();
877 vertexBufferArray[i1++] = vertices[i].z();
878 vertexBufferArray[i1++] = normals[i].x();
879 vertexBufferArray[i1++] = normals[i].y();
880 vertexBufferArray[i1++] = normals[i].z();
881 }
882
883 vertexGeometry = new G4Qt3DCompat::QGeometry();
884 vertexGeometry->setObjectName("vertexGeometry");
885 vertexBuffer = new G4Qt3DCompat::QBuffer(vertexGeometry);
886 vertexBuffer->setObjectName("Vertex buffer");
887 vertexBuffer->setData(vertexByteArray);
888
889
890 positionAtt = new G4Qt3DCompat::QAttribute;
891 positionAtt->setObjectName("Position attribute");
892 positionAtt->setName(G4Qt3DCompat::QAttribute::defaultPositionAttributeName());
893 positionAtt->setBuffer(vertexBuffer);
894 positionAtt->setAttributeType(G4Qt3DCompat::QAttribute::VertexAttribute);
895 positionAtt->setVertexBaseType(
BASETYPE);
896 positionAtt->setVertexSize(3);
897 positionAtt->setCount((
G4int)nVerts);
898 positionAtt->setByteOffset(0);
899 positionAtt->setByteStride(2*vertexByteSize);
900
901
902 normalAtt = new G4Qt3DCompat::QAttribute;
903 normalAtt->setObjectName("Normal attribute");
904 normalAtt->setName(G4Qt3DCompat::QAttribute::defaultNormalAttributeName());
905 normalAtt->setBuffer(vertexBuffer);
906 normalAtt->setAttributeType(G4Qt3DCompat::QAttribute::VertexAttribute);
907 normalAtt->setVertexBaseType(
BASETYPE);
908 normalAtt->setVertexSize(3);
909 normalAtt->setCount((
G4int)nVerts);
910 normalAtt->setByteOffset(vertexByteSize);
911 normalAtt->setByteStride(2*vertexByteSize);
912 }
913
914 G4Qt3DCompat::QBuffer* lineBuffer = nullptr;
918
919
920 QByteArray lineByteArray;
921 const auto lineBufferByteSize = 2*nLines*vertexByteSize;
922 lineByteArray.resize((
G4int)lineBufferByteSize);
923 auto lineBufferArray =
reinterpret_cast<PRECISION*
>(lineByteArray.data());
925 for (const auto& line: lines) {
926 lineBufferArray[i2++] = line.first.x();
927 lineBufferArray[i2++] = line.first.y();
928 lineBufferArray[i2++] = line.first.z();
929 lineBufferArray[i2++] = line.second.x();
930 lineBufferArray[i2++] = line.second.y();
931 lineBufferArray[i2++] = line.second.z();
932 }
933
934 lineGeometry = new G4Qt3DCompat::QGeometry();
935 lineGeometry->setObjectName("lineGeometry");
936 lineBuffer = new G4Qt3DCompat::QBuffer(lineGeometry);
937 lineBuffer->setObjectName("Line buffer");
938 lineBuffer->setData(lineByteArray);
939
940
941 lineAtt = new G4Qt3DCompat::QAttribute;
942 lineAtt->setObjectName("Position attribute");
943 lineAtt->setName(G4Qt3DCompat::QAttribute::defaultPositionAttributeName());
944 lineAtt->setBuffer(lineBuffer);
945 lineAtt->setAttributeType(G4Qt3DCompat::QAttribute::VertexAttribute);
946 lineAtt->setVertexBaseType(
BASETYPE);
947 lineAtt->setVertexSize(3);
948 lineAtt->setCount((
G4int)nLines);
949 lineAtt->setByteOffset(0);
950 lineAtt->setByteStride(vertexByteSize);
951
952 dummyNormalLineAtt = new G4Qt3DCompat::QAttribute;
953 dummyNormalLineAtt->setObjectName("Normal attribute");
954 dummyNormalLineAtt->setName(G4Qt3DCompat::QAttribute::defaultNormalAttributeName());
955 dummyNormalLineAtt->setBuffer(lineBuffer);
956 dummyNormalLineAtt->setAttributeType(G4Qt3DCompat::QAttribute::VertexAttribute);
957 dummyNormalLineAtt->setVertexBaseType(
BASETYPE);
958 dummyNormalLineAtt->setVertexSize(3);
959 dummyNormalLineAtt->setCount(0);
960 dummyNormalLineAtt->setByteOffset(0);
961 dummyNormalLineAtt->setByteStride(vertexByteSize);
962 }
963
964
965
967 Qt3DExtras::QDiffuseSpecularMaterial* material;
968 Qt3DRender::QGeometryRenderer* renderer;
969 switch (drawing_style) {
970
972
973 lineGeometry->addAttribute(lineAtt);
974 lineGeometry->addAttribute(dummyNormalLineAtt);
975
976 material = new Qt3DExtras::QDiffuseSpecularMaterial();
977 material->setObjectName("materialForWireframe");
979 material->setShininess(0.);
980 material->setSpecular(0.);
981 wireframeEntity->addComponent(material);
982
983 renderer = new Qt3DRender::QGeometryRenderer;
984 renderer->setObjectName("polyhedronWireframeRenderer");
985 renderer->setGeometry(lineGeometry);
986 renderer->setVertexCount(2*(
G4int)nLines);
987 renderer->setPrimitiveType(Qt3DRender::QGeometryRenderer::Lines);
988 wireframeEntity->addComponent(renderer);
989
990 break;
991
993
994
995
996 vertexGeometry->addAttribute(positionAtt);
997 vertexGeometry->addAttribute(normalAtt);
998
999 material = new Qt3DExtras::QDiffuseSpecularMaterial();
1000 material->setObjectName("materialForHiddenLines");
1001 material->setAmbient(Qt::white);
1002 material->setShininess(0.);
1003 material->setSpecular(0.);
1004 surfaceEntity->addComponent(material);
1005
1006 renderer = new Qt3DRender::QGeometryRenderer;
1007 renderer->setObjectName("polyhedronSurfaceRenderer");
1008 renderer->setGeometry(vertexGeometry);
1009 renderer->setVertexCount((
G4int)nVerts);
1010 renderer->setPrimitiveType(Qt3DRender::QGeometryRenderer::Triangles);
1011 surfaceEntity->addComponent(renderer);
1012
1013
1014
1015 lineGeometry->addAttribute(lineAtt);
1016 lineGeometry->addAttribute(dummyNormalLineAtt);
1017
1018 material = new Qt3DExtras::QDiffuseSpecularMaterial();
1019 material->setObjectName("materialForWireFrame");
1021 material->setShininess(0.);
1022 material->setSpecular(0.);
1023 wireframeEntity->addComponent(material);
1024
1025 renderer = new Qt3DRender::QGeometryRenderer;
1026 renderer->setObjectName("polyhedronWireframeRenderer");
1027 renderer->setGeometry(lineGeometry);
1028 renderer->setVertexCount(2*(
G4int)nLines);
1029 renderer->setPrimitiveType(Qt3DRender::QGeometryRenderer::Lines);
1030 wireframeEntity->addComponent(renderer);
1031
1032 break;
1033
1035
1036 vertexGeometry->addAttribute(positionAtt);
1037 vertexGeometry->addAttribute(normalAtt);
1038
1039 material = new Qt3DExtras::QDiffuseSpecularMaterial();
1040 material->setObjectName("materialForSurface");
1042 if (colour.GetAlpha() < 1.) material->setAlphaBlendingEnabled(true);
1043 surfaceEntity->addComponent(material);
1044
1045 renderer = new Qt3DRender::QGeometryRenderer;
1046 renderer->setObjectName("polyhedronSurfaceRenderer");
1047 renderer->setGeometry(vertexGeometry);
1048 renderer->setVertexCount((
G4int)nVerts);
1049 renderer->setPrimitiveType(Qt3DRender::QGeometryRenderer::Triangles);
1050 surfaceEntity->addComponent(renderer);
1051
1052 break;
1053
1055
1056
1057
1058 vertexGeometry->addAttribute(positionAtt);
1059 vertexGeometry->addAttribute(normalAtt);
1060
1061 material = new Qt3DExtras::QDiffuseSpecularMaterial();
1062 material->setObjectName("materialForSurface");
1064 if (colour.GetAlpha() < 1.) material->setAlphaBlendingEnabled(true);
1065 surfaceEntity->addComponent(material);
1066
1067 renderer = new Qt3DRender::QGeometryRenderer;
1068 renderer->setObjectName("polyhedronSurfaceRenderer");
1069 renderer->setGeometry(vertexGeometry);
1070 renderer->setVertexCount((
G4int)nVerts);
1071 renderer->setPrimitiveType(Qt3DRender::QGeometryRenderer::Triangles);
1072 surfaceEntity->addComponent(renderer);
1073
1074
1075
1076 lineGeometry->addAttribute(lineAtt);
1077 lineGeometry->addAttribute(dummyNormalLineAtt);
1078
1079 material = new Qt3DExtras::QDiffuseSpecularMaterial();
1080 material->setObjectName("materialForWireframe");
1082 material->setShininess(0.);
1083 material->setSpecular(0.);
1084 wireframeEntity->addComponent(material);
1085
1086 renderer = new Qt3DRender::QGeometryRenderer;
1087 renderer->setObjectName("polyhedronSurfaceRenderer");
1088 renderer->setGeometry(lineGeometry);
1089 renderer->setVertexCount(2*(
G4int)nLines);
1090 renderer->setPrimitiveType(Qt3DRender::QGeometryRenderer::Lines);
1091 wireframeEntity->addComponent(renderer);
1092
1093 break;
1094
1096
1097 break;
1098 }
1099}
HepGeom::Normal3D< G4double > G4Normal3D
HepGeom::Point3D< G4double > G4Point3D
G4ViewParameters::DrawingStyle GetDrawingStyle(const G4VisAttributes *)
G4int GetNoFacets() const
G4bool GetNextFacet(G4int &n, G4Point3D *nodes, G4int *edgeFlags=nullptr, G4Normal3D *normals=nullptr) const