用 Qt3D 画三角形

Posted

技术标签:

【中文标题】用 Qt3D 画三角形【英文标题】:Draw triangle with Qt3D 【发布时间】:2019-02-21 06:22:49 【问题描述】:

我需要用 Qt3D 绘制一个多边形,但我发现了如何用三角形来做到这一点。所以。我找到了一个代码,它与 Qt3D 的 OpenGL 更紧密地配合使用。它正在画线,但我换了一条线来画三角形。当我设置setPrimitiveType(Qt3DRender::QGeometryRenderer::TriangleStrip); 时,它就可以工作了。但我想用 setPrimitiveType(Qt3DRender::QGeometryRenderer::Triangles);,由于某种原因不起作用并且不绘制任何东西。也许我做了错误的顶点和索引属性配置?或者也许我使用了不正确的输入? (我的输入示例在底部)

void MainWindow::drawTriangles(const QPolygonF polygon, QColor color, Qt3DCore::QEntity *_rootEntity)

    int numOfVertices = polygon.size();
    auto *geometry = new Qt3DRender::QGeometry(_rootEntity);

    // Create and fill vertex buffer
    QByteArray bufferBytes;
    bufferBytes.resize(3 * numOfVertices * static_cast<int>(sizeof(float)));
    float *positions = reinterpret_cast<float*>(bufferBytes.data());

    for(auto point : polygon)
        float fHalfMapWidth = (dFittedMaxMapX_ - dFittedMinMapX_) / 2;
        float fHalfMapHeight = (dFittedMaxMapY_ - dFittedMinMapY_) / 2;

        *positions++ = static_cast<float>(point.x() - (dFittedMinMapX_ + fHalfMapWidth));
        *positions++ = 0.0f; //We need to drow only on the surface
        *positions++ = static_cast<float>(point.y() - (dFittedMinMapY_ + fHalfMapHeight));
    

    auto *buf = new Qt3DRender::QBuffer(geometry);
    buf->setData(bufferBytes);

    auto *positionAttribute = new Qt3DRender::QAttribute(geometry); // Create a vertex attribute
    positionAttribute->setName(Qt3DRender::QAttribute::defaultPositionAttributeName());
    positionAttribute->setVertexBaseType(Qt3DRender::QAttribute::Float); // In vertex buffer we store floats
    positionAttribute->setVertexSize(3); // Size of a vertex
    positionAttribute->setAttributeType(Qt3DRender::QAttribute::VertexAttribute); // Type of an attribute
    positionAttribute->setBuffer(buf); // Set vertex buffer
    positionAttribute->setByteStride(3 * sizeof(float)); // Stride between vertices
    geometry->addAttribute(positionAttribute); // Add the attribute in our  Qt3DRender::QGeometry

    // Create and fill an index buffer
    QByteArray indexBytes;
    indexBytes.resize(numOfVertices * static_cast<int>(sizeof(unsigned int))); // start to end
    unsigned int *indices = reinterpret_cast<unsigned int*>(indexBytes.data());
    for(unsigned int i = 0; i < static_cast<unsigned int>(numOfVertices); ++i) 
        *indices++ = i;
    

    auto *indexBuffer = new Qt3DRender::QBuffer(geometry);
    indexBuffer->setData(indexBytes);

    auto *indexAttribute = new Qt3DRender::QAttribute(geometry); // Create an index attribute
    indexAttribute->setVertexBaseType(Qt3DRender::QAttribute::UnsignedInt); // In the index buffer we will store Unsigned int
    indexAttribute->setAttributeType(Qt3DRender::QAttribute::IndexAttribute); // Type of an attribute
    indexAttribute->setBuffer(indexBuffer); // Set the index buffer
    indexAttribute->setCount(static_cast<unsigned int>(numOfVertices)); // As i understand, this is a number of vertices
    geometry->addAttribute(indexAttribute); // Add attribute in our Qt3DRender::QGeometry


    auto *poly = new Qt3DRender::QGeometryRenderer(_rootEntity);
    poly->setGeometry(geometry); 
    //poly->setPrimitiveType(Qt3DRender::QGeometryRenderer::TriangleStrip); - Working variant
    poly->setPrimitiveType(Qt3DRender::QGeometryRenderer::Triangles); // - This is not working variant

    //Create a material
    auto *material = new Qt3DExtras::QPhongMaterial(_rootEntity);
    material->setAmbient(color);

    auto *trianglesEntity = new Qt3DCore::QEntity(_rootEntity);
    trianglesEntity->addComponent(poly); // Add our primitives
    trianglesEntity->addComponent(material); // Add material

我尝试使用三点和四点。当我使用 4 个点时,如果我将原始类型设置为“TriangleStrip”,它就会绘制。但是“三角形”在这两种情况下都不起作用。

mapBorder << QPointF(300,200) << QPointF(100,200) << QPointF(200,0) << QPointF(300,200) << QPointF(200,0) << QPointF(500,200) << QPointF(500,300); 
drawTriangles(mapBorder, QColor(Qt::black), mapEntity_);

或者你能给我另一个建议如何绘制多边形或三角形。

【问题讨论】:

有 4 个点和 triangleStrip,您正在绘制 2 个三角形,尝试按顺时针顺序重新排列三角形,我认为您是在逆时针方向进行。或者可能反转您设置的默认值doc.qt.io/qt-5.11/qt3drender-qcullface.html 我尝试将我的积分顺序更改为顺时针,并且成功了。好的!但我需要绘制两种情况(顺时针和逆时针)。你的意思是我可以通过使用 QCullFace 来改变这种行为吗?如果是这样,您能帮我了解如何在我的情况下使用它吗?我真的不明白,在这些代码行之后我要做什么// using namespace Qt3DRender; QRenderStateSet *renderStateSet = new QRenderStateSet(); QCullFace *cullFront = new QCullFace(); cullFront-&gt;setMode(QCullFace::Front); renderStateSet-&gt;addRenderState(cullFront); 【参考方案1】:

您的积分按顺时针顺序设置,您可以通过此链接重新排序或设置不同的行为:doc.qt.io/qt-5.11/qt3drender-qcullface.html

如果您两者都需要,我会尝试Qt3DRender::QCullFace::NoCulling

【讨论】:

以上是关于用 Qt3D 画三角形的主要内容,如果未能解决你的问题,请参考以下文章

Effective前端3:用CSS画一个三角形

如何用python画钝角三角形

Effective前端用CSS画一个三角形

怎么用python画三角形并填色?以及画五角星并且填色?(急!!!)

python怎么画三角形

用CSS画一个带阴影的三角形的示例代码