3d - 如何在 JavaFX 中相同地渲染三角形的正面和背面

标签 3d javafx rendering

我开始使用 JavaFX 来可视化 3D 非结构化网格,但我正面临渲染问题。

如 [1] 中所述,除非您使用 CullFace.NONE 选项,否则 JavaFX 只会渲染三角形的正面。但是后面的脸是黑色的。

由于 3D 网格是由外部工具(例如 Gmsh http://geuz.org/gmsh/)生成的,因此我无法控制面的方向。使用网格的科学软件也不需要定向网格。

因此,我不想事后重新定向网格,只是为了相同地渲染三角形的正面和背面。 JavaFX 8 可以做到这一点吗?如何?

感谢您的回答。

值得注意的是:我也在 Oracle 论坛 [2] 上发布了一个类似的问题,但它们似乎很空洞。如果你们中的一些人知道 JavaFX 社区活跃的地方,那么链接会很有用。如果我有有用的答案要分享,我当然会更新这两个线程。

亲切的问候

Sphere meshed in 3D

[1] How to make sense of JavaFX triangle mesh?

[2] https://community.oracle.com/thread/3593434

最佳答案

解决方案 1

您可以通过绘制两组具有不同面方向的网格来解决该问题。请参阅下面的结果和代码。然而,这会使数据量和处理时间加倍。

PS:还有一个问题值得一提。目前尚不清楚在当前版本的 JavaFX (@August 2014) 中,您是否可以为网格边缘着色不同于面的颜色。如果您需要制作平面拼贴的可见单个拼贴,这将是必要的。解决方案是再次添加两组网格对象。但这使所需资源翻了两番。

此外,人们想剔除一些不必要的边缘。在下图中,只需要突出显示地板边缘,而不是对角线。

解决方案 2

将每个网格面替换为 3D 网格对象,即您创建的不是矩形表面,而是创建平板。因此,一个开放的盒子对象将由五块板制成,盒子的内部和外部将具有相同的颜色。这个解决方案,就像第一个一样,仍然是一个黑客,仍然会产生处理开销。

数字

实际 JavaFX 渲染与所需渲染(在 Matlab 中生成)之间的比较:

http://s30.postimg.org/iuotogvgh/3d_tower.jpg

部分解决方案:

http://s30.postimg.org/83dcwtpkx/3d_boxes.png

代码

import javafx.scene.Group;
import javafx.scene.paint.Color;
import javafx.scene.paint.PhongMaterial;
import javafx.scene.shape.CullFace;
import javafx.scene.shape.DrawMode;
import javafx.scene.shape.MeshView;
import javafx.scene.shape.TriangleMesh;

/**
// * Draw polygonal 3D box.
// * 
// * INPUT
// * - footprint: 2D polygon coordinates; 
// *   closed path (i.e. first and last coordinates are identical); ex:
// *   float[] footprint = {
// *       10, -1,
// *       -1, -1,
// *       -1, 5,
// *       10, 5,
// *       10, -1
// *   };
// * - zLevel: z-coordinate of actual floor level: int k; float zLevel = k * HEIGHT - HEIGHT;
// * - HEIGHT: height of the box: private static final float HEIGHT = (float) 50;
// * 
// * NOTE: we have to use the mesh method since the straightforward way
// * to construct a rectangle - "rectangle" method - produces blurry edges.
// *
// */
public class DrawPolygonalBox {

// Draw polygonal 3D box.
public static Group draw(float[] footprint, float zLevel, float HEIGHT) {

    Group box = new Group();
    int y = 0;

    // for each footprint coordinate make a rectangle
    int n = footprint.length - 2;

    // one side of the box
    for (int k = 0; k < n; k = k + 2) {

        float[] points = {
            footprint[k], y + zLevel, footprint[k + 1],
            footprint[k + 2], y + zLevel, footprint[k + 3],
            footprint[k + 2], y + zLevel + HEIGHT, footprint[k + 3],
            footprint[k], y + zLevel + HEIGHT, footprint[k + 1]
        };            
        float[] texCoords = {
            1, 1,
            1, 0,
            0, 1,
            0, 0
        };
        int[] faces = {
            0, 0, 2, 2, 1, 1,
            0, 0, 3, 3, 2, 2
        };
        int[] faces2 = {
            0, 0, 1, 1, 2, 2,
            0, 0, 2, 2, 3, 3
        };

        TriangleMesh mesh1 = new TriangleMesh();
        mesh1.getPoints().setAll(points);
        mesh1.getTexCoords().setAll(texCoords);
        mesh1.getFaces().setAll(faces);

        TriangleMesh mesh2 = new TriangleMesh();
        mesh2.getPoints().setAll(points);
        mesh2.getTexCoords().setAll(texCoords);
        mesh2.getFaces().setAll(faces2);

        final MeshView rectangle1 = new MeshView(mesh1);
        rectangle1.setMaterial(new PhongMaterial(Color.web("#FF0000",0.25)));
        rectangle1.setCullFace(CullFace.BACK);

        final MeshView rectangle2 = new MeshView(mesh2);
        rectangle2.setMaterial(new PhongMaterial(Color.web("#FF0000",0.25)));
        rectangle2.setCullFace(CullFace.BACK);

        final MeshView wire1 = new MeshView(mesh1);
        wire1.setMaterial(new PhongMaterial(Color.web("#000000",0.5)));
        wire1.setCullFace(CullFace.BACK);
        wire1.setDrawMode(DrawMode.LINE);

        final MeshView wire2 = new MeshView(mesh2);
        wire2.setMaterial(new PhongMaterial(Color.web("#000000",0.5)));
        wire2.setCullFace(CullFace.BACK);
        wire2.setDrawMode(DrawMode.LINE);

        // add to group
        box.getChildren().addAll(rectangle1, wire1, rectangle2, wire2);
    }

    return box;
}

}

关于3d - 如何在 JavaFX 中相同地渲染三角形的正面和背面,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25190409/

相关文章:

c# - .Net Matrix3D Transform() 究竟做了什么/为什么我得到 "- infinity"?

c# - WPF 中 3D 表面问题的图像缩放质量

java - 如何在 javafx 中定位窗口阶段?

java - 在 HijrahChronology 中配置自定义变体以进行日期校正 jdk 8

ios - 为什么 Flatlist 中的图像有时无法渲染?

python - Panda3D与Python,只渲染一帧等问题

c++ - 在 OpenGL 中变换相机位置

java - 除了生成新线程来运行简单的后台任务之外,还有其他选择吗?

javascript - 节点上带有换行文本的 Flex TreeView

javascript - 在 React Native 中重新渲染 - 问题