diff --git a/Comparison/src/main/java/cz/fidentis/analyst/symmetry/AproxSymmetryPlane.java b/Comparison/src/main/java/cz/fidentis/analyst/symmetry/ApproxSymmetryPlane.java
similarity index 52%
rename from Comparison/src/main/java/cz/fidentis/analyst/symmetry/AproxSymmetryPlane.java
rename to Comparison/src/main/java/cz/fidentis/analyst/symmetry/ApproxSymmetryPlane.java
index 6ddae7b0d4dcbd1e3e1bdfefb2495105dc732ed2..7bc26d6373d4baacb35baf097c34653f687663f3 100644
--- a/Comparison/src/main/java/cz/fidentis/analyst/symmetry/AproxSymmetryPlane.java
+++ b/Comparison/src/main/java/cz/fidentis/analyst/symmetry/ApproxSymmetryPlane.java
@@ -1,31 +1,33 @@
 package cz.fidentis.analyst.symmetry;
 
 /**
+ * 
+ * Symmetry plane with votes used for the decision about symmetry estimate of 3D models.
  *
- * @author Natália Bebjaková
+ * @author Natalia Bebjakova
  * 
- * Respresents plane with added atribute - votes, that play role
- * in decision about symmetry estimate of the 3D model
  */
-public class AproxSymmetryPlane extends Plane implements Comparable<AproxSymmetryPlane> {
-    public Integer votes;
+public class ApproxSymmetryPlane extends Plane implements Comparable<ApproxSymmetryPlane> {
+    
+    private int votes;
 
     /**
      * returns number of votes that were given to plane while computing the symmetry 
      * 
      * @return Number of votes 
      */
-    public Integer getVotes() {
+    public int getVotes() {
         return votes;
     }
 
     /**
-     * 
+     * Constructor.
      * @param plane Original plane without votes 
      * @param votes number of votes given to the plane
+     * @throws IllegalArgumentExpcption if the @code{plane} argument is null
      */
-    public AproxSymmetryPlane(Plane plane, int votes) {
-        super(plane.a, plane.b, plane.c, plane.d);
+    public ApproxSymmetryPlane(Plane plane, int votes) {
+        super(plane);
         this.votes = votes;
     }
     
@@ -36,7 +38,7 @@ public class AproxSymmetryPlane extends Plane implements Comparable<AproxSymmetr
      * @return number that decides which plane has more votes 
      */
     @Override
-    public int compareTo(AproxSymmetryPlane other) {
-        return this.votes.compareTo(other.votes);
+    public int compareTo(ApproxSymmetryPlane other) {
+        return Integer.compare(votes, other.votes);
     }
 }
diff --git a/Comparison/src/main/java/cz/fidentis/analyst/symmetry/Plane.java b/Comparison/src/main/java/cz/fidentis/analyst/symmetry/Plane.java
index cb9cfaa3db6c98feb84d7b5b976d0c65f971f781..b2827693c2e1aaa9472ce16737afbc290fafb4b1 100644
--- a/Comparison/src/main/java/cz/fidentis/analyst/symmetry/Plane.java
+++ b/Comparison/src/main/java/cz/fidentis/analyst/symmetry/Plane.java
@@ -1,25 +1,19 @@
 package cz.fidentis.analyst.symmetry;
 
-import cz.fidentis.analyst.mesh.core.MeshFacet;
-import cz.fidentis.analyst.mesh.core.MeshFacetImpl;
-import cz.fidentis.analyst.mesh.core.MeshPointImpl;
 import javax.vecmath.Vector3d;
 
 /**
- *
- * @author Natália Bebjaková
+ * Symmetry plane.
  * 
- * Representation of the symmetry plane 
+ * @author Natalia Bebjakova
  */
 public class Plane {
-    public double a;
-    public double b;
-    public double c;
-    public double d;
-    private static MeshFacet facet;
+   
+    private Vector3d normal;
+    private double   distance;
 
     /**
-     * Creates new plane 
+     * Creates new plane.
      * 
      * @param a a coordinate 
      * @param b b coordinate
@@ -27,66 +21,36 @@ public class Plane {
      * @param d d coordinate
      */
     public Plane(double a, double b, double c, double d) {
-        this.a = a;
-        this.b = b;
-        this.c = c;
-        this.d = d;
+        normal = new Vector3d(a,b,c);
+        distance = d;
     }
     
     /**
-     * Normalize the plane
+     * Constructor.
+     * @param normal Normal vector of the plane
+     * @param dist distance
+     * @throws IllegalArgumentExpcption if the @code{plane} argument is null
      */
-    public void normalize() {
-        double normalLength = Math.sqrt(a * a + b * b + c * c);
-        
-        a /= normalLength;
-        b /= normalLength;
-        c /= normalLength;
-        d /= normalLength;
+    public Plane(Vector3d normal, double dist) {
+        this(normal.x, normal.y, normal.z, dist);
     }
     
     /**
-     * Returns plane that is represented by mesh which can be merged with model  
-     * 
-     * @param centroid middle point of the plane 
-     * @param a a coordinate
-     * @param b b coordinate
-     * @param scale distance of points given by bounding box
-     * @return plane represented as mesh 
+     * Copy constructor.
+     * @param plane original plane
+     * @throws IllegalArgumentExpcption if the @code{plane} argument is null
      */
-    public static SymmetryEstimator createPlaneMesh(Vector3d centroid, Vector3d a, Vector3d b, double scale) {
-        Vector3d[] points = new Vector3d[4];
-        
-        Vector3d aScaled = new Vector3d(a);
-        Vector3d bScaled = new Vector3d(b);
-        aScaled.scale(scale);
-        bScaled.scale(scale);
-        
-        points[0] = new Vector3d(centroid);
-        points[0].sub(aScaled);
-        points[0].sub(bScaled);
-        
-        points[1] = new Vector3d(centroid);
-        points[1].sub(aScaled);
-        points[1].add(bScaled);
-        
-        points[2] = new Vector3d(centroid);
-        points[2].add(aScaled);
-        points[2].add(bScaled);
-        
-        points[3] = new Vector3d(centroid);
-        points[3].add(aScaled);
-        points[3].sub(bScaled);
-        
-        facet = new MeshFacetImpl();
-        for (Vector3d point : points) {
-            facet.addVertex(new MeshPointImpl(point, null, null));
-        }
-        facet.calculateVertexNormals();
-
-        SymmetryEstimator planeMesh = new SymmetryEstimator(facet, Config.getDefault());
-        
-        return planeMesh;
+    public Plane(Plane plane) {
+        this(plane.getNormal(), plane.getDistance());
+    }
+    
+    /**
+     * Normalize the plane
+     */
+    public void normalize() {
+        double normalLength = normal.length();
+        normal.normalize();
+        distance /= normalLength; // Do we really want this? --ro
     }
     
     /**
@@ -96,6 +60,18 @@ public class Plane {
      */
     @Override
     public String toString(){
-        return "APPROXIMATE PLANE:\n" + a + "\n" + b + "\n" + c + "\n" + d + "\n";
+        return "APPROXIMATE PLANE:" + System.lineSeparator() + 
+                normal.x + System.lineSeparator() + 
+                normal.y + System.lineSeparator() + 
+                normal.z + System.lineSeparator() + 
+                distance + System.lineSeparator();
+    }
+    
+    public Vector3d getNormal() {
+        return normal;
+    }
+    
+    public double getDistance() {
+        return distance;
     }
 }
diff --git a/Comparison/src/main/java/cz/fidentis/analyst/symmetry/SymmetryEstimator.java b/Comparison/src/main/java/cz/fidentis/analyst/symmetry/SymmetryEstimator.java
index 50e9f043b3d572ea22ce7d7ac486c295fcabd574..0507e945fc054d4d25c6fd66b35143618acadf20 100644
--- a/Comparison/src/main/java/cz/fidentis/analyst/symmetry/SymmetryEstimator.java
+++ b/Comparison/src/main/java/cz/fidentis/analyst/symmetry/SymmetryEstimator.java
@@ -3,57 +3,78 @@ package cz.fidentis.analyst.symmetry;
 import cz.fidentis.analyst.mesh.core.CornerTableRow;
 import cz.fidentis.analyst.mesh.core.MeshFacet;
 import cz.fidentis.analyst.mesh.core.MeshPoint;
-import cz.fidentis.analyst.mesh.core.MeshPointImpl;
 import cz.fidentis.analyst.mesh.core.BoundingBox;
+import cz.fidentis.analyst.mesh.core.MeshFacetImpl;
+import cz.fidentis.analyst.mesh.core.MeshPointImpl;
 import cz.fidentis.analyst.mesh.core.MeshTriangle;
 import java.util.ArrayList;
-import java.util.Collection;
 import java.util.Collections;
-import java.util.HashMap;
 import java.util.List;
-import java.util.Map;
 import javax.swing.ImageIcon;
 import javax.swing.JOptionPane;
 import javax.swing.JPanel;
 import javax.swing.ProgressMonitor;
 import javax.swing.UIManager;
+//import javax.swing.ImageIcon;
+//import javax.swing.JOptionPane;
+//import javax.swing.JPanel;
+//import javax.swing.ProgressMonitor;
+//import javax.swing.UIManager;
 import javax.vecmath.Vector3d;
 
 /**
- *
- * @author Natália Bebjaková
- * 
  * Main class for computing approximate plane of symmetry of the 3D model.
  * For computing the symmetry, for every 
  * Default values of the configuration are given due to the best results on tested objects.
  * On many different 3D models, it exists other values of config that will have better impact on result. 
  * 
+ * @author Natalia Bebjakova
  */
 public class SymmetryEstimator {
+    
+    private final MeshFacet facet; // Facet of the model on which symmetry is computed 
+    
+    private TriangleVertexAreas[] areas; // Representation for areas of Voronoi region of triangles
+    
+    private BoundingBox boundingBox; // Represent min-max box. It is automatically maintained by given point array.
+    
+    private List<MeshTriangle> triangles; // Helping array of triangles computed from corner table 
+    
+    //private JPanel panel; // panel for configuration of symmetry counting 
+    
+    private final Config config;
+    
     /**
-     * Facet of the model on which symmetry is computed 
-     */
-    private final MeshFacet facet;   
-    /**
-     * Representation for areas of Voronoi region of triangles
-     */
-    private TriangleVertexAreas[] areas;
-    /**
-     * Represent min-max box. It is automatically maintained by given point array.
-     */
-    private BoundingBox boundingBox;
-    /**
-     * Helping array of triangles computed from corner table 
+     * Constructor.
+     * 
+     * @param f facet on which the symmetry will be computed 
+     * @param config configuration of optional parameters of the algorithm
      */
-    private List<MeshTriangle> triangles;
+    public SymmetryEstimator(MeshFacet f, Config config) {
+        this.facet = f;
+        this.config = config;
+        
+        this.triangles = f.asTriangles();
+        
+        this.areas = new TriangleVertexAreas[triangles.size()];
+        for (int i = 0; i < areas.length; i++) {
+            areas[i] = computeTriangleVertexAreas(triangles.get(i));
+        }
+
+        boundingBox = facet.getBoundingBox();
+    }
     
     /**
-     * panel for configuration of symmetry counting 
+     * Helper constructor.
+     * @param centroid Centroid
+     * @param a A
+     * @param b B
+     * @param scale Scale
      */
-    private JPanel panel;
+    protected SymmetryEstimator(Vector3d centroid, Vector3d a, Vector3d b, double scale) {
+        this(createMeshFacet(centroid, a, b, scale), Config.getDefault());
+    }
     
-    private final Config config;
-
     /**
      * 
      * @return configuration of optional parameters of the algorithm
@@ -66,19 +87,18 @@ public class SymmetryEstimator {
      * 
      * @return panel for configuration of symmetry counting 
      */
-    public JPanel getPanel() {
-        return panel;
-    }
+    //public JPanel getPanel() {
+    //    return panel;
+    //}
 
     /**
      * 
      * @param panel new panel for configuration of symmetry counting 
      */
-    public void setPanel(JPanel panel) {
-        this.panel = panel;
-    }
+    //public void setPanel(JPanel panel) {
+    //    this.panel = panel;
+    //}
     
-
     /**
      * 
      * @return Facet of the model on which symmetry is computed 
@@ -88,35 +108,267 @@ public class SymmetryEstimator {
     }
 
     /**
-     * Creates new class for computing symmetry
+     * If bounding box is not created yet, it creates new one. 
      * 
-     * @param f facet on which symmetry will be computed 
-     * @param config configuration of optional parameters of the algorithm
+     * @return Represent min-max box of the boundries of the model. 
      */
-    public SymmetryEstimator(MeshFacet f, Config config) {
-        this.facet = f;
-        this.config = config;
+    public BoundingBox getBoundingBox() {
+        if (boundingBox == null) {
+            boundingBox = facet.getBoundingBox();
+        }
+        return boundingBox; 
+    }
+    
+    /**
+     * Computes the approximate plane of symmetry.
+     * @param panel parrent window for the progress window (can be null)
+     * @return approximate plane of symmtetry
+     */
+    public Plane getApproxSymmetryPlane(JPanel panel) {
         
-        this.triangles = f.asTriangles();
+        ///UIManager.put("ProgressMonitor.progressText", "Counting symmetry...");
+ 
+        ArrayList<ApproxSymmetryPlane> planes = new ArrayList<>();
+        //List<Vector3d> normals = calculateNormals();
+        if (!facet.hasVertexNormals()) {
+            facet.calculateVertexNormals();
+        }
+        double[] curvatures = new double[facet.getNumberOfVertices()];
+        for (int i = 0; i < facet.getNumberOfVertices(); i++) {
+            if (facet.getNumberOfVertices() == 2500) {
+                curvatures[i] = this.getMaxCurvature(i);
+            } else {
+                curvatures[i] = this.getGaussianCurvature(i);
+            }
+            if (Double.isNaN(curvatures[i])){
+                curvatures[i] = Double.MIN_VALUE;
+            }
+        }
+        ArrayList<Double> sortedCurvatures = new ArrayList<>();
+        for (int i = 0; i < curvatures.length; i++) {
+            sortedCurvatures.add(curvatures[i]);
+        }
+        Collections.sort(sortedCurvatures);
+           
+        if(config.getSignificantPointCount() > facet.getNumberOfVertices()) {
+            config.setSignificantPointCount((facet.getNumberOfVertices()) - 1);
+        }
+        double bottomCurvature = sortedCurvatures.get(sortedCurvatures.size() - 1 - config.getSignificantPointCount());   
         
-        this.areas = new TriangleVertexAreas[triangles.size()];
-        for (int i = 0; i < areas.length; i++) {
-            areas[i] = computeTriangleVertexAreas(triangles.get(i));
+        ArrayList<Integer> significantPoints = new ArrayList<>();
+        ArrayList<Double> significantCurvatures = new ArrayList<>();
+        
+        for (int i = 0; i < facet.getNumberOfVertices(); i++) {
+            if (curvatures[i] >= bottomCurvature) {
+                significantCurvatures.add(curvatures[i]);
+                significantPoints.add(i);
+            }
+        }
+        
+        Plane plane = new Plane(0, 0, 0, 0);
+        int lastVotes = 0;
+        
+        ProgressMonitor progressMonitor = null;
+        if (panel != null) {
+            UIManager.put("ProgressMonitor.progressText", "Counting symmetry...");
+            progressMonitor = new ProgressMonitor(panel, "Counting...", "Steps", 0, significantPoints.size());
+        }
+        
+        double onePercent = significantCurvatures.size() / 100.0;
+        double percentsPerStep = 1 / onePercent;
+        for (int i = 0; i < significantPoints.size(); i++) {
+            for (int j = 0; j < significantPoints.size(); j++) {
+                if (i != j) {
+                    double minRatio = config.getMinCurvRatio();
+                    double maxRatio = 1.0 / minRatio;
+                    if (significantCurvatures.get(i) / significantCurvatures.get(j) >= minRatio && significantCurvatures.get(i) /
+                            significantCurvatures.get(j) <= maxRatio) {
+                        
+                        Vector3d p1 = new Vector3d(facet.getVertex(significantPoints.get(i)).getPosition());
+                        Vector3d p2 = new Vector3d(facet.getVertex(significantPoints.get(j)).getPosition());
+                        
+                        Vector3d avrg = new Vector3d(p1);
+                        avrg.add(p2);
+                        avrg.scale(0.5);
+                        
+                        Vector3d normal = new Vector3d(p1);
+                        normal.sub(p2);
+                        normal.normalize();
+                        
+                        double d = -(normal.x * avrg.x) - (normal.y * avrg.y) - (normal.z * avrg.z);
+                       
+                        Vector3d ni = new Vector3d(facet.getVertex(significantPoints.get(i)).getNormal());
+                        Vector3d nj = new Vector3d(facet.getVertex(significantPoints.get(j)).getNormal());
+                        ni.normalize();
+                        nj.normalize();
+
+                        Vector3d normVec = ni;
+                        normVec.sub(nj);
+                        normVec.normalize();
+                        double normCos = normVec.dot(normal);
+
+                        if (Math.abs(normCos) >= config.getMinNormAngleCos()) {
+                            Plane newPlane = new Plane(normal.x, normal.y, normal.z, d);
+                            int currentVotes = getVotes(newPlane, 
+                                significantCurvatures, 
+                                significantPoints, 
+                                config.getMinCurvRatio(), 
+                                config.getMinAngleCos(), 
+                                config.getMinNormAngleCos(), 
+                                boundingBox.getMaxDiag() * config.getMaxRelDistance());
+
+                            planes.add(new ApproxSymmetryPlane(newPlane, currentVotes));
+                            
+                            if (currentVotes > lastVotes) {
+                                lastVotes = currentVotes;
+                                plane = newPlane;
+                            }
+                        }
+                        
+                    }
+                }
+            }
+            if (panel != null) {
+                progressMonitor.setNote("Task step: " + (int) ((i + 1) * percentsPerStep));
+                progressMonitor.setProgress(i);
+            }
         }
+        
+        Collections.sort(planes);
+        ArrayList<ApproxSymmetryPlane> finalPlanes = new ArrayList<>();
+        for (int i = 0; i < planes.size(); i++) {
+            if (planes.get(i).getVotes() == lastVotes){
+                finalPlanes.add(planes.get(i));
+            }
+        }
+        //Plane finalPlane = new Plane(0, 0, 0, 0);
+        double newA = 0, newB = 0, newC = 0, newD = 0;
+        Vector3d refDir = finalPlanes.get(0).getNormal();
+        for (int i = 0; i < finalPlanes.size(); i++) {
+            Vector3d normDir = finalPlanes.get(i).getNormal();
+            if (normDir.dot(refDir) < 0) {
+                newA -= normDir.x;
+                newB -= normDir.y;
+                newC -= normDir.z;
+                newD -= finalPlanes.get(i).getDistance();
+            } else {
+                newA += normDir.x;
+                newB += normDir.y;
+                newC += normDir.z;
+                newD += finalPlanes.get(i).getDistance();
+            }
+        }
+        Plane finalPlane = new Plane(newA, newB, newC, newD);
+        finalPlane.normalize();
+        if (config.isAveraging()){
+            plane = finalPlane;
+        }
+        if (panel != null) {
+            JOptionPane.showMessageDialog(panel, "Symmetry estimate done.", "Done", 0,
+                   new ImageIcon(getClass().getResource("/cz/fidentis/analyst/gui/resources/exportedModel.png")));
 
-        boundingBox = facet.getBoundingBox();
+            progressMonitor.close();
+        }
+        return plane;
     }
     
     /**
-     * If bounding box is not created yet, it creates new one. 
      * 
-     * @return Represent min-max box of the boundries of the model. 
+     * @param plane Plane computed as symmetry plane
+     * @return mesh that represents facet with computed plane of approximate symmetry
      */
-    public BoundingBox getBoundingBox() {
-        if (boundingBox == null) {
-            boundingBox = facet.getBoundingBox();
+    public SymmetryEstimator mergeWithPlane(Plane plane) {
+        Vector3d normal = plane.getNormal();
+        Vector3d midPoint = boundingBox.getMidPoint().getPosition();
+
+        double alpha = -((normal.x * midPoint.x) + 
+                (normal.y * midPoint.y) + (normal.z * midPoint.z) +
+                plane.getDistance()) / (normal.dot(normal));
+        
+        Vector3d midPointOnPlane = new Vector3d(midPoint);
+        Vector3d nn = new Vector3d(normal);
+        nn.scale(alpha);
+        midPointOnPlane.add(nn);
+
+        double val = normal.x * midPointOnPlane.x + normal.y *
+                midPointOnPlane.y + normal.z *
+                midPointOnPlane.z + plane.getDistance();
+
+        Vector3d a = new Vector3d();
+        if (Math.abs(normal.dot(new Vector3d(0.0, 1.0, 0.0))) > Math.abs(normal.dot(new Vector3d (1.0, 0.0, 0.0)))) {
+            a.cross(normal, new Vector3d(1.0, 0.0, 0.0));
+        } else {
+            a.cross(normal, new Vector3d(0.0, 1.0, 0.0));
         }
-        return boundingBox; 
+        a.normalize();
+
+        Vector3d b = new Vector3d();
+        b.cross(normal,a);
+        b.normalize();
+        
+        SymmetryEstimator planeMesh = new SymmetryEstimator(midPointOnPlane, a, b,
+                (boundingBox.getMaxPoint().subtractPosition(boundingBox.getMinPoint())).getPosition().x);
+       
+        return mergeMeshWith(planeMesh);
+    }
+    
+    /**
+     * 
+     * @param s mesh that will be merged
+     * @return mesh with merged vertices from both meshes 
+     */
+    public SymmetryEstimator mergeMeshWith(SymmetryEstimator s) {
+        CornerTableRow row1 = new CornerTableRow(facet.getNumberOfVertices(), -1);
+        CornerTableRow row2 = new CornerTableRow(facet.getNumberOfVertices() + 1, facet.getNumberOfVertices() + 3);
+        CornerTableRow row3 = new CornerTableRow(facet.getNumberOfVertices() + 2, -1);
+        CornerTableRow row4 = new CornerTableRow(facet.getNumberOfVertices() + 2, -1);
+        CornerTableRow row5 = new CornerTableRow(facet.getNumberOfVertices() + 3, facet.getNumberOfVertices() + 1);
+        CornerTableRow row6 = new CornerTableRow(facet.getNumberOfVertices(), -1);
+        
+        facet.getCornerTable().addRow(row1);
+        facet.getCornerTable().addRow(row2);
+        facet.getCornerTable().addRow(row3);
+        facet.getCornerTable().addRow(row4);
+        facet.getCornerTable().addRow(row5);
+        facet.getCornerTable().addRow(row6);
+
+        for(int n = 0; n < 4; n++) {
+            facet.addVertex(s.getFacet().getVertices().get(n));
+        }        
+        return this;
+    }
+
+    private static MeshFacet createMeshFacet(Vector3d centroid, Vector3d a, Vector3d b, double scale) {
+        Vector3d[] points = new Vector3d[4];
+        
+        Vector3d aScaled = new Vector3d(a);
+        Vector3d bScaled = new Vector3d(b);
+        aScaled.scale(scale);
+        bScaled.scale(scale);
+        
+        points[0] = new Vector3d(centroid);
+        points[0].sub(aScaled);
+        points[0].sub(bScaled);
+        
+        points[1] = new Vector3d(centroid);
+        points[1].sub(aScaled);
+        points[1].add(bScaled);
+        
+        points[2] = new Vector3d(centroid);
+        points[2].add(aScaled);
+        points[2].add(bScaled);
+        
+        points[3] = new Vector3d(centroid);
+        points[3].add(aScaled);
+        points[3].sub(bScaled);
+        
+        MeshFacet facet = new MeshFacetImpl();
+        for (Vector3d point : points) {
+            facet.addVertex(new MeshPointImpl(point, null, null));
+        }
+        facet.calculateVertexNormals();
+        
+        return facet;
     }
 
     /**
@@ -125,10 +377,16 @@ public class SymmetryEstimator {
      * 
      * @author Natália Bebjaková
      */
-    private class TriangleVertexAreas {
-            public double v1Area;
-            public double v2Area;
-            public double v3Area;
+    private final class TriangleVertexAreas {
+            public final double v1Area;
+            public final double v2Area;
+            public final double v3Area;
+            
+            private TriangleVertexAreas(double v1, double v2, double v3) {
+                v1Area = v1;
+                v2Area = v2;
+                v3Area = v3;
+            }
     }
    
     /**
@@ -165,41 +423,41 @@ public class SymmetryEstimator {
         MeshPoint v3Half = (t.vertex2.addPosition(t.vertex1)).dividePosition(2);
 
         
-        TriangleVertexAreas area = new TriangleVertexAreas();
+        //TriangleVertexAreas area = new TriangleVertexAreas();
 
         if (d1 < 0) {           
-            area.v3Area = ((v2Half.subtractPosition(t.vertex3)).crossProduct(v1Half.subtractPosition(t.vertex3))).abs() / 2.0;
-            area.v2Area = ((v3Half.subtractPosition(t.vertex2)).crossProduct(v1Half.subtractPosition(t.vertex2))).abs() / 2.0;
-            area.v1Area = (((v1Half.subtractPosition(t.vertex1)).crossProduct(v3Half.subtractPosition(t.vertex1))).abs() / 2.0) + 
+            double v3Area = ((v2Half.subtractPosition(t.vertex3)).crossProduct(v1Half.subtractPosition(t.vertex3))).abs() / 2.0;
+            double v2Area = ((v3Half.subtractPosition(t.vertex2)).crossProduct(v1Half.subtractPosition(t.vertex2))).abs() / 2.0;
+            double v1Area = (((v1Half.subtractPosition(t.vertex1)).crossProduct(v3Half.subtractPosition(t.vertex1))).abs() / 2.0) + 
                     (((v1Half.subtractPosition(t.vertex1)).crossProduct(v2Half.subtractPosition(t.vertex1))).abs() / 2.0);
-            return area;
+            return new TriangleVertexAreas(v1Area, v2Area, v3Area);
         }
         if (d2 < 0) {
-            area.v1Area = ((v3Half.subtractPosition(t.vertex1)).crossProduct(v2Half.subtractPosition(t.vertex1))).abs() / 2.0;
-            area.v3Area = ((v1Half.subtractPosition(t.vertex3)).crossProduct(v2Half.subtractPosition(t.vertex3))).abs() / 2.0;
-            area.v2Area = (((v2Half.subtractPosition(t.vertex2)).crossProduct(v1Half.subtractPosition(t.vertex2))).abs() / 2.0) + 
+            double v1Area = ((v3Half.subtractPosition(t.vertex1)).crossProduct(v2Half.subtractPosition(t.vertex1))).abs() / 2.0;
+            double v3Area = ((v1Half.subtractPosition(t.vertex3)).crossProduct(v2Half.subtractPosition(t.vertex3))).abs() / 2.0;
+            double v2Area = (((v2Half.subtractPosition(t.vertex2)).crossProduct(v1Half.subtractPosition(t.vertex2))).abs() / 2.0) + 
                     (((v2Half.subtractPosition(t.vertex2)).crossProduct(v3Half.subtractPosition(t.vertex2))).abs() / 2.0);
-            return area;
+            return new TriangleVertexAreas(v1Area, v2Area, v3Area);
         }
         if (d3 < 0) {
-            area.v2Area = ((v1Half.subtractPosition(t.vertex2)).crossProduct(v3Half.subtractPosition(t.vertex2))).abs() / 2.0;
-            area.v1Area = ((v2Half.subtractPosition(t.vertex1)).crossProduct(v3Half.subtractPosition(t.vertex1))).abs() / 2.0;
-            area.v3Area = (((v3Half.subtractPosition(t.vertex3)).crossProduct(v2Half.subtractPosition(t.vertex3))).abs() / 2.0) + 
+            double v2Area = ((v1Half.subtractPosition(t.vertex2)).crossProduct(v3Half.subtractPosition(t.vertex2))).abs() / 2.0;
+            double v1Area = ((v2Half.subtractPosition(t.vertex1)).crossProduct(v3Half.subtractPosition(t.vertex1))).abs() / 2.0;
+            double v3Area = (((v3Half.subtractPosition(t.vertex3)).crossProduct(v2Half.subtractPosition(t.vertex3))).abs() / 2.0) + 
                     (((v3Half.subtractPosition(t.vertex3)).crossProduct(v1Half.subtractPosition(t.vertex3))).abs() / 2.0);
-            return area;
+            return new TriangleVertexAreas(v1Area, v2Area, v3Area);
         }
         
         MeshPoint circumcenter = t.vertex1.multiplyPosition(d1).addPosition(t.vertex2.multiplyPosition(d2).addPosition(t.vertex3.multiplyPosition(d3)));
         
-        area.v1Area = (((v2Half.subtractPosition(t.vertex1)).crossProduct(circumcenter.subtractPosition(t.vertex1))).abs() / 2.0) + 
+        double v1Area = (((v2Half.subtractPosition(t.vertex1)).crossProduct(circumcenter.subtractPosition(t.vertex1))).abs() / 2.0) + 
                 (((v3Half.subtractPosition(t.vertex1)).crossProduct(circumcenter.subtractPosition(t.vertex1))).abs() / 2.0);
         
-        area.v2Area = (((v3Half.subtractPosition(t.vertex2)).crossProduct(circumcenter.subtractPosition(t.vertex2))).abs() / 2.0) +
+        double v2Area = (((v3Half.subtractPosition(t.vertex2)).crossProduct(circumcenter.subtractPosition(t.vertex2))).abs() / 2.0) +
                 (((v1Half.subtractPosition(t.vertex2)).crossProduct(circumcenter.subtractPosition(t.vertex2))).abs() / 2.0);
         
-        area.v3Area = (((v1Half.subtractPosition(t.vertex3)).crossProduct(circumcenter.subtractPosition(t.vertex3))).abs() / 2.0) +
+        double v3Area = (((v1Half.subtractPosition(t.vertex3)).crossProduct(circumcenter.subtractPosition(t.vertex3))).abs() / 2.0) +
                 (((v2Half.subtractPosition(t.vertex3)).crossProduct(circumcenter.subtractPosition(t.vertex3))).abs() / 2.0);
-        return area;
+        return new TriangleVertexAreas(v1Area, v2Area, v3Area);
     }
     
     /**
@@ -428,11 +686,10 @@ public class SymmetryEstimator {
             double minAngleCos,
             double minNormAngleCos,
             double maxDist) {
-        
         plane.normalize();
         
-        Vector3d normal = new Vector3d(plane.a, plane.b, plane.c);
-        double d = plane.d;
+        Vector3d normal = plane.getNormal();
+        double d = plane.getDistance();
         double maxCurvRatio = 1.0 / minCurvRatio;
         int votes = 0;
         //List<Vector3d> normals = calculateNormals();
@@ -481,214 +738,4 @@ public class SymmetryEstimator {
         return votes;
     }
     
-    /**
-     * 
-     * 
-     * @return approximate plane of symmtetry
-     */
-    public Plane getAproxSymmetryPlane() {
-        
-        UIManager.put("ProgressMonitor.progressText", "Counting symmetry...");
- 
-        ArrayList<AproxSymmetryPlane> planes = new ArrayList<>();
-        //List<Vector3d> normals = calculateNormals();
-        if (!facet.hasVertexNormals()) {
-            facet.calculateVertexNormals();
-        }
-        double[] curvatures = new double[facet.getNumberOfVertices()];
-        for (int i = 0; i < facet.getNumberOfVertices(); i++) {
-            if (facet.getNumberOfVertices() == 2500) {
-                curvatures[i] = this.getMaxCurvature(i);
-            } else {
-                curvatures[i] = this.getGaussianCurvature(i);
-            }
-            if (Double.isNaN(curvatures[i])){
-                curvatures[i] = Double.MIN_VALUE;
-            }
-        }
-        ArrayList<Double> sortedCurvatures = new ArrayList<>();
-        for (int i = 0; i < curvatures.length; i++) {
-            sortedCurvatures.add(curvatures[i]);
-        }
-        Collections.sort(sortedCurvatures);
-           
-        if(config.getSignificantPointCount() > facet.getNumberOfVertices()) {
-            config.setSignificantPointCount((facet.getNumberOfVertices()) - 1);
-        }
-        double bottomCurvature = sortedCurvatures.get(sortedCurvatures.size() - 1 - config.getSignificantPointCount());   
-        
-        ArrayList<Integer> significantPoints = new ArrayList<>();
-        ArrayList<Double> significantCurvatures = new ArrayList<>();
-        
-        for (int i = 0; i < facet.getNumberOfVertices(); i++) {
-            if (curvatures[i] >= bottomCurvature) {
-                significantCurvatures.add(curvatures[i]);
-                significantPoints.add(i);
-            }
-        }
-        
-        Plane plane = new Plane(0, 0, 0, 0);
-        int lastVotes = 0;
-        
-        ProgressMonitor progressMonitor;
-        progressMonitor = new ProgressMonitor(panel, "Cunting...",
-                "Steps", 0, significantPoints.size());
-        
-        double onePercent = significantCurvatures.size() / 100.0;
-        double percentsPerStep = 1 / onePercent;
-        for (int i = 0; i < significantPoints.size(); i++) {
-            for (int j = 0; j < significantPoints.size(); j++) {
-                if (i != j) {
-                    double minRatio = config.getMinCurvRatio();
-                    double maxRatio = 1.0 / minRatio;
-                    if (significantCurvatures.get(i) / significantCurvatures.get(j) >= minRatio && significantCurvatures.get(i) /
-                            significantCurvatures.get(j) <= maxRatio) {
-                        
-                        Vector3d p1 = new Vector3d(facet.getVertex(significantPoints.get(i)).getPosition());
-                        Vector3d p2 = new Vector3d(facet.getVertex(significantPoints.get(j)).getPosition());
-                        
-                        Vector3d avrg = new Vector3d(p1);
-                        avrg.add(p2);
-                        avrg.scale(0.5);
-                        
-                        Vector3d normal = new Vector3d(p1);
-                        normal.sub(p2);
-                        normal.normalize();
-                        
-                        double d = -(normal.x * avrg.x) - (normal.y * avrg.y) - (normal.z * avrg.z);
-                       
-                        Vector3d ni = new Vector3d(facet.getVertex(significantPoints.get(i)).getNormal());
-                        Vector3d nj = new Vector3d(facet.getVertex(significantPoints.get(j)).getNormal());
-                        ni.normalize();
-                        nj.normalize();
-
-                        Vector3d normVec = ni;
-                        normVec.sub(nj);
-                        normVec.normalize();
-                        double normCos = normVec.dot(normal);
-
-                        if (Math.abs(normCos) >= config.getMinNormAngleCos()) {
-                            Plane newPlane = new Plane(normal.x, normal.y, normal.z, d);
-                            int currentVotes = getVotes(newPlane, 
-                                significantCurvatures, 
-                                significantPoints, 
-                                config.getMinCurvRatio(), 
-                                config.getMinAngleCos(), 
-                                config.getMinNormAngleCos(), 
-                                boundingBox.getMaxDiag() * config.getMaxRelDistance());
-
-                            planes.add(new AproxSymmetryPlane(newPlane, currentVotes));
-                            
-                            if (currentVotes > lastVotes) {
-                                lastVotes = currentVotes;
-                                plane = newPlane;
-                            }
-                        }
-                        
-                    }
-                }
-            }
-            progressMonitor.setNote("Task step: " + (int) ((i + 1) * percentsPerStep));
-            progressMonitor.setProgress(i);
-        }
-        
-        Collections.sort(planes);
-        ArrayList<AproxSymmetryPlane> finalPlanes = new ArrayList<>();
-        for (int i = 0; i < planes.size(); i++) {
-            if (planes.get(i).votes == lastVotes){
-                finalPlanes.add(planes.get(i));
-            }
-        }
-        Plane finalPlane = new Plane(0, 0, 0, 0);
-        Vector3d refDir = new Vector3d(finalPlanes.get(0).a, finalPlanes.get(0).b, finalPlanes.get(0).c);
-        for (int i = 0; i < finalPlanes.size(); i++) {
-            Vector3d normDir = new Vector3d(finalPlanes.get(i).a, finalPlanes.get(i).b, finalPlanes.get(i).c);
-            if (normDir.dot(refDir) < 0) {
-                finalPlane.a -= finalPlanes.get(i).a;
-                finalPlane.b -= finalPlanes.get(i).b;
-                finalPlane.c -= finalPlanes.get(i).c;
-                finalPlane.d -= finalPlanes.get(i).d;
-            } else {
-                finalPlane.a += finalPlanes.get(i).a;
-                finalPlane.b += finalPlanes.get(i).b;
-                finalPlane.c += finalPlanes.get(i).c;
-                finalPlane.d += finalPlanes.get(i).d;
-            }
-        }
-        finalPlane.normalize();
-        if (config.isAveraging()){
-            plane = finalPlane;
-        }
-        JOptionPane.showMessageDialog(panel, "Symmetry estimate done.", "Done", 0,
-                new ImageIcon(getClass().getResource("/cz/fidentis/analyst/gui/resources/exportedModel.png")));
-
-        progressMonitor.close();
-        return plane;
-    }
-    
-    /**
-     * 
-     * @param plane Plane computed as symmetry plane
-     * @return mesh that represents facet with computed plane of approximate symmetry
-     */
-    public SymmetryEstimator mergeWithPlane(Plane plane) {
-        Vector3d normal = new Vector3d(plane.a, plane.b, plane.c);
-        Vector3d midPoint = boundingBox.getMidPoint().getPosition();
-
-        double alpha = -((plane.a * midPoint.x) + 
-                (plane.b * midPoint.y) + (plane.c * midPoint.z) +
-                plane.d) / (normal.dot(normal));
-        
-        Vector3d midPointOnPlane = new Vector3d(midPoint);
-        Vector3d nn = new Vector3d(normal);
-        nn.scale(alpha);
-        midPointOnPlane.add(nn);
-
-        double val = plane.a * midPointOnPlane.x + plane.b *
-                midPointOnPlane.y + plane.c *
-                midPointOnPlane.z + plane.d;
-
-        Vector3d a = new Vector3d();
-        if (Math.abs(normal.dot(new Vector3d(0.0, 1.0, 0.0))) > Math.abs(normal.dot(new Vector3d (1.0, 0.0, 0.0)))) {
-            a.cross(normal, new Vector3d(1.0, 0.0, 0.0));
-        } else {
-            a.cross(normal, new Vector3d(0.0, 1.0, 0.0));
-        }
-        a.normalize();
-
-        Vector3d b = new Vector3d();
-        b.cross(normal,a);
-        b.normalize();
-       
-        SymmetryEstimator planeMesh = Plane.createPlaneMesh(midPointOnPlane, a, b,
-                (boundingBox.getMaxPoint().subtractPosition(boundingBox.getMinPoint())).getPosition().x);
-
-        return mergeMeshWith(planeMesh);
-    }
-    
-    /**
-     * 
-     * @param s mesh that will be merged
-     * @return mesh with merged vertices from both meshes 
-     */
-    public SymmetryEstimator mergeMeshWith(SymmetryEstimator s) {
-        CornerTableRow row1 = new CornerTableRow(facet.getNumberOfVertices(), -1);
-        CornerTableRow row2 = new CornerTableRow(facet.getNumberOfVertices() + 1, facet.getNumberOfVertices() + 3);
-        CornerTableRow row3 = new CornerTableRow(facet.getNumberOfVertices() + 2, -1);
-        CornerTableRow row4 = new CornerTableRow(facet.getNumberOfVertices() + 2, -1);
-        CornerTableRow row5 = new CornerTableRow(facet.getNumberOfVertices() + 3, facet.getNumberOfVertices() + 1);
-        CornerTableRow row6 = new CornerTableRow(facet.getNumberOfVertices(), -1);
-        
-        facet.getCornerTable().addRow(row1);
-        facet.getCornerTable().addRow(row2);
-        facet.getCornerTable().addRow(row3);
-        facet.getCornerTable().addRow(row4);
-        facet.getCornerTable().addRow(row5);
-        facet.getCornerTable().addRow(row6);
-
-        for(int n = 0; n < 4; n++) {
-            facet.addVertex(s.getFacet().getVertices().get(n));
-        }        
-        return this;
-    }
 }
diff --git a/GUI/src/main/java/cz/fidentis/analyst/gui/SymmetryPanel.form b/GUI/src/main/java/cz/fidentis/analyst/gui/SymmetryPanel.form
index 65d045392839a2d7c63dd98b9731e7518c2821a6..3af48b80f709f79399b8138ff9d0553279069aea 100644
--- a/GUI/src/main/java/cz/fidentis/analyst/gui/SymmetryPanel.form
+++ b/GUI/src/main/java/cz/fidentis/analyst/gui/SymmetryPanel.form
@@ -77,7 +77,7 @@
                                   </Group>
                                   <EmptySpace max="-2" attributes="0"/>
                                   <Group type="103" groupAlignment="0" attributes="0">
-                                      <Component id="minCurvatio4" pref="157" max="32767" attributes="0"/>
+                                      <Component id="minCurvatio4" pref="0" max="32767" attributes="0"/>
                                       <Group type="102" alignment="0" attributes="0">
                                           <Group type="103" groupAlignment="0" attributes="0">
                                               <Component id="minCurvatio" min="-2" max="-2" attributes="0"/>
@@ -208,7 +208,6 @@
             <Property name="cursor" type="java.awt.Cursor" editor="org.netbeans.modules.form.editors2.CursorEditor">
               <Color id="Hand Cursor"/>
             </Property>
-            <Property name="opaque" type="boolean" value="false"/>
           </Properties>
         </Component>
         <Component class="javax.swing.JSlider" name="angleCosineSlider">
@@ -218,7 +217,6 @@
             <Property name="cursor" type="java.awt.Cursor" editor="org.netbeans.modules.form.editors2.CursorEditor">
               <Color id="Hand Cursor"/>
             </Property>
-            <Property name="opaque" type="boolean" value="false"/>
           </Properties>
         </Component>
         <Component class="javax.swing.JLabel" name="minCurvatio">
@@ -250,7 +248,6 @@
             <Property name="cursor" type="java.awt.Cursor" editor="org.netbeans.modules.form.editors2.CursorEditor">
               <Color id="Hand Cursor"/>
             </Property>
-            <Property name="opaque" type="boolean" value="false"/>
           </Properties>
         </Component>
         <Component class="javax.swing.JLabel" name="minCurvatio3">
@@ -272,7 +269,6 @@
             <Property name="cursor" type="java.awt.Cursor" editor="org.netbeans.modules.form.editors2.CursorEditor">
               <Color id="Hand Cursor"/>
             </Property>
-            <Property name="opaque" type="boolean" value="false"/>
           </Properties>
         </Component>
         <Component class="javax.swing.JLabel" name="minCurvatio4">
@@ -293,7 +289,6 @@
             <Property name="cursor" type="java.awt.Cursor" editor="org.netbeans.modules.form.editors2.CursorEditor">
               <Color id="Hand Cursor"/>
             </Property>
-            <Property name="opaque" type="boolean" value="false"/>
           </Properties>
         </Component>
         <Component class="javax.swing.JLabel" name="significantPointLabel">
@@ -325,7 +320,6 @@
         <Component class="javax.swing.JCheckBox" name="averagingCheckBox">
           <Properties>
             <Property name="selected" type="boolean" value="true"/>
-            <Property name="opaque" type="boolean" value="false"/>
           </Properties>
           <Events>
             <EventHandler event="mouseClicked" listener="java.awt.event.MouseListener" parameters="java.awt.event.MouseEvent" handler="averagingCheckBoxMouseClicked"/>
diff --git a/GUI/src/main/java/cz/fidentis/analyst/gui/SymmetryPanel.java b/GUI/src/main/java/cz/fidentis/analyst/gui/SymmetryPanel.java
index 2a9889fe2734364490585a4744783e2dcece20bf..cbb2d2bdac7011a02bb78f0344d100bafdd64574 100644
--- a/GUI/src/main/java/cz/fidentis/analyst/gui/SymmetryPanel.java
+++ b/GUI/src/main/java/cz/fidentis/analyst/gui/SymmetryPanel.java
@@ -11,6 +11,8 @@ import javax.swing.ImageIcon;
 import javax.swing.JOptionPane;
 import javax.swing.JSlider;
 import javax.swing.JTextField;
+import javax.swing.ProgressMonitor;
+import javax.swing.UIManager;
 import javax.swing.event.ChangeEvent;
 
 /**
@@ -155,8 +157,7 @@ public final class SymmetryPanel extends javax.swing.JPanel {
         MeshModel model = new MeshModel();
         canvas.changeModel(canvas.getLoadedModel());
         symCounter = new SymmetryEstimator(canvas.getModel().getFacets().get(0), config);
-        symCounter.setPanel(this);
-        finalPlane = symCounter.getAproxSymmetryPlane();
+        finalPlane = symCounter.getApproxSymmetryPlane(this);
         SymmetryEstimator counted = symCounter.mergeWithPlane(finalPlane);
         model.addFacet(counted.getFacet());
         
@@ -525,8 +526,8 @@ public final class SymmetryPanel extends javax.swing.JPanel {
      * @param evt Final computed plane is shown to user
      */
     private void showPlaneLabelMouseClicked(java.awt.event.MouseEvent evt) {//GEN-FIRST:event_showPlaneLabelMouseClicked
-        JOptionPane.showMessageDialog(frameMain, "Approximate plane of symmetry:  \n" + finalPlane.a + "\n" + finalPlane.b + "\n" + finalPlane.c + "\n" +
-            finalPlane.d + "\n", "Final plane.", 0, new ImageIcon(getClass().getResource("/cz/fidentis/analyst/gui/resources/showPlanePane.png")));
+        JOptionPane.showMessageDialog(frameMain, "Approximate plane of symmetry:  \n" + finalPlane.getNormal().x + "\n" + finalPlane.getNormal().y + "\n" + finalPlane.getNormal().z + "\n" +
+            finalPlane.getDistance() + "\n", "Final plane.", 0, new ImageIcon(getClass().getResource("/cz/fidentis/analyst/gui/resources/showPlanePane.png")));
     }//GEN-LAST:event_showPlaneLabelMouseClicked
 
     /**