All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.sun.javafx.sg.prism.NGShape3D Maven / Gradle / Ivy

There is a newer version: 24-ea+19
Show newest version
/*
 * Copyright (c) 2013, 2024, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package com.sun.javafx.sg.prism;

import com.sun.javafx.geom.Vec3d;
import com.sun.javafx.geom.transform.Affine3D;
import com.sun.javafx.util.Utils;
import com.sun.prism.Graphics;
import com.sun.prism.Material;
import com.sun.prism.MeshView;
import com.sun.prism.ResourceFactory;

import javafx.application.ConditionalFeature;
import javafx.application.Platform;
import javafx.scene.shape.CullFace;
import javafx.scene.shape.DrawMode;

/**
 * TODO: 3D - Need documentation
 */
public abstract class NGShape3D extends NGNode {
    private NGPhongMaterial material;
    private DrawMode drawMode;
    private CullFace cullFace;
    private boolean materialDirty = false;
    private boolean drawModeDirty = false;
    NGTriangleMesh mesh;
    private MeshView meshView;

    public void setMaterial(NGPhongMaterial material) {
        this.material = material;
        materialDirty = true;
        visualsChanged();
    }
    public void setDrawMode(Object drawMode) {
        this.drawMode = (DrawMode) drawMode;
        drawModeDirty = true;
        visualsChanged();
    }

    public void setCullFace(Object cullFace) {
        this.cullFace = (CullFace) cullFace;
        visualsChanged();
    }

    void invalidate() {
        meshView = null;
        visualsChanged();
    }

    private void renderMeshView(Graphics g) {

        //validate state
        g.setup3DRendering();

        ResourceFactory rf = g.getResourceFactory();
        if (rf == null || rf.isDisposed()) {
            return;
        }

        // Check whether the meshView is valid; dispose and recreate if needed
        if (meshView != null && !meshView.isValid()) {
            meshView.dispose();
            meshView = null;
        }

        if (meshView == null && mesh != null) {
            meshView = rf.createMeshView(mesh.createMesh(rf));
            materialDirty = drawModeDirty = true;
        }

        if (meshView == null || !mesh.validate()) {
            return;
        }

        Material mtl =  material.createMaterial(rf);
        if (materialDirty) {
            meshView.setMaterial(mtl);
            materialDirty = false;
        }

        // NOTE: Always check determinant in case of mirror transform.
        int cullingMode = cullFace.ordinal();
        if (cullFace.ordinal() != MeshView.CULL_NONE
                && g.getTransformNoClone().getDeterminant() < 0) {
            cullingMode = cullingMode == MeshView.CULL_BACK
                    ? MeshView.CULL_FRONT : MeshView.CULL_BACK;
        }
        meshView.setCullingMode(cullingMode);

        if (drawModeDirty) {
            meshView.setWireframe(drawMode == DrawMode.LINE);
            drawModeDirty = false;
        }

        setupLights(g);

        meshView.render(g);
    }

    private void setupLights(Graphics g) {
        int lightIndex = 0;
        NGLightBase[] lights = g.getLights();
        if (noLights(lights)) {
            setDefaultLight(g);
            lightIndex++;
        } else {
            float ambientRed = 0.0f;
            float ambientBlue = 0.0f;
            float ambientGreen = 0.0f;

            for (NGLightBase lightBase : lights) {
                if (lightBase == null) {
                    // The array of lights can have nulls
                    break;
                }
                if (!lightBase.affects(this)) {
                    continue;
                }
                // Transparent component is ignored
                float rL = lightBase.getColor().getRed();
                float gL = lightBase.getColor().getGreen();
                float bL = lightBase.getColor().getBlue();
                // Black color is ignored
                if (rL == 0.0f && gL == 0.0f && bL == 0.0f) {
                    continue;
                }
                if (lightBase instanceof NGAmbientLight) {
                    // Accumulate ambient lights
                    ambientRed   += rL;
                    ambientGreen += gL;
                    ambientBlue  += bL;
                }
                /* TODO: 3D
                 * There is a limit on the number of point lights that can affect
                 * a 3D shape. (Currently we simply select the first 3)
                 * Thus it is important to select the most relevant lights.
                 *
                 * One such way would be to sort lights according to
                 * intensity, which becomes especially relevant when lights
                 * are attenuated. Only the most intense set of lights
                 * would be set.
                 * The approximate intensity a light will have on a given
                 * shape, could be defined by:
                 *
                 * Where d is distance from point light
                 * float attenuationFactor = 1/(c + cL * d + cQ * d * d);
                 * float intensity = rL * 0.299f + gL * 0.587f + bL * 0.114f;
                 * intensity *= attenuationFactor;
                */
                else if (lightBase instanceof NGSpotLight light) {
                    addSpotLight(light, lightIndex++, rL, gL, bL);
                } else if (lightBase instanceof NGPointLight light) {
                    addPointLight(light, lightIndex++, rL, gL, bL);
                } else if (lightBase instanceof NGDirectionalLight light) {
                    addDirectionalLight(light, lightIndex++, rL, gL, bL);
                }
            }
            ambientRed = Utils.clamp(0, ambientRed, 1);
            ambientGreen = Utils.clamp(0, ambientGreen, 1);
            ambientBlue = Utils.clamp(0, ambientBlue, 1);
            meshView.setAmbientLight(ambientRed, ambientGreen, ambientBlue);
        }
        // TODO: 3D Required for D3D implementation of lights, which is limited to 3

        while (lightIndex < 3) { // Reset any previously set lights
            resetLight(lightIndex++);
        }
    }

