From: Antonio Ospite <ao2@ao2.it> Date: Sat, 6 Aug 2016 17:27:25 +0000 (+0200) Subject: Initial import X-Git-Url: https://git.ao2.it/experiments/line_equation.git/commitdiff_plain/refs/heads/master?ds=inline Initial import --- 9a97c7bceb623e978c2f34cfbbbbc7c5ea21a5ee diff --git a/line_equation.py b/line_equation.py new file mode 100755 index 0000000..0bce46c --- /dev/null +++ b/line_equation.py @@ -0,0 +1,227 @@ +#!/usr/bin/env python3 +# +# line_equation - some experiments about representing a line +# +# Copyright (C) 2016 Antonio Ospite <ao2@ao2.it> +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see <http://www.gnu.org/licenses/>. + +"""Some experiments about representing a line passing for two points""" + +import math +import matplotlib.pyplot as plt +import numpy as np + +EPSILON = 10e-6 +INFINITY = 10e6 + +# pylint: disable=invalid-name + + +def distance(p1, p2): + """Return the distance between the two points p1 and p2""" + dx = p2[0] - p1[0] + dy = p2[1] - p1[1] + + return math.sqrt(dx * dx + dy * dy) + + +def line_eq_slope_intercept(p1, p2): + """Return the coefficients of the line equation in the slope/intercept + form: + + y = mx + q + """ + dx = p2[0] - p1[0] + dy = p2[1] - p1[1] + + # For vertical lines the slope would be undefined in theory. + # For practical purposes a large value can work. + if dx == 0: + slope = INFINITY * math.copysign(1, dy) + else: + slope = dy / dx + + intercept = -slope * p1[0] + p1[1] + + return slope, intercept + + +def line_angle(p1, p2): + """Return the angle between the line passing for p1 and p2, and the positive + x axis. + + The return value is in radians. + """ + dx = p2[0] - p1[0] + dy = p2[1] - p1[1] + + angle = math.atan2(dy, dx) + if angle < 0: + angle += 2 * math.pi + + return angle + + +def slope_intercept_to_a_b_c(slope, intercept): + """Convert the coefficients from the slope/intercept form: + + y = mx + q + + to the coefficients in the linear equation form: + + ax + by + c = 0 + """ + a = -slope + b = 1. + c = -intercept + + return a, b, c + + +def line_eq_two_points(p1, p2): + """Return the coefficients for the line equation, in the linear equation + form: + + ax + by + c = 0 + """ + a = p2[1] - p1[1] + b = - (p2[0] - p1[0]) + c = - (p1[0] * p2[1] - p2[0] * p1[1]) + + return a, b, c + + +def line_calc_normal(p1, p2): + """Return the unit normal to the line passing for two points.""" + dx = p2[0] - p1[0] + dy = p2[1] - p1[1] + + n_length = math.sqrt(dx*dx + dy*dy) + if n_length < EPSILON: + nx = 0 + ny = 0 + else: + nx = dy / n_length + ny = -dx / n_length + + return nx, ny + + +def list_rotate(l, n): + return l[n:] + l[:n] + + +def main(): + points = [ + [132.143, 941.429], + [480.714, 905.714], + [480.714, 735.714], + [180.0, 783.571] + ] + + # The points are assumed to be in clockwise order + for p1, p2 in zip(points, list_rotate(points, 1)): + print("p1", p1) + print("p2", p2) + + line_x_extremes = [p1[0], p2[0]] + line_y_extremes = [p1[1], p2[1]] + + plt.plot(line_x_extremes, line_y_extremes, 'k') + + x_samples = np.linspace(min(line_x_extremes), max(line_x_extremes), 15) + + # slope/intercept form + m, q = line_eq_slope_intercept(p1, p2) + + if abs(m) == INFINITY: + print("m is undefined: vertical line!") + y_samples = \ + np.linspace(min(line_y_extremes), max(line_y_extremes), 15) + else: + y_samples = x_samples * m + q + + plt.plot(x_samples, y_samples, 'wo', markersize=15) + + # linear equation form + a, b, c = line_eq_two_points(p1, p2) + + if b == 0: + print("b is 0: vertical line!") + y_samples = \ + np.linspace(min(line_y_extremes), max(line_y_extremes), 15) + else: + y_samples = (-x_samples * a - c) / b + + plt.plot(x_samples, y_samples, 'ws', markersize=10) + + # parametric forms + t = np.linspace(0, 1, len(x_samples)) + + # point and direction form + dx = p2[0] - p1[0] + dy = p2[1] - p1[1] + + x_samples = p1[0] + dx * t + y_samples = p1[1] + dy * t + plt.plot(x_samples, y_samples, 'm+', markersize=10) + + # or using a point and the normal, but in this case the normal has to + # be rotated by 90 degrees counterclockwise + nx, ny = line_calc_normal(p1, p2) + rnx = -ny + rny = nx + + length = distance(p1, p2) + x_samples = p1[0] + rnx * t * length + y_samples = p1[1] + rny * t * length + plt.plot(x_samples, y_samples, 'bx', markersize=10) + + # draw the normal + mid_x = (p1[0] + p2[0]) / 2 + mid_y = (p1[1] + p2[1]) / 2 + n_length = 20 + plt.arrow(mid_x, mid_y, nx * n_length, ny * n_length, + head_width=10, head_length=5, fc='k', ec='k') + + # check the other functions + print("angle: %g" % math.degrees(line_angle(p1, p2))) + a1, b1, c1 = slope_intercept_to_a_b_c(m, q) + print(a1, b1, c1) + if b != 0: + print(a / b, b / b, c / b) + else: + print(a, b, c) + print() + + # draw the vertices on top of everything else + xvalues = [p[0] for p in points] + yvalues = [p[1] for p in points] + plt.plot(xvalues, yvalues, 'ro') + plt.title('Line passing for two points') + plt.xlabel('x') + plt.ylabel('y') + + plt.plot([], 'k', label="line") + plt.plot([], 'wo', label="y = mx + q") + plt.plot([], 'ws', label="ax + by + c = 0") + plt.plot([], 'm+', label="P = P1 + D * t") + plt.plot([], 'bx', label="P = P1 + N * t * length") + plt.legend(loc='lower left', numpoints=1, fancybox=True, prop={'size': 10}) + plt.show() + + +if __name__ == "__main__": + main()