Description
There are multiple ways to determine the projected point onto a line, my personal approach uses the parametric equation of lines.
Considering the following : We have a line defined by points A(Xa, Ya) and B(Xb, Yb), we have an arbitrary point C(Xc, Yc), a vector V(Xv, Yv) and we want to determine the coordinates of point D(Xd, Yd) representing the projected point from C to line AB.
data:image/s3,"s3://crabby-images/0007f/0007f481a32384f98d2cfde39638e49584779b9c" alt="20181013-2"
data:image/s3,"s3://crabby-images/0007f/0007f481a32384f98d2cfde39638e49584779b9c" alt="20181013-2"
The parametric equation represents the function of the line an it’s a factor of t where t represents the relative position of a point on the line. For our line we have the following equations:
data:image/s3,"s3://crabby-images/664eb/664eb1901d212c4b9e012032b677c1a225509143" alt="20181013-3 20181013-3"
data:image/s3,"s3://crabby-images/664eb/664eb1901d212c4b9e012032b677c1a225509143" alt="20181013-3 20181013-3"
data:image/s3,"s3://crabby-images/369f8/369f899615a21cd3d5ede214c7b28dd2739fa1ba" alt="20181013-4 20181013-4"
data:image/s3,"s3://crabby-images/369f8/369f899615a21cd3d5ede214c7b28dd2739fa1ba" alt="20181013-4 20181013-4"
From these 2 equations we can determine a1 and b1 by substituding t with 1. The parameter “t” will map a point on a line with values from 0 to 1, where a value of 0 represents the start point and a value of 1 represents the end point.
By considering t = 1 we must also consider that xt and yt belong to the end point. Therefore :
data:image/s3,"s3://crabby-images/aca55/aca55366cc4a7902efef0e6e35915dd484496408" alt="20181013-5 20181013-5"
data:image/s3,"s3://crabby-images/aca55/aca55366cc4a7902efef0e6e35915dd484496408" alt="20181013-5 20181013-5"
data:image/s3,"s3://crabby-images/8b698/8b6984feb63fe3a69d93cab137283e813ae5ab72" alt="20181013-6 20181013-6"
data:image/s3,"s3://crabby-images/8b698/8b6984feb63fe3a69d93cab137283e813ae5ab72" alt="20181013-6 20181013-6"
The goal is to find 2 parametric equations of 2 lines, and find the common value of t between them. From the the point C and the vector V we will create a line. We do so by determining translating C with a small distance in the direction of V. The new point C1 will be the end point of line CC1
data:image/s3,"s3://crabby-images/b6e75/b6e7531efeea93e62125815e00eaf76db38218d1" alt="20181013-7 20181013-7"
data:image/s3,"s3://crabby-images/b6e75/b6e7531efeea93e62125815e00eaf76db38218d1" alt="20181013-7 20181013-7"
data:image/s3,"s3://crabby-images/e7946/e7946eb578be11172ae5b899dfd3581a82300613" alt="20181013-8 20181013-8"
data:image/s3,"s3://crabby-images/e7946/e7946eb578be11172ae5b899dfd3581a82300613" alt="20181013-8 20181013-8"
Now that we have our second line defined, we can also determine it’s coefficients for the parametric equation:
data:image/s3,"s3://crabby-images/c745d/c745d36ad91b9513293aa7f594d3f41557782fac" alt="20181013-9 20181013-9"
data:image/s3,"s3://crabby-images/c745d/c745d36ad91b9513293aa7f594d3f41557782fac" alt="20181013-9 20181013-9"
data:image/s3,"s3://crabby-images/27464/2746422833481517fd5def909a4f53949f8445c3" alt="20181013-10 20181013-10"
data:image/s3,"s3://crabby-images/27464/2746422833481517fd5def909a4f53949f8445c3" alt="20181013-10 20181013-10"
We have 2 lines and 2 parametric equations (to rule them all!). The point that we have to determine has a different “t” for each line but the same “xt” and “”yt”
data:image/s3,"s3://crabby-images/46677/46677f5c2d4bdbf4a998a5f8455177f609e13c14" alt="20181013-11 20181013-11"
data:image/s3,"s3://crabby-images/46677/46677f5c2d4bdbf4a998a5f8455177f609e13c14" alt="20181013-11 20181013-11"
Solving these equations leads us to the parameter “t” of the projected point on line AB:
data:image/s3,"s3://crabby-images/1d402/1d40238a3c82766afeb7c424b10ee726f1ecf35c" alt="20181013-12 20181013-12"
data:image/s3,"s3://crabby-images/1d402/1d40238a3c82766afeb7c424b10ee726f1ecf35c" alt="20181013-12 20181013-12"
Implementation
public static double ParameterAtPoint(Line2D line2D, Point2D point2D) { //determine line coefficients double A = line2D.EndPoint.X - line2D.StartPoint.X; //double B = line2D.EndPoint.Y - line2D.StartPoint.Y; double t = (point2D.X - line2D.StartPoint.X) / A; return t; } public static Point2D getProjectedPointOnLine(Point2D toProject, Line2D line2D, Vector2D direction) { Point2D otherPoint = new Point2D(toProject.X + direction.X, toProject.Y + direction.Y); double Al = line2D.EndPoint.X - line2D.StartPoint.X; double Bl = line2D.EndPoint.Y - line2D.StartPoint.Y; double Apl = otherPoint.X - toProject.X; double Bpl = otherPoint.Y - toProject.Y; //t parameter on the projection line double t2 = ((toProject.Y - line2D.StartPoint.Y) * Al - (toProject.X - line2D.StartPoint.X) * Bl) / (Apl * Bl - Al * Bpl); //t parameter value on the input line double t1 = ((line2D.StartPoint.Y - toProject.Y) * Apl - (line2D.StartPoint.X - toProject.X) * Bpl) / (Al * Bpl - Apl * Bl); return PointAtParameter(line2D, t1); }