    private boolean noLights(NGLightBase[] lights) {
        return lights == null || lights.length == 0 || lights[0] == null;
    }

    /**
     * Creates a white point light at the camera's (eye) position. The light uses the default attenuation parameters,
     * which means that it is not attenuated (isAttenuated == 0).
     */
    private void setDefaultLight(Graphics g) {
        meshView.setAmbientLight(0.0f, 0.0f, 0.0f);
        Vec3d cameraPos = g.getCameraNoClone().getPositionInWorld(null);
        var direction = NGPointLight.getSimulatedDirection();
        meshView.setLight(0,
                (float) cameraPos.x, (float) cameraPos.y, (float) cameraPos.z,
                1.0f, 1.0f, 1.0f, 1.0f,
                NGPointLight.getDefaultCa(), NGPointLight.getDefaultLa(), NGPointLight.getDefaultQa(), 1,
                NGPointLight.getDefaultMaxRange(),
                (float) direction.getX(), (float) direction.getY(), (float) direction.getZ(),
                NGPointLight.getSimulatedInnerAngle(),
                NGPointLight.getSimulatedOuterAngle(),
                NGPointLight.getSimulatedFalloff());
    }

    private void addPointLight(NGPointLight light, int lightIndex, float r, float g, float b) {
        Affine3D lightWT = light.getWorldTransform();
        var direction = NGPointLight.getSimulatedDirection();
        meshView.setLight(lightIndex,
                (float) lightWT.getMxt(), (float) lightWT.getMyt(), (float) lightWT.getMzt(),
                r, g, b, 1.0f,
                light.getCa(), light.getLa(), light.getQa(), 1,
                light.getMaxRange(),
                (float) direction.getX(), (float) direction.getY(), (float) direction.getZ(),
                NGPointLight.getSimulatedInnerAngle(),
                NGPointLight.getSimulatedOuterAngle(),
                NGPointLight.getSimulatedFalloff());
    }

    private void addSpotLight(NGSpotLight light, int lightIndex, float r, float g, float b) {
        Affine3D lightWT = light.getWorldTransform();
        var normDir = light.getDirection().normalize();
        meshView.setLight(lightIndex,
                (float) lightWT.getMxt(), (float) lightWT.getMyt(), (float) lightWT.getMzt(),
                r, g, b, 1.0f,
                light.getCa(), light.getLa(), light.getQa(), 1,
                light.getMaxRange(),
                (float) normDir.getX(), (float) normDir.getY(), (float) normDir.getZ(),
                light.getInnerAngle(), light.getOuterAngle(), light.getFalloff());
    }

    private void addDirectionalLight(NGDirectionalLight light, int lightIndex, float r, float g, float b) {
        var normDir = light.getDirection().normalize();
        meshView.setLight(lightIndex,
                0, 0, 0,                 // position is irrelevant
                r, g, b, 1.0f,
                1, 0, 0, 0,              // attenuation is irrelevant
                Float.POSITIVE_INFINITY, // range is irrelevant
                (float) normDir.getX(), (float) normDir.getY(), (float) normDir.getZ(),
                0, 0, 0);                // spotlight factors are irrelevant
    }

    private void resetLight(int lightIndex) {
        meshView.setLight(lightIndex,
                0, 0, 0,    // x y z
                0, 0, 0, 0, // r g b lightOn
                1, 0, 0, 0, 0, // ca la qa isAttenuated maxRange
                0, 0, 1,    // dirX Y Z
                0, 0, 0);   // inner outer falloff
    }

    public void setMesh(NGTriangleMesh triangleMesh) {
        this.mesh = triangleMesh;
        meshView = null;
        visualsChanged();
    }

    @Override
    protected void renderContent(Graphics g) {
        if (!Platform.isSupported(ConditionalFeature.SCENE3D) ||
             material == null ||
             g instanceof com.sun.prism.PrinterGraphics)
        {
            return;
        }
        renderMeshView(g);
    }

    // This node requires 3D graphics state for rendering
    @Override
    boolean isShape3D() {
        return true;
    }

    @Override
    protected boolean hasOverlappingContents() {
        return false;
    }

    @Override
    public void release() {
        // TODO: 3D - Need to release native resources
        // material, mesh and meshview have native backing that need clean up.
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy