diff --git a/Comparison/pom.xml b/Comparison/pom.xml
index 60d41a8f0e4066ef57cdf60b7f549cf2028f231c..9406c23bb387aab061d271a1b2223f0489cf4073 100644
--- a/Comparison/pom.xml
+++ b/Comparison/pom.xml
@@ -59,6 +59,14 @@
                     <target>8</target>
                 </configuration>
             </plugin>
+            <!-- <plugin>
+                <groupId>org.openjfx</groupId>
+                <artifactId>javafx-maven-plugin</artifactId>
+                <version>0.0.4</version>
+                <configuration>
+                    <mainClass>org.openjfx.App</mainClass>
+                </configuration>
+            </plugin> -->
         </plugins>
     </build>
     <dependencies>
@@ -77,6 +85,17 @@
             <artifactId>vecmath</artifactId>
             <version>${version.javax.vecmath}</version>
         </dependency>
+        <!-- <dependency>
+            <groupId>org.openjfx</groupId>
+            <artifactId>javafx-controls</artifactId>
+            <version>11.0.2</version>
+        </dependency> -->
+        <dependency>
+            <groupId>org.junit.jupiter</groupId>
+            <artifactId>junit-jupiter</artifactId>
+            <version>5.6.0</version>
+            <scope>test</scope>
+        </dependency>
     </dependencies>
     <properties>
         <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
diff --git a/Comparison/src/main/java/cz/fidentis/analyst/comparison/ClosestVertices.java b/Comparison/src/main/java/cz/fidentis/analyst/comparison/ClosestVertices.java
new file mode 100644
index 0000000000000000000000000000000000000000..7c9cc087308ed3fa95d3ed7c16fc943a735b5b0d
--- /dev/null
+++ b/Comparison/src/main/java/cz/fidentis/analyst/comparison/ClosestVertices.java
@@ -0,0 +1,39 @@
+package cz.fidentis.analyst.comparison;
+
+import cz.fidentis.analyst.mesh.core.MeshPoint;
+
+/**
+ * 
+ * @author Matej Lukes
+ */
+public class ClosestVertices {
+    
+    private MeshPoint firstVertex;
+    private MeshPoint secondVertex;
+    private double distance;
+
+    /**
+     * Constructor.
+     * 
+     * @param firstVertex Firt vertex of the pair
+     * @param secondVertex Second vertex of the pair
+     * @param distance Distance
+     */
+    public ClosestVertices(MeshPoint firstVertex, MeshPoint secondVertex, double distance) {
+        this.firstVertex = firstVertex;
+        this.secondVertex = secondVertex;
+        this.distance = distance;
+    }
+
+    public MeshPoint getFirstVertex() {
+        return firstVertex;
+    }
+
+    public MeshPoint getSecondVertex() {
+        return secondVertex;
+    }
+
+    public double getDistance() {
+        return distance;
+    }
+}
diff --git a/Comparison/src/main/java/cz/fidentis/analyst/comparison/Comparison.java b/Comparison/src/main/java/cz/fidentis/analyst/comparison/Comparison.java
new file mode 100644
index 0000000000000000000000000000000000000000..a7ce1c307db8f155356eb51bcae6efbae9070a13
--- /dev/null
+++ b/Comparison/src/main/java/cz/fidentis/analyst/comparison/Comparison.java
@@ -0,0 +1,95 @@
+package cz.fidentis.analyst.comparison;
+
+import cz.fidentis.analyst.mesh.core.MeshFacet;
+import cz.fidentis.analyst.mesh.io.MeshObjLoader;
+
+import java.io.File;
+import java.io.IOException;
+import java.util.List;
+import java.util.concurrent.CompletableFuture;
+
+/**
+ * @author Matej Lukes
+ */
+public class Comparison {
+    private HausdorffDistance hausdorffDistance;
+
+    private MeshFacet mainFacet;
+    private MeshFacet comparedFacet;
+
+    /**
+     * Asynchronously loads main meshModel
+     *
+     * @param path path to meshModel file
+     * @return CompletableFuture
+     */
+    public CompletableFuture loadMainModel(String path) {
+
+        return CompletableFuture.runAsync(() -> {
+            try {
+                mainFacet = MeshObjLoader.read(new File(path)).getFacets().get(1);
+            } catch (IOException e) {
+                e.printStackTrace();
+            }
+        });
+    }
+
+    /**
+     * Asynchronously loads compared meshModel
+     *
+     * @param path path to meshModel file
+     * @return CompletableFuture
+     */
+    public CompletableFuture loadComparedModel(String path) {
+        return CompletableFuture.runAsync(() -> {
+            try {
+                comparedFacet = MeshObjLoader.read(new File(path)).getFacets().get(1);
+            } catch (IOException e) {
+                e.printStackTrace();
+            }
+        });
+    }
+
+    /**
+     * Asynchronously registers compared meshFacet to main meshFacet
+     *
+     * @param method registration method
+     * @return CompletableFuture
+     */
+    public CompletableFuture register(RegistrationMethod method) {
+        return CompletableFuture.runAsync(() -> comparedFacet = Registration
+                .register(mainFacet, comparedFacet, method));
+    }
+
+    /**
+     * Asynchronously compares MeshFacets from vertices to vertices
+     *
+     * @return list containing vertex from first facet, closest vertex to it from second facet, distance
+     */
+    public CompletableFuture<List<ClosestVertices>> compareHausdorffDistanceToVertices() {
+        hausdorffDistance = new HausdorffDistance(mainFacet, comparedFacet);
+        return CompletableFuture.supplyAsync(() -> hausdorffDistance.calculateHausdorffDistanceToVertices());
+    }
+
+    /**
+     * Asynchronously compares MeshFacets from vertices to any point on mesh
+     *
+     * @return list containing vertex from first facet, closest point to it from second facet, distance
+     */
+    public CompletableFuture<List<ClosestVertices>> compareHausdorffDistanceToMesh() {
+        hausdorffDistance = new HausdorffDistance(mainFacet, comparedFacet);
+        return CompletableFuture.supplyAsync(() -> hausdorffDistance.calculateHausdorffDistanceToMesh());
+    }
+
+    /**
+     * returns progress percentage
+     *
+     * @return progress percentage
+     */
+    public double getComparisonProgress() {
+        if (hausdorffDistance == null) {
+            return -1;
+        }
+        return hausdorffDistance.getProgressPercentage();
+    }
+}
diff --git a/Comparison/src/main/java/cz/fidentis/analyst/comparison/HausdorffDistance.java b/Comparison/src/main/java/cz/fidentis/analyst/comparison/HausdorffDistance.java
new file mode 100644
index 0000000000000000000000000000000000000000..d3226f5d0d9e2108bbc0e2f7158eb4ff7f84f9af
--- /dev/null
+++ b/Comparison/src/main/java/cz/fidentis/analyst/comparison/HausdorffDistance.java
@@ -0,0 +1,355 @@
+package cz.fidentis.analyst.comparison;
+
+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.MeshTriangle;
+
+import javax.vecmath.Vector3d;
+import java.util.ArrayList;
+import java.util.Comparator;
+import java.util.List;
+import java.util.Optional;
+import java.util.concurrent.ExecutionException;
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.Executors;
+import java.util.concurrent.Future;
+import java.util.concurrent.atomic.AtomicInteger;
+import java.util.stream.Collectors;
+
+/**
+ * @author Matej Lukes
+ */
+public class HausdorffDistance {
+    
+    private MeshFacet mainFacet;
+    private MeshFacet comparedFacet;
+
+    private AtomicInteger progress = new AtomicInteger();
+    private int numberOfVertices;
+
+    /**
+     * @param mainFacet     main MeshFacet
+     * @param comparedFacet compared MeshFacet
+     */
+    public HausdorffDistance(MeshFacet mainFacet, MeshFacet comparedFacet) {
+        this.mainFacet = mainFacet;
+        this.comparedFacet = comparedFacet;
+        this.numberOfVertices = mainFacet.getNumberOfVertices();
+    }
+
+    /**
+     * returns progress percentage
+     *
+     * @return progress
+     */
+    public double getProgressPercentage() {
+        return ((double) progress.get() / numberOfVertices) * 100;
+    }
+
+    /**
+     * Finds the nearest vertex on the second facet.
+     *
+     * @param vertex vertex from
+     * @return vertex, nearest vertex from second facet, distance
+     */
+    private ClosestVertices getNearestVertex(MeshPoint vertex) {
+        Optional<Pair<MeshPoint, Double>> closestVertexAndDistance = comparedFacet.getVertices().parallelStream()
+                .map((meshPoint) -> new Pair<>(meshPoint, getDistanceBetweenPoints(vertex, meshPoint.getPosition())))
+                .max((Comparator.comparingDouble(Pair::getValue)));
+        return closestVertexAndDistance.map(vector3dDoublePair -> new ClosestVertices(vertex,
+                vector3dDoublePair.getKey(),
+                vector3dDoublePair.getValue())).orElse(null);
+    }
+
+    /**
+     * returns distance between two points
+     *
+     * @param point1 first point
+     * @param point2 second point
+     * @return distance
+     */
+    private double getDistanceBetweenPoints(MeshPoint point1, Vector3d point2) {
+        Vector3d helperVector = new Vector3d();
+        helperVector.sub(point1.getPosition(), point2);
+        return Math.signum(helperVector.dot(point1.getNormal())) * helperVector.length();
+    }
+
+    /**
+     * calculates Hausdorff Distance to the nearest vertex from second facet for each vertex in first facet
+     * this implementation uses executor
+     *
+     * @return list containing vertex from first facet, closest vertex to it from second facet, distance
+     */
+    public List<ClosestVertices> calculateHausdorffDistanceToVertices() {
+        progress.set(0);
+        int numberOfVertices = mainFacet.getNumberOfVertices();
+        List<Future<ClosestVertices>> closestVerticesFutures = new ArrayList<>(numberOfVertices);
+        ExecutorService executor = Executors.newCachedThreadPool();
+
+        for (final MeshPoint vertex : mainFacet.getVertices()) {
+            closestVerticesFutures.add(executor.submit(() -> {
+                ClosestVertices result = getNearestVertex(vertex);
+                progress.addAndGet(1);
+                return result;
+            }));
+        }
+
+        List<ClosestVertices> closestVertices = new ArrayList<>(numberOfVertices);
+        for (Future<ClosestVertices> future :
+                closestVerticesFutures) {
+            executor.submit(() -> {
+                try {
+                    ClosestVertices result = future.get();
+                    synchronized (closestVertices) {
+                        closestVertices.add(result);
+                    }
+                } catch (InterruptedException | ExecutionException e) {
+                    e.printStackTrace();
+                }
+            });
+        }
+
+        executor.shutdown();
+        return closestVertices;
+    }
+
+    /**
+     * calculates Hausdorff Distance to the nearest vertex from second facet for each vertex in first facet
+     * this implementation uses parallel streams
+     *
+     * @return list containing vertex from first facet, closest vertex to it from second facet, distance
+     */
+    public List<ClosestVertices> calculateHausdorffDistanceToVertices2() {
+        progress.set(0);
+        return mainFacet.getVertices().parallelStream()
+                .map((vertex) -> {
+                    ClosestVertices result = getNearestVertex(vertex);
+                    progress.addAndGet(1);
+                    return result;
+                })
+                .collect(Collectors.toList());
+    }
+
+    /**
+     * calculates Hausdorff Distance to the nearest point on second facet for each vertex
+     * this implementation uses executor
+     *
+     * @return list containing vertex from first facet, closest point to it from second facet, distance
+     */
+    public List<ClosestVertices> calculateHausdorffDistanceToMesh() {
+        progress.set(0);
+        int numberOfVertices = mainFacet.getNumberOfVertices();
+        List<Future<ClosestVertices>> closestPointsFutures = new ArrayList<>(numberOfVertices);
+        ExecutorService executor = Executors.newCachedThreadPool();
+
+        for (final MeshPoint vertex : mainFacet.getVertices()) {
+            closestPointsFutures.add(executor.submit(() -> {
+                ClosestVertices result = calculateNearestPointOnMesh(vertex,
+                        comparedFacet.getCornerTable()
+                                .getTriangleIndexesByVertexIndex(comparedFacet.getVertices()
+                                        .indexOf(getNearestVertex(vertex)
+                                                .getSecondVertex())));
+                progress.addAndGet(1);
+                return result;
+            }));
+        }
+
+        List<ClosestVertices> closestVertices = new ArrayList<>(numberOfVertices);
+        for (Future<ClosestVertices> future : closestPointsFutures) {
+            executor.submit(() -> {
+                try {
+                    ClosestVertices result = future.get();
+                    synchronized (closestVertices) {
+                        closestVertices.add(result);
+                    }
+                } catch (InterruptedException | ExecutionException e) {
+                    e.printStackTrace();
+                }
+            });
+        }
+
+        executor.shutdown();
+        return closestVertices;
+    }
+
+    /**
+     * calculates Hausdorff Distance to the nearest point on second facet for each vertex
+     * this implementation uses parallel streams
+     *
+     * @return list containing vertex from first facet, closest point to it from second facet, distance
+     */
+    public List<ClosestVertices> calculateHausdorffDistanceToMesh2() {
+        progress.set(0);
+        return mainFacet.getVertices().parallelStream()
+                .map((meshPoint) -> {
+                    ClosestVertices result = calculateNearestPointOnMesh(meshPoint,
+                            comparedFacet.getCornerTable()
+                                    .getTriangleIndexesByVertexIndex(comparedFacet.getVertices()
+                                            .indexOf(getNearestVertex(meshPoint)
+                                                    .getSecondVertex())));
+                    progress.addAndGet(1);
+                    return result;
+                }).collect(Collectors.toList());
+    }
+
+    /**
+     * calculates Hausdorff Distance to the nearest point on second facet for vertex
+     *
+     * @param vertex                     vertex from first facet
+     * @param indicesOfTrianglesOfVertex indices of triangles that contain the nearest vertex on second mesh
+     * @return vertex from first facet, closest point to it from second facet, distance
+     */
+    private ClosestVertices calculateNearestPointOnMesh(MeshPoint vertex, List<Integer> indicesOfTrianglesOfVertex) {
+        Vector3d vertexPosition = vertex.getPosition();
+        List<Pair<Vector3d, Double>> projections = new ArrayList<>(indicesOfTrianglesOfVertex.size());
+        Vector3d helperVector = new Vector3d();
+
+        List<MeshTriangle> trList = comparedFacet.asTriangles();
+        for (int index : indicesOfTrianglesOfVertex) {
+            List<Vector3d> triangle = new ArrayList<>();
+            triangle.add(trList.get(index).vertex1.getPosition());
+            triangle.add(trList.get(index).vertex2.getPosition());
+            triangle.add(trList.get(index).vertex3.getPosition());
+            //List<Vector3d> triangle = comparedFacet.asTriangles()..getVerticesOfTriangle(index).stream()
+            //        .map(MeshPoint::getPosition)
+            //        .collect(Collectors.toList());
+            Vector3d projection = getProjectionToTrianglePlane(vertexPosition, triangle);
+            if (isPointInTriangle(projection, triangle)) {
+                helperVector.sub(vertexPosition, projection);
+                projections.add(new Pair<>(projection, helperVector.length()));
+            } else {
+                projection = getProjectionToClosestEdge(projection, triangle);
+                helperVector.sub(vertexPosition, projection);
+                projections.add(new Pair<>(projection, helperVector.length()));
+            }
+        }
+
+        Pair<Vector3d, Double> closestPosition = projections.stream()
+                .min(Comparator.comparingDouble(Pair::getValue)).orElseGet(() -> new Pair<>(null, Double.MAX_VALUE));
+        return new ClosestVertices(vertex,
+                new MeshPointImpl(closestPosition.getKey(), null, null),
+                closestPosition.getValue());
+    }
+
+    /**
+     * returns perpendicular projection from vertex to plane of triangle
+     *
+     * @param vertex   vertex from which the projection is created
+     * @param triangle triangle that defines the plane
+     * @return projection to plane of triangle
+     */
+    private Vector3d getProjectionToTrianglePlane(Vector3d vertex, List<Vector3d> triangle) {
+        Vector3d ab = new Vector3d();
+        ab.sub(triangle.get(0), triangle.get(1));
+        Vector3d ac = new Vector3d();
+        ac.sub(triangle.get(0), triangle.get(2));
+        Vector3d normal = new Vector3d();
+        normal.cross(ab, ac);
+        normal.normalize();
+
+        Vector3d helperVector = new Vector3d(vertex);
+        helperVector.sub(triangle.get(0));
+        double distance = helperVector.dot(normal);
+        helperVector.scaleAdd(-distance, normal, helperVector);
+        return helperVector;
+    }
+
+    /**
+     * checks if a point in plane of triangle lies within the triangle
+     *
+     * @param point    checked point
+     * @param triangle triangle
+     * @return true if point is in triangle, false otherwise
+     */
+    private boolean isPointInTriangle(Vector3d point, List<Vector3d> triangle) {
+        List<Vector3d> pointToVertices = triangle.stream()
+                .map((vertex) -> {
+                    Vector3d v = new Vector3d(vertex);
+                    v.sub(point);
+                    return v;
+                }).collect(Collectors.toList());
+
+        double angleSum = 0;
+        for (int i = 0; i < 3; i++) {
+            angleSum += pointToVertices.get(i).angle(pointToVertices.get((i + 1) % 3));
+        }
+        angleSum -= Math.PI;
+        return -0.01 < angleSum && angleSum < 0.01;
+    }
+
+    /**
+     * returns projection to to the nearest edge of triangle
+     *
+     * @param point    point in plane of triangle
+     * @param triangle triangle
+     * @return perpendicular projection to the nearest edge
+     */
+    private Vector3d getProjectionToClosestEdge(Vector3d point, List<Vector3d> triangle) {
+        Vector3d[] projections = new Vector3d[3];
+        for (int i = 0; i < 3; i++) {
+            projections[i] = getProjectionToEdge(point, triangle.get(i), triangle.get((i + 1) % 3));
+        }
+
+        double minDistance = Double.MAX_VALUE;
+        Vector3d closestProjection = null;
+        Vector3d helperVector = new Vector3d();
+        for (Vector3d projection :
+                projections) {
+            helperVector.sub(point, projection);
+            double distance = helperVector.length();
+            if (distance < minDistance) {
+                minDistance = distance;
+                closestProjection = projection;
+            }
+        }
+        return closestProjection;
+    }
+
+    /**
+     * returns projection to edge
+     *
+     * @param point       point in plane of triangle
+     * @param edgeVertex1 first vertex of edge
+     * @param edgeVertex2 second vertex of edge
+     * @return projection to edge
+     */
+    private Vector3d getProjectionToEdge(Vector3d point, Vector3d edgeVertex1, Vector3d edgeVertex2) {
+        Vector3d ab = new Vector3d();
+        ab.sub(edgeVertex1, edgeVertex2);
+        Vector3d ap = new Vector3d();
+        ap.sub(edgeVertex1, point);
+        double t = ab.dot(ap) / ab.lengthSquared();
+        return new Vector3d(edgeVertex1.x + t * ab.x, edgeVertex1.y + t * ab.y, edgeVertex1.z + t * ab.z);
+    }
+    
+    
+    /**
+     * Helper class for pairs.
+     * 
+     * @param <K> key 
+     * @param <V> value
+     */
+    private class Pair<K,V> {
+        private K key;
+        private V value;
+       
+        /**
+         * Constructor.
+         * @param key key
+         * @param value value
+         */
+        public Pair(K key, V value) {
+            this.key = key;
+            this.value = value;
+        }
+        
+        public K getKey() {
+            return key;
+        }
+        
+        public V getValue() {
+            return value;
+        }
+    }
+}
diff --git a/Comparison/src/main/java/cz/fidentis/analyst/comparison/Registration.java b/Comparison/src/main/java/cz/fidentis/analyst/comparison/Registration.java
new file mode 100644
index 0000000000000000000000000000000000000000..dc681e550c9a874deb2cbff8adb7b4c89bea5b3c
--- /dev/null
+++ b/Comparison/src/main/java/cz/fidentis/analyst/comparison/Registration.java
@@ -0,0 +1,27 @@
+package cz.fidentis.analyst.comparison;
+
+import cz.fidentis.analyst.mesh.core.MeshFacet;
+
+/**
+ * 
+ * @author Matej Lukes
+ */
+public class Registration {
+
+    /**
+     * Heler method - TO DO
+     * 
+     * @param facet main facet
+     * @param registeredFacet refistered facet
+     * @param method registration method
+     * @return TO DO
+     */
+    public static MeshFacet register(MeshFacet facet, MeshFacet registeredFacet, RegistrationMethod method) {
+        switch (method) {
+            case NO_REGISTRATION:
+                return registeredFacet;
+            default:
+                return null;
+        }
+    }
+}
diff --git a/Comparison/src/main/java/cz/fidentis/analyst/comparison/RegistrationMethod.java b/Comparison/src/main/java/cz/fidentis/analyst/comparison/RegistrationMethod.java
new file mode 100644
index 0000000000000000000000000000000000000000..730fcf274e47849e2ef3b8bd6b19d04ecc446201
--- /dev/null
+++ b/Comparison/src/main/java/cz/fidentis/analyst/comparison/RegistrationMethod.java
@@ -0,0 +1,8 @@
+package cz.fidentis.analyst.comparison;
+
+/**
+ * @author Matej Lukes
+ */
+public enum RegistrationMethod {
+    NO_REGISTRATION
+}
diff --git a/Comparison/src/test/java/cz/fidentis/analyst/comparison/ComparisonTest.java b/Comparison/src/test/java/cz/fidentis/analyst/comparison/ComparisonTest.java
new file mode 100644
index 0000000000000000000000000000000000000000..ee833841710249d8083e02bcc1a01cab13045fab
--- /dev/null
+++ b/Comparison/src/test/java/cz/fidentis/analyst/comparison/ComparisonTest.java
@@ -0,0 +1,8 @@
+package cz.fidentis.analyst.comparison;
+
+import static org.junit.jupiter.api.Assertions.*;
+import org.junit.jupiter.api.Test;
+
+public class ComparisonTest {
+
+}
\ No newline at end of file