aboutsummaryrefslogtreecommitdiffstats
path: root/src/graphui/classes/com/jogamp/graph/ui/Group.java
diff options
context:
space:
mode:
Diffstat (limited to 'src/graphui/classes/com/jogamp/graph/ui/Group.java')
-rw-r--r--src/graphui/classes/com/jogamp/graph/ui/Group.java249
1 files changed, 249 insertions, 0 deletions
diff --git a/src/graphui/classes/com/jogamp/graph/ui/Group.java b/src/graphui/classes/com/jogamp/graph/ui/Group.java
new file mode 100644
index 000000000..8174b279c
--- /dev/null
+++ b/src/graphui/classes/com/jogamp/graph/ui/Group.java
@@ -0,0 +1,249 @@
+/**
+ * Copyright 2023 JogAmp Community. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without modification, are
+ * permitted provided that the following conditions are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright notice, this list of
+ * conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright notice, this list
+ * of conditions and the following disclaimer in the documentation and/or other materials
+ * provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY JogAmp Community ``AS IS'' AND ANY EXPRESS OR IMPLIED
+ * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+ * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JogAmp Community OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
+ * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+ * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
+ * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ *
+ * The views and conclusions contained in the software and documentation are those of the
+ * authors and should not be interpreted as representing official policies, either expressed
+ * or implied, of JogAmp Community.
+ */
+package com.jogamp.graph.ui;
+
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.Comparator;
+import java.util.List;
+
+import com.jogamp.graph.curve.opengl.GLRegion;
+import com.jogamp.graph.curve.opengl.RegionRenderer;
+import com.jogamp.opengl.GL2ES2;
+import com.jogamp.opengl.GLProfile;
+import com.jogamp.opengl.math.geom.AABBox;
+import com.jogamp.opengl.util.PMVMatrix;
+
+import jogamp.graph.ui.TreeTool;
+
+/**
+ * Group of UI {@link Shape}s, optionally utilizing a {@link Group.Layout}.
+ * @see Scene
+ * @see Shape
+ * @see Group.Layout
+ */
+public class Group extends Shape implements Container {
+ /** Layout for the group, called @ {@link Group#validate(GL2ES2)} or {@link Group#validate(GLProfile)}. */
+ public static interface Layout {
+ /** Performing the layout, called @ {@link Group#validate(GL2ES2)} or {@link Group#validate(GLProfile)}. */
+ void layout(Group g);
+ }
+
+ private final List<Shape> shapes = new ArrayList<Shape>();
+ private Layout layouter;
+
+ /**
+ * Create a Graph based {@link GLRegion} UI {@link Shape}.
+ */
+ public Group() {
+ super();
+ }
+
+ /**
+ * Create a Graph based {@link GLRegion} UI {@link Shape} w/ given {@link Group.Layour}.
+ */
+ public Group(final Layout l) {
+ super();
+ this.layouter = l;
+ }
+
+ /** Return current {@link Group.Layout}. */
+ public Layout getLayour() { return layouter; }
+
+ /** Set {@link Group.Layout}. */
+ public void setLayout(final Layout l) { layouter = l; }
+
+ @Override
+ public List<Shape> getShapes() {
+ return shapes;
+ }
+ @Override
+ public void addShape(final Shape s) {
+ shapes.add(s);
+ }
+
+ /** Removes given shape, keeps it alive. */
+ @Override
+ public void removeShape(final Shape s) {
+ shapes.remove(s);
+ }
+
+ /** Removes all given shapes and destroys them. */
+ public void removeShape(final GL2ES2 gl, final RegionRenderer renderer, final Shape s) {
+ s.setDebugBox(0f);
+ shapes.remove(s);
+ s.destroy(gl, renderer);
+ }
+
+ @Override
+ public void addShapes(final Collection<? extends Shape> shapes) {
+ for(final Shape s : shapes) {
+ addShape(s);
+ }
+ }
+ /** Removes all given shapes, keeps them alive. */
+ @Override
+ public void removeShapes(final Collection<? extends Shape> shapes) {
+ for(final Shape s : shapes) {
+ removeShape(s);
+ }
+ }
+ /** Removes all given shapes and destroys them. */
+ public void removeShapes(final GL2ES2 gl, final RegionRenderer renderer, final Collection<? extends Shape> shapes) {
+ for(final Shape s : shapes) {
+ removeShape(gl, renderer, s);
+ }
+ }
+
+ @Override
+ public boolean hasColorChannel() {
+ return false; // FIXME
+ }
+
+ @Override
+ protected final void clearImpl0(final GL2ES2 gl, final RegionRenderer renderer) {
+ for(final Shape s : shapes) {
+ // s.clearImpl0(gl, renderer);;
+ s.clear(gl, renderer);;
+ }
+ }
+
+ @Override
+ protected final void destroyImpl0(final GL2ES2 gl, final RegionRenderer renderer) {
+ for(final Shape s : shapes) {
+ // s.destroyImpl0(gl, renderer);
+ s.destroy(gl, renderer);;
+ }
+ }
+
+ private void layout() {
+ if( null != layouter ) {
+ layouter.layout(this);
+ }
+ }
+
+ private final boolean doFrustumCulling = true; // FIXME
+
+ @Override
+ protected final void drawImpl0(final GL2ES2 gl, final RegionRenderer renderer, final int[] sampleCount, final float[] rgba) {
+ final PMVMatrix pmv = renderer.getMatrix();
+ final int shapeCount = shapes.size();
+ for(int i=0; i<shapeCount; i++) {
+ final Shape shape = shapes.get(i);
+ if( shape.isEnabled() ) {
+ pmv.glPushMatrix();
+ shape.setTransform(pmv);
+
+ if( !doFrustumCulling || !pmv.glGetFrustum().isAABBoxOutside( shape.getBounds() ) ) {
+ if( null == rgba ) {
+ shape.drawToSelect(gl, renderer, sampleCount);
+ } else {
+ shape.draw(gl, renderer, sampleCount);
+ }
+ }
+ pmv.glPopMatrix();
+ }
+ }
+ }
+
+ @Override
+ protected void validateImpl(final GLProfile glp, final GL2ES2 gl) {
+ if( isShapeDirty() ) {
+ layout();
+ final PMVMatrix pmv = new PMVMatrix();
+ final AABBox tmpBox = new AABBox();
+ final float[] vec3Tmp0 = new float[3];
+ final float[] vec3Tmp1 = new float[3];
+ for(final Shape s : shapes) {
+ // s.validateImpl(glp, gl);
+ if( null != gl ) {
+ s.validate(gl);
+ } else {
+ s.validate(glp);
+ }
+ pmv.glPushMatrix();
+ s.setTransform(pmv);
+ s.getBounds().transformMv(tmpBox, pmv, vec3Tmp0, vec3Tmp0);
+ pmv.glPopMatrix();
+ box.resize(tmpBox);
+ }
+ }
+ }
+
+ @Override
+ public boolean contains(final Shape s) {
+ if( shapes.contains(s) ) {
+ return true;
+ }
+ for(final Shape shape : shapes) {
+ if( shape instanceof Container ) {
+ if( ((Container)shape).contains(s) ) {
+ return true;
+ }
+ }
+ }
+ return false;
+ }
+
+ @Override
+ public AABBox getBounds(final PMVMatrix pmv, final Shape shape) {
+ pmv.reset();
+ setTransform(pmv);
+ final AABBox res = new AABBox();
+ if( null == shape ) {
+ return res;
+ }
+ final float[] vec3Tmp0 = new float[3];
+ final float[] vec3Tmp1 = new float[3];
+ forOne(pmv, shape, () -> {
+ shape.getBounds().transformMv(res, pmv, vec3Tmp0, vec3Tmp1);
+ });
+ return res;
+ }
+
+ @Override
+ public boolean forOne(final PMVMatrix pmv, final Shape shape, final Runnable action) {
+ return TreeTool.forOne(shapes, pmv, shape, action);
+ }
+
+ @Override
+ public boolean forAll(final Visitor1 v) {
+ return TreeTool.forAll(shapes, v);
+ }
+
+ @Override
+ public boolean forAll(final PMVMatrix pmv, final Visitor2 v) {
+ return TreeTool.forAll(shapes, pmv, v);
+ }
+
+ @Override
+ public boolean forSortedAll(final Comparator<Shape> sortComp, final PMVMatrix pmv, final Visitor2 v) {
+ return TreeTool.forSortedAll(sortComp, shapes, pmv, v);
+ }
+}
+