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

org.abego.treelayout.TreeLayout Maven / Gradle / Ivy

/*
 * [The "BSD license"]
 * Copyright (c) 2011, abego Software GmbH, Germany (http://www.abego.org)
 * 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.
 * 3. Neither the name of the abego Software GmbH nor the names of its 
 *    contributors may be used to endorse or promote products derived from this 
 *    software without specific prior written permission.
 *    
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "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 THE COPYRIGHT HOLDER 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.
 */

package org.abego.treelayout;

import static org.abego.treelayout.internal.util.Contract.checkArg;

import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D;
import java.io.PrintStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.IdentityHashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

import org.abego.treelayout.Configuration.AlignmentInLevel;
import org.abego.treelayout.Configuration.Location;
import org.abego.treelayout.internal.util.java.lang.string.StringUtil;

/**
 * Implements the actual tree layout algorithm.
 * 

* The nodes with their final layout can be retrieved through * {@link #getNodeBounds()}. *

* See this summary to get an overview how to * use TreeLayout. * * * @author Udo Borkowski ([email protected]) * * @param Type of elements used as nodes in the tree */ public class TreeLayout { /* * Differences between this implementation and original algorithm * -------------------------------------------------------------- * * For easier reference the same names (or at least similar names) as in the * paper of Buchheim, Jünger, and Leipert are used in this * implementation. However in the external interface "first" and "last" are * used instead of "left most" and "right most". The implementation also * supports tree layouts with the root at the left (or right) side. In that * case using "left most" would refer to the "top" child, i.e. using "first" * is less confusing. * * Also the y coordinate is not the level but directly refers the y * coordinate of a level, taking node's height and gapBetweenLevels into * account. When the root is at the left or right side the y coordinate * actually becomes an x coordinate. * * Instead of just using a constant "distance" to calculate the position to * the next node we refer to the "size" (width or height) of the node and a * "gapBetweenNodes". */ // ------------------------------------------------------------------------ // tree private final TreeForTreeLayout tree; /** * Returns the Tree the layout is created for. * * @return the Tree the layout is created for */ public TreeForTreeLayout getTree() { return tree; } // ------------------------------------------------------------------------ // nodeExtentProvider private final NodeExtentProvider nodeExtentProvider; /** * Returns the {@link NodeExtentProvider} used by this {@link TreeLayout}. * * @return the {@link NodeExtentProvider} used by this {@link TreeLayout} */ public NodeExtentProvider getNodeExtentProvider() { return nodeExtentProvider; } private double getNodeHeight(TreeNode node) { return nodeExtentProvider.getHeight(node); } private double getNodeWidth(TreeNode node) { return nodeExtentProvider.getWidth(node); } private double getWidthOrHeightOfNode(TreeNode treeNode, boolean returnWidth) { return returnWidth ? getNodeWidth(treeNode) : getNodeHeight(treeNode); } /** * When the level changes in Y-axis (i.e. root location Top or Bottom) the * height of a node is its thickness, otherwise the node's width is its * thickness. *

* The thickness of a node is used when calculating the locations of the * levels. * * @param treeNode * @return */ private double getNodeThickness(TreeNode treeNode) { return getWidthOrHeightOfNode(treeNode, !isLevelChangeInYAxis()); } /** * When the level changes in Y-axis (i.e. root location Top or Bottom) the * width of a node is its size, otherwise the node's height is its size. *

* The size of a node is used when calculating the distance between two * nodes. * * @param treeNode * @return */ private double getNodeSize(TreeNode treeNode) { return getWidthOrHeightOfNode(treeNode, isLevelChangeInYAxis()); } // ------------------------------------------------------------------------ // configuration private final Configuration configuration; /** * Returns the Configuration used by this {@link TreeLayout}. * * @return the Configuration used by this {@link TreeLayout} */ public Configuration getConfiguration() { return configuration; } private boolean isLevelChangeInYAxis() { Location rootLocation = configuration.getRootLocation(); return rootLocation == Location.Top || rootLocation == Location.Bottom; } private int getLevelChangeSign() { Location rootLocation = configuration.getRootLocation(); return rootLocation == Location.Bottom || rootLocation == Location.Right ? -1 : 1; } // ------------------------------------------------------------------------ // bounds private double boundsLeft = Double.MAX_VALUE; private double boundsRight = Double.MIN_VALUE; private double boundsTop = Double.MAX_VALUE; private double boundsBottom = Double.MIN_VALUE; private void updateBounds(TreeNode node, double centerX, double centerY) { double width = getNodeWidth(node); double height = getNodeHeight(node); double left = centerX - width / 2; double right = centerX + width / 2; double top = centerY - height / 2; double bottom = centerY + height / 2; if (boundsLeft > left) { boundsLeft = left; } if (boundsRight < right) { boundsRight = right; } if (boundsTop > top) { boundsTop = top; } if (boundsBottom < bottom) { boundsBottom = bottom; } } /** * Returns the bounds of the tree layout. *

* The bounds of a TreeLayout is the smallest rectangle containing the * bounds of all nodes in the layout. It always starts at (0,0). * * @return the bounds of the tree layout */ public Rectangle2D getBounds() { return new Rectangle2D.Double(0, 0, boundsRight - boundsLeft, boundsBottom - boundsTop); } // ------------------------------------------------------------------------ // size of level private final List sizeOfLevel = new ArrayList(); private void calcSizeOfLevels(TreeNode node, int level) { double oldSize; if (sizeOfLevel.size() <= level) { sizeOfLevel.add(Double.valueOf(0)); oldSize = 0; } else { oldSize = sizeOfLevel.get(level); } double size = getNodeThickness(node); // size = nodeExtentProvider.getHeight(node); if (oldSize < size) { sizeOfLevel.set(level, size); } if (!tree.isLeaf(node)) { for (TreeNode child : tree.getChildren(node)) { calcSizeOfLevels(child, level + 1); } } } /** * Returns the number of levels of the tree. * * @return [level > 0] */ public int getLevelCount() { return sizeOfLevel.size(); } /** * Returns the size of a level. *

* When the root is located at the top or bottom the size of a level is the * maximal height of the nodes of that level. When the root is located at * the left or right the size of a level is the maximal width of the nodes * of that level. * * @param level   * @return the size of the level [level >= 0 && level < levelCount] */ public double getSizeOfLevel(int level) { checkArg(level >= 0, "level must be >= 0"); checkArg(level < getLevelCount(), "level must be < levelCount"); return sizeOfLevel.get(level); } // ------------------------------------------------------------------------ // NormalizedPosition /** * The algorithm calculates the position starting with the root at 0. I.e. * the left children will get negative positions. However we want the result * to be normalized to (0,0). *

* {@link NormalizedPosition} will normalize the position (given relative to * the root position), taking the current bounds into account. This way the * left most node bounds will start at x = 0, the top most node bounds at y * = 0. */ private class NormalizedPosition extends Point2D { private double x_relativeToRoot; private double y_relativeToRoot; public NormalizedPosition(double x_relativeToRoot, double y_relativeToRoot) { setLocation(x_relativeToRoot, y_relativeToRoot); } @Override public double getX() { return x_relativeToRoot - boundsLeft; } @Override public double getY() { return y_relativeToRoot - boundsTop; } @Override // never called from outside public void setLocation(double x_relativeToRoot, double y_relativeToRoot) { this.x_relativeToRoot = x_relativeToRoot; this.y_relativeToRoot = y_relativeToRoot; } } // ------------------------------------------------------------------------ // The Algorithm private final boolean useIdentity; private final Map mod; private final Map thread; private final Map prelim; private final Map change; private final Map shift; private final Map ancestor; private final Map number; private final Map positions; private double getMod(TreeNode node) { Double d = mod.get(node); return d != null ? d.doubleValue() : 0; } private void setMod(TreeNode node, double d) { mod.put(node, d); } private TreeNode getThread(TreeNode node) { TreeNode n = thread.get(node); return n != null ? n : null; } private void setThread(TreeNode node, TreeNode thread) { this.thread.put(node, thread); } private TreeNode getAncestor(TreeNode node) { TreeNode n = ancestor.get(node); return n != null ? n : node; } private void setAncestor(TreeNode node, TreeNode ancestor) { this.ancestor.put(node, ancestor); } private double getPrelim(TreeNode node) { Double d = prelim.get(node); return d != null ? d.doubleValue() : 0; } private void setPrelim(TreeNode node, double d) { prelim.put(node, d); } private double getChange(TreeNode node) { Double d = change.get(node); return d != null ? d.doubleValue() : 0; } private void setChange(TreeNode node, double d) { change.put(node, d); } private double getShift(TreeNode node) { Double d = shift.get(node); return d != null ? d.doubleValue() : 0; } private void setShift(TreeNode node, double d) { shift.put(node, d); } /** * The distance of two nodes is the distance of the centers of both noded. *

* I.e. the distance includes the gap between the nodes and half of the * sizes of the nodes. * * @param v * @param w * @return the distance between node v and w */ private double getDistance(TreeNode v, TreeNode w) { double sizeOfNodes = getNodeSize(v) + getNodeSize(w); double distance = sizeOfNodes / 2 + configuration.getGapBetweenNodes(v, w); return distance; } private TreeNode nextLeft(TreeNode v) { return tree.isLeaf(v) ? getThread(v) : tree.getFirstChild(v); } private TreeNode nextRight(TreeNode v) { return tree.isLeaf(v) ? getThread(v) : tree.getLastChild(v); } /** * * @param node * [tree.isChildOfParent(node, parentNode)] * @param parentNode * parent of node * @return */ private int getNumber(TreeNode node, TreeNode parentNode) { Integer n = number.get(node); if (n == null) { int i = 1; for (TreeNode child : tree.getChildren(parentNode)) { number.put(child, i++); } n = number.get(node); } return n.intValue(); } /** * * @param vIMinus * @param v * @param parentOfV * @param defaultAncestor * @return the greatest distinct ancestor of vIMinus and its right neighbor * v */ private TreeNode ancestor(TreeNode vIMinus, TreeNode v, TreeNode parentOfV, TreeNode defaultAncestor) { TreeNode ancestor = getAncestor(vIMinus); // when the ancestor of vIMinus is a sibling of v (i.e. has the same // parent as v) it is also the greatest distinct ancestor vIMinus and // v. Otherwise it is the defaultAncestor return tree.isChildOfParent(ancestor, parentOfV) ? ancestor : defaultAncestor; } private void moveSubtree(TreeNode wMinus, TreeNode wPlus, TreeNode parent, double shift) { int subtrees = getNumber(wPlus, parent) - getNumber(wMinus, parent); setChange(wPlus, getChange(wPlus) - shift / subtrees); setShift(wPlus, getShift(wPlus) + shift); setChange(wMinus, getChange(wMinus) + shift / subtrees); setPrelim(wPlus, getPrelim(wPlus) + shift); setMod(wPlus, getMod(wPlus) + shift); } /** * In difference to the original algorithm we also pass in the leftSibling * and the parent of v. *

* Why adding the parameter 'parent of v' (parentOfV) ? *

* In this method we need access to the parent of v. Not every tree * implementation may support efficient (i.e. constant time) access to it. * On the other hand the (only) caller of this method can provide this * information with only constant extra time. *

* Also we need access to the "left most sibling" of v. Not every tree * implementation may support efficient (i.e. constant time) access to it. * On the other hand the "left most sibling" of v is also the "first child" * of the parent of v. The first child of a parent node we can get in * constant time. As we got the parent of v we can so also get the * "left most sibling" of v in constant time. *

* Why adding the parameter 'leftSibling' ? *

* In this method we need access to the "left sibling" of v. Not every tree * implementation may support efficient (i.e. constant time) access to it. * However it is easy for the caller of this method to provide this * information with only constant extra time. *

*

*

* In addition these extra parameters avoid the need for * {@link TreeForTreeLayout} to include extra methods "getParent", * "getLeftSibling", or "getLeftMostSibling". This keeps the interface * {@link TreeForTreeLayout} small and avoids redundant implementations. * * @param v * @param defaultAncestor * @param leftSibling * [nullable] the left sibling v, if there is any * @param parentOfV * the parent of v * @return the (possibly changes) defaultAncestor */ private TreeNode apportion(TreeNode v, TreeNode defaultAncestor, TreeNode leftSibling, TreeNode parentOfV) { TreeNode w = leftSibling; if (w == null) { // v has no left sibling return defaultAncestor; } // v has left sibling w // The following variables "v..." are used to traverse the contours to // the subtrees. "Minus" refers to the left, "Plus" to the right // subtree. "I" refers to the "inside" and "O" to the outside contour. TreeNode vOPlus = v; TreeNode vIPlus = v; TreeNode vIMinus = w; // get leftmost sibling of vIPlus, i.e. get the leftmost sibling of // v, i.e. the leftmost child of the parent of v (which is passed // in) TreeNode vOMinus = tree.getFirstChild(parentOfV); Double sIPlus = getMod(vIPlus); Double sOPlus = getMod(vOPlus); Double sIMinus = getMod(vIMinus); Double sOMinus = getMod(vOMinus); TreeNode nextRightVIMinus = nextRight(vIMinus); TreeNode nextLeftVIPlus = nextLeft(vIPlus); while (nextRightVIMinus != null && nextLeftVIPlus != null) { vIMinus = nextRightVIMinus; vIPlus = nextLeftVIPlus; vOMinus = nextLeft(vOMinus); vOPlus = nextRight(vOPlus); setAncestor(vOPlus, v); double shift = (getPrelim(vIMinus) + sIMinus) - (getPrelim(vIPlus) + sIPlus) + getDistance(vIMinus, vIPlus); if (shift > 0) { moveSubtree(ancestor(vIMinus, v, parentOfV, defaultAncestor), v, parentOfV, shift); sIPlus = sIPlus + shift; sOPlus = sOPlus + shift; } sIMinus = sIMinus + getMod(vIMinus); sIPlus = sIPlus + getMod(vIPlus); sOMinus = sOMinus + getMod(vOMinus); sOPlus = sOPlus + getMod(vOPlus); nextRightVIMinus = nextRight(vIMinus); nextLeftVIPlus = nextLeft(vIPlus); } if (nextRightVIMinus != null && nextRight(vOPlus) == null) { setThread(vOPlus, nextRightVIMinus); setMod(vOPlus, getMod(vOPlus) + sIMinus - sOPlus); } if (nextLeftVIPlus != null && nextLeft(vOMinus) == null) { setThread(vOMinus, nextLeftVIPlus); setMod(vOMinus, getMod(vOMinus) + sIPlus - sOMinus); defaultAncestor = v; } return defaultAncestor; } /** * * @param v * [!tree.isLeaf(v)] */ private void executeShifts(TreeNode v) { double shift = 0; double change = 0; for (TreeNode w : tree.getChildrenReverse(v)) { change = change + getChange(w); setPrelim(w, getPrelim(w) + shift); setMod(w, getMod(w) + shift); shift = shift + getShift(w) + change; } } /** * In difference to the original algorithm we also pass in the leftSibling * (see {@link #apportion(Object, Object, Object, Object)} for a * motivation). * * @param v * @param leftSibling * [nullable] the left sibling v, if there is any */ private void firstWalk(TreeNode v, TreeNode leftSibling) { if (tree.isLeaf(v)) { // No need to set prelim(v) to 0 as the getter takes care of this. TreeNode w = leftSibling; if (w != null) { // v has left sibling setPrelim(v, getPrelim(w) + getDistance(v, w)); } } else { // v is not a leaf TreeNode defaultAncestor = tree.getFirstChild(v); TreeNode previousChild = null; for (TreeNode w : tree.getChildren(v)) { firstWalk(w, previousChild); defaultAncestor = apportion(w, defaultAncestor, previousChild, v); previousChild = w; } executeShifts(v); double midpoint = (getPrelim(tree.getFirstChild(v)) + getPrelim(tree .getLastChild(v))) / 2.0; TreeNode w = leftSibling; if (w != null) { // v has left sibling setPrelim(v, getPrelim(w) + getDistance(v, w)); setMod(v, getPrelim(v) - midpoint); } else { // v has no left sibling setPrelim(v, midpoint); } } } /** * In difference to the original algorithm we also pass in extra level * information. * * @param v * @param m * @param level * @param levelStart */ private void secondWalk(TreeNode v, double m, int level, double levelStart) { // construct the position from the prelim and the level information // The rootLocation affects the way how x and y are changed and in what // direction. double levelChangeSign = getLevelChangeSign(); boolean levelChangeOnYAxis = isLevelChangeInYAxis(); double levelSize = getSizeOfLevel(level); double x = getPrelim(v) + m; double y; AlignmentInLevel alignment = configuration.getAlignmentInLevel(); if (alignment == AlignmentInLevel.Center) { y = levelStart + levelChangeSign * (levelSize / 2); } else if (alignment == AlignmentInLevel.TowardsRoot) { y = levelStart + levelChangeSign * (getNodeThickness(v) / 2); } else { y = levelStart + levelSize - levelChangeSign * (getNodeThickness(v) / 2); } if (!levelChangeOnYAxis) { double t = x; x = y; y = t; } positions.put(v, new NormalizedPosition(x, y)); // update the bounds updateBounds(v, x, y); // recurse if (!tree.isLeaf(v)) { double nextLevelStart = levelStart + (levelSize + configuration.getGapBetweenLevels(level + 1)) * levelChangeSign; for (TreeNode w : tree.getChildren(v)) { secondWalk(w, m + getMod(v), level + 1, nextLevelStart); } } } // ------------------------------------------------------------------------ // nodeBounds private Map nodeBounds; /** * Returns the layout of the tree nodes by mapping each node of the tree to * its bounds (position and size). *

* For each rectangle x and y will be >= 0. At least one rectangle will have * an x == 0 and at least one rectangle will have an y == 0. * * @return maps each node of the tree to its bounds (position and size). */ public Map getNodeBounds() { if (nodeBounds == null) { nodeBounds = this.useIdentity ? new IdentityHashMap() : new HashMap(); for (Entry entry : positions.entrySet()) { TreeNode node = entry.getKey(); Point2D pos = entry.getValue(); double w = getNodeWidth(node); double h = getNodeHeight(node); double x = pos.getX() - w / 2; double y = pos.getY() - h / 2; nodeBounds.put(node, new Rectangle2D.Double(x, y, w, h)); } } return nodeBounds; } // ------------------------------------------------------------------------ // constructor /** * Creates a TreeLayout for a given tree. *

* In addition to the tree the {@link NodeExtentProvider} and the * {@link Configuration} must be given. * * @param tree   * @param nodeExtentProvider   * @param configuration   * @param useIdentity * [default: false] when true, identity ("==") is used instead of * equality ("equals(...)") when checking nodes. Within a tree * each node must only exist once (using this check). */ public TreeLayout(TreeForTreeLayout tree, NodeExtentProvider nodeExtentProvider, Configuration configuration, boolean useIdentity) { this.tree = tree; this.nodeExtentProvider = nodeExtentProvider; this.configuration = configuration; this.useIdentity = useIdentity; if (this.useIdentity) { this.mod = new IdentityHashMap(); this.thread = new IdentityHashMap(); this.prelim = new IdentityHashMap(); this.change = new IdentityHashMap(); this.shift = new IdentityHashMap(); this.ancestor = new IdentityHashMap(); this.number = new IdentityHashMap(); this.positions = new IdentityHashMap(); } else { this.mod = new HashMap(); this.thread = new HashMap(); this.prelim = new HashMap(); this.change = new HashMap(); this.shift = new HashMap(); this.ancestor = new HashMap(); this.number = new HashMap(); this.positions = new HashMap(); } // No need to explicitly set mod, thread and ancestor as their getters // are taking care of the initial values. This avoids a full tree walk // through and saves some memory as no entries are added for // "initial values". TreeNode r = tree.getRoot(); firstWalk(r, null); calcSizeOfLevels(r, 0); secondWalk(r, -getPrelim(r), 0, 0); } public TreeLayout(TreeForTreeLayout tree, NodeExtentProvider nodeExtentProvider, Configuration configuration) { this(tree, nodeExtentProvider, configuration, false); } // ------------------------------------------------------------------------ // checkTree private void addUniqueNodes(Map nodes, TreeNode newNode) { if (nodes.put(newNode,newNode) != null) { throw new RuntimeException(String.format( "Node used more than once in tree: %s", newNode)); } for (TreeNode n : tree.getChildren(newNode)) { addUniqueNodes(nodes,n); } } /** * Check if the tree is a "valid" tree. *

* Typically you will use this method during development when you get an * unexpected layout from your trees. *

* The following checks are performed: *

    *
  • Each node must only occur once in the tree.
  • *
*/ public void checkTree() { Map nodes = this.useIdentity ? new IdentityHashMap() :new HashMap(); // Traverse the tree and check if each node is only used once. addUniqueNodes(nodes,tree.getRoot()); } // ------------------------------------------------------------------------ // dumpTree private void dumpTree(PrintStream output, TreeNode node, int indent, DumpConfiguration dumpConfiguration) { StringBuilder sb = new StringBuilder(); for (int i = 0; i < indent; i++) { sb.append(dumpConfiguration.indent); } if (dumpConfiguration.includeObjectToString) { sb.append("["); sb.append(node.getClass().getName() + "@" + Integer.toHexString(node.hashCode())); if (node.hashCode() != System.identityHashCode(node)) { sb.append("/identityHashCode:"); sb.append(Integer.toHexString(System.identityHashCode(node))); } sb.append("]"); } sb.append(StringUtil.quote(node != null ? node.toString() : null)); if (dumpConfiguration.includeNodeSize) { sb.append(" (size: "); sb.append(getNodeWidth(node)); sb.append("x"); sb.append(getNodeHeight(node)); sb.append(")"); } output.println(sb.toString()); for (TreeNode n : tree.getChildren(node)) { dumpTree(output, n, indent + 1, dumpConfiguration); } } public static class DumpConfiguration { /** * The text used to indent the output per level. */ public final String indent; /** * When true the dump also includes the size of each node, otherwise * not. */ public final boolean includeNodeSize; /** * When true, the text as returned by {@link Object#toString()}, is * included in the dump, in addition to the text returned by the * possibly overridden toString method of the node. When the hashCode * method is overridden the output will also include the * "identityHashCode". */ public final boolean includeObjectToString; /** * * @param indent [default: " "] * @param includeNodeSize [default: false] * @param includePointer [default: false] */ public DumpConfiguration(String indent, boolean includeNodeSize, boolean includePointer) { this.indent = indent; this.includeNodeSize = includeNodeSize; this.includeObjectToString = includePointer; } public DumpConfiguration() { this(" ",false,false); } } /** * Prints a dump of the tree to the given printStream, using the node's * "toString" method. * * @param printStream   * @param dumpConfiguration * [default: new DumpConfiguration()] */ public void dumpTree(PrintStream printStream, DumpConfiguration dumpConfiguration) { dumpTree(printStream,tree.getRoot(),0, dumpConfiguration); } public void dumpTree(PrintStream printStream) { dumpTree(printStream,new DumpConfiguration()); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy