Login  Register

Re: shadows are lost while Java3D 1.5.2 to 1.7.0 migration

Posted by questioning on Sep 01, 2022; 5:41am
URL: https://forum.jogamp.org/shadows-are-lost-while-Java3D-1-5-2-to-1-7-0-migration-tp4041834p4041836.html

Hello @philjord
Code (imports for Java3D 1.7.0):

    import org.jogamp.java3d.*;
    import org.jogamp.java3d.utils.behaviors.keyboard.KeyNavigatorBehavior;
    import org.jogamp.java3d.utils.behaviors.mouse.MouseRotate;
    import org.jogamp.java3d.utils.behaviors.mouse.MouseWheelZoom;
    import org.jogamp.java3d.utils.geometry.GeometryInfo;
    import org.jogamp.java3d.utils.geometry.NormalGenerator;
    import org.jogamp.java3d.utils.universe.PlatformGeometry;
    import org.jogamp.java3d.utils.universe.SimpleUniverse;
    import org.jogamp.vecmath.*;

    import javax.swing.*;
    import java.awt.*;
    import java.util.ArrayList;
   
    public class Plate {

        int jagged;
        private SimpleUniverse universe = null;
        private Canvas3D canvas = null;
        private TransformGroup viewtrans = null;

        private TransformGroup tg = null;
        private Transform3D t3d = null;
        private JPanel canvasPanel;

        public Plate() {
            canvasPanel = new JPanel(new BorderLayout());
            JFrame f = new JFrame();
            f.setLayout(new BorderLayout());
            JPanel p = new JPanel(new BorderLayout());
            p.add(BorderLayout.CENTER, canvasPanel);
            f.add(p, BorderLayout.CENTER);
            f.setSize(500, 500);
            f.setVisible(true);

        }

        public static void main(String[] args) {
            new Plate().paint3D();
        }

        public void paint3D() {
            set3dLoadedStatus();
            Runnable runnable = new Runnable() {
                public void run() {
                    try {
                        GraphicsConfiguration config = SimpleUniverse
                                .getPreferredConfiguration();

                        canvas = new Canvas3D(config);

                        universe = new SimpleUniverse(canvas);


                        universe.getViewingPlatform().setNominalViewingTransform();

                        universe.getViewer().getView().setBackClipDistance(10.0);

                        BranchGroup scene = new BranchGroup();

                        BoundingSphere bounds = new BoundingSphere(new Point3d(), 10000.0);

                        viewtrans = universe.getViewingPlatform().getViewPlatformTransform();

                        KeyNavigatorBehavior keyNavBeh = new KeyNavigatorBehavior(viewtrans);
                        keyNavBeh.setSchedulingBounds(bounds);
                        PlatformGeometry platformGeom = new PlatformGeometry();
                        platformGeom.addChild(keyNavBeh);
                        universe.getViewingPlatform().setPlatformGeometry(platformGeom);

                        BranchGroup objRoot = new BranchGroup();
                        objRoot.setCapability(BranchGroup.ALLOW_DETACH);
                        objRoot.setCapability(BranchGroup.ALLOW_CHILDREN_WRITE);
                        objRoot.setCapability(BranchGroup.ALLOW_CHILDREN_READ);
                        tg = new TransformGroup();
                        t3d = new Transform3D();

                        tg.setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);

                        t3d.setTranslation(new Vector3d(-0.15, -0.3, -5.0));
                        t3d.setRotation(new AxisAngle4f(1f, 1f, 1f, 1f));
                        t3d.setScale(getInitialScale());

                        tg.setTransform(t3d);


                        objRoot.addChild(tg);



                        int jagged = getJagged();
                        double[] rectDip = getRectDip();

                        int boxX = 16;

                        createVoxelizedShape(tg, boxX, jagged, rectDip);

                        float directionalPower = 0.55f;

                        final Color3f color3f = new Color3f(directionalPower, directionalPower, directionalPower);

                        DirectionalLight directionalLight = new DirectionalLight(true, color3f, new Vector3f(-0.3f, 0.2f, -1.0f));

                        directionalLight.setInfluencingBounds(new BoundingSphere(new Point3d(), 10000.0));

                        objRoot.addChild(directionalLight);


                        AmbientLight ambientLight = new AmbientLight(true, new Color3f(1f - directionalPower, 1f - directionalPower, 1f - directionalPower));

                        ambientLight.setInfluencingBounds(new BoundingSphere(new Point3d(), 10000.0));

                        objRoot.addChild(ambientLight);


                        MouseRotate myMouseRotate = new MouseRotate();
                        myMouseRotate.setTransformGroup(tg);
                        myMouseRotate.setSchedulingBounds(new BoundingSphere());
                        objRoot.addChild(myMouseRotate);

                        MouseWheelZoom mouseWheelZoom = new MouseWheelZoom();
                        mouseWheelZoom.setTransformGroup(tg);
                        mouseWheelZoom.setSchedulingBounds(tg.getBounds());
                        objRoot.addChild(mouseWheelZoom);
                        objRoot.compile();

                        scene.addChild(objRoot);

                        Background background = new Background();
                        background.setColor(getBackgroundColor());
                        background.setApplicationBounds(bounds);
                        scene.addChild(background);
                        universe.addBranchGraph(scene);

                        SwingUtilities.invokeLater(() -> {
                            try {
                                canvasPanel.removeAll();
                                canvasPanel.add(canvas);
                                canvasPanel.repaint();
                                canvasPanel.revalidate();

                            } catch (Exception e) {
                                set3dLoadedStatus("<html>ERROR: " + e.getMessage() + "</html>");
                            }
                        });
                    } catch (Exception e) {
                        e.printStackTrace();
                        SwingUtilities.invokeLater(() -> {
                            set3dLoadedStatus("something went wrong, try again");
                        });
                    }
                }
            };
            new Thread(runnable).start();
        }

        private double getInitialScale() {
            return 3;
        }

        protected int getJagged() {
            return 1;
        }

        protected double[] getRectDip() {

            return new double[]{1, 1, 1};
        }

        protected void set3dLoadedStatus() {
            set3dLoadedStatus("3D model is loading ...");
        }

        protected void set3dLoadedStatus(String message) {
            canvasPanel.removeAll();
            canvasPanel.add(new JLabel(message));
            canvasPanel.repaint();
            canvasPanel.revalidate();
        }

        public void createVoxelizedShape(TransformGroup tg, int boxX, int jagged, double[] rectDip) {
            this.jagged = jagged;

            double gridspace = 1.0 / boxX;

            double xr, yr, zr;
            double rectScaleX, rectScaleY, rectScaleZ;
            int xj, yj, zj;

            rectScaleX = rectDip[0];
            rectScaleY = rectDip[1];
            rectScaleZ = rectDip[2];

            double gridSpaceX = gridspace * rectScaleX;
            double gridSpaceY = gridspace * rectScaleY;
            double gridSpaceZ = gridspace * rectScaleZ;


            ArrayList<Point3d> point3dArrayList = new ArrayList<Point3d>();

            int boxY = getBoxY(boxX, rectScaleX, rectScaleY, rectScaleZ, jagged);
            int boxZ = getBoxZ(boxX, rectScaleX, rectScaleY, rectScaleZ, jagged);


            for (int k = 0; k < boxZ; k++)
                for (int j = 0; j < boxY; j++)
                    for (int i = 0; i < boxX; i++) {
                        xj = 2 * jagged * (i / jagged) + jagged - boxX;
                        yj = 2 * jagged * (j / jagged) + jagged - boxY;
                        zj = 2 * jagged * (k / jagged) + jagged - boxZ;
                        /* all the following coordinates should be scaled by the same sizeX. So we scale xj,yj,zj by 2boxX with extra
                         * ratio for rectangular dipoles. Thus, yr and zr are not necessarily in fixed ranges (like from -1/2 to 1/2).
                         * This is done to treat adequately cases when particle dimensions are the same (along different axes), but e.g.
                         * boxY!=boxX (so there are some extra void dipoles). All anisotropies in the particle itself are treated in
                         * the specific shape modules below (see e.g. ELLIPSOID).
                         */
                        xr = (0.5 * xj) / boxX;
                        yr = (0.5 * yj) / boxX * (rectScaleY / rectScaleX);
                        zr = (0.5 * zj) / boxX * (rectScaleZ / rectScaleX);

                        if (isPointInsideShapeVolume(xr, yr, zr)) {
                            addVoxel(xr, yr, zr, gridSpaceX, gridSpaceY, gridSpaceZ, point3dArrayList);
                        }
                    }


            if (point3dArrayList.size() < 1) {
                return;
            }

            ArrayList<Double> points = new ArrayList<Double>();
            for (Point3d point3d : point3dArrayList) {
                points.add(point3d.x);
                points.add(point3d.y);
                points.add(point3d.z);
            }

            GeometryInfo gi = new GeometryInfo(GeometryInfo.QUAD_ARRAY);
            gi.setCoordinates(points.stream().mapToDouble(i -> i).toArray());
            NormalGenerator normalGenerator = new NormalGenerator();
            normalGenerator.generateNormals(gi);
            Appearance ap = new Appearance();
            PolygonAttributes polyAttrbutes = new PolygonAttributes();
            polyAttrbutes.setPolygonMode(PolygonAttributes.POLYGON_FILL);
            polyAttrbutes.setCullFace(PolygonAttributes.CULL_NONE);
            ap.setPolygonAttributes(polyAttrbutes);
            Material material = new Material();
            material.setShininess(50f);
            material.setAmbientColor(getVoxelColor());
            material.setDiffuseColor(getVoxelColor());
            ap.setMaterial(material);


            Shape3D shape1 = new Shape3D(gi.getGeometryArray(), ap);
            tg.addChild(shape1);

            ap = new Appearance();
            polyAttrbutes = new PolygonAttributes();
            polyAttrbutes.setPolygonMode(PolygonAttributes.POLYGON_LINE);
            polyAttrbutes.setCullFace(PolygonAttributes.CULL_NONE);
            ap.setPolygonAttributes(polyAttrbutes);
            material = new Material();
            material.setShininess(50f);
            material.setDiffuseColor(getVoxelLineColor());
            material.setAmbientColor(getVoxelLineColor());
            ap.setMaterial(material);
            Shape3D shape2 = new Shape3D(gi.getGeometryArray(), ap);
            tg.addChild(shape2);
        }


        public int getBoxY(int boxX, double rectScaleX, double rectScaleY, double rectScaleZ, int jagged) {
            return fitBox_yz(boxX * (rectScaleX / rectScaleY), jagged);
        }


        public int getBoxZ(int boxX, double rectScaleX, double rectScaleY, double rectScaleZ, int jagged) {
            return fitBox_yz((firstParam + 1) * boxX * (rectScaleX / rectScaleZ), jagged);
        }

        double firstParam = 0.5;

        protected boolean isPointInsideShapeVolume(double xr, double yr, double zr) {
            double ro2 = xr * xr + yr * yr;
            if (ro2 <= 0.25) {
                double tmp1 = Math.abs(zr) - firstParam * 0.5;
                if (tmp1 <= 0 || tmp1 * tmp1 + ro2 <= 0.25) {
                    return true;
                }
            }
            return false;
        }

        protected void addVoxel(double xr, double yr, double zr, double gridSpaceX, double gridSpaceY, double gridSpaceZ, ArrayList<Point3d> point3dArrayList) {
            Point3d point1 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);
            Point3d point2 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);
            Point3d point3 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);
            Point3d point4 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);

            Point3d point5 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);
            Point3d point6 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);
            Point3d point7 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);
            Point3d point8 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);


            point3dArrayList.add(point1);
            point3dArrayList.add(point2);
            point3dArrayList.add(point3);
            point3dArrayList.add(point4);
            point3dArrayList.add(point5);
            point3dArrayList.add(point6);
            point3dArrayList.add(point7);
            point3dArrayList.add(point8);


            Point3d point9 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);
            Point3d point10 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);
            Point3d point11 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);
            Point3d point12 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);

            Point3d point13 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);
            Point3d point14 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);
            Point3d point15 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);
            Point3d point16 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);

            point3dArrayList.add(point9);
            point3dArrayList.add(point10);
            point3dArrayList.add(point11);
            point3dArrayList.add(point12);
            point3dArrayList.add(point13);
            point3dArrayList.add(point14);
            point3dArrayList.add(point15);
            point3dArrayList.add(point16);


            Point3d point17 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);
            Point3d point18 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);
            Point3d point19 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);
            Point3d point20 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr + jagged * 0.5 * gridSpaceZ);

            Point3d point21 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);
            Point3d point22 = new Point3d(xr + jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);
            Point3d point23 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr - jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);
            Point3d point24 = new Point3d(xr - jagged * 0.5 * gridSpaceX, yr + jagged * 0.5 * gridSpaceY, zr - jagged * 0.5 * gridSpaceZ);

            point3dArrayList.add(point17);
            point3dArrayList.add(point18);
            point3dArrayList.add(point19);
            point3dArrayList.add(point20);
            point3dArrayList.add(point21);
            point3dArrayList.add(point22);
            point3dArrayList.add(point23);
            point3dArrayList.add(point24);
        }

        public static int fitBox(int box, int jagged) {
            return jagged * ((box + jagged - 1) / jagged);
        }

        public static int fitBox_yz(double size, int jagged)
            /* given the size of the particle in y or z direction (in units of dipoles), finds the closest grid size, which would
             * satisfy the FitBox function. The rounding is performed so to minimize the maximum difference between the stack of
             * dipoles and corresponding particle dimension.
             * The distance between the center of the outer super-dipole (J^3 original dipoles) and the particle (enclosing box)
             * boundary is between 0.25 and 0.75 of super-dipole size, corresponding to the discretization along the x-axis, for
             * which the optimum distance of 0.5 (the dipole cube fits tight into the boundary) is automatically satisfied.
             *
             * !!! However, it is still possible that the whole outer layer of dipoles would be void because the estimate does not
             * take into account the details of the shape e.g. its curvature. For instance, 'adda -grid 4 -shape ellipsoid 1 2.13'
             * results in grid 4x4x9, but the layers z=0, z=8 will be void. This is because the dipole centers in this layers always
             * have non-zero x and y coordinates (at least half-dipole in absolute value) and do not fall inside the ellipsoid,
             * although the points {+-4,0,0} do fall into it.
             */ {
            return jagged * (int) Math.round(size / jagged);
        }


        public Color3f getBackgroundColor() {
            return new Color3f(0.75f, 0.69f, 0.680f);
        }

        protected Color3f getVoxelColor() {
            return new Color3f(255, 0, 255);
        }

        protected Color3f getVoxelLineColor() {
            return new Color3f(0, 0, 0);
        }


        protected Color3f getSurfaceColor() {
            return new Color3f(255, 0, 255);
        }

        ;

        protected String error;

        public String getError() {
            return error;
        }
    }