Source file point.ml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
open Math

type t = {x: float; y: float}

let make x y : t = {x; y}

let ( % ) = make

module Tbl = Hashtbl.Make (struct
  type nonrec t = t

  let equal = ( == )

  let hash = Hashtbl.hash
end)

let orig = make 0. 0.

let center {x; y} {x= x'; y= y'} =
  let newx =
    if x > x' then x' +. ((x -. x') /. 2.) else x +. ((x' -. x) /. 2.)
  and newy =
    if y > y' then y' +. ((y -. y') /. 2.) else y +. ((y' -. y) /. 2.)
  in
  make newx newy

let determinant a b c =
  ((b.x -. a.x) *. (c.y -. a.y)) -. ((b.y -. a.y) *. (c.x -. a.x))

let iso_barycenter pts =
  let rec aux pts sumx sumy nb =
    match pts with
    | [] -> make (sumx /. nb) (sumy /. nb)
    | h :: tl -> aux tl (sumx +. h.x) (sumy +. h.y) (nb +. 1.)
  in
  aux pts 0. 0. 0.

let barycenter weighted_pts =
  let rec aux pts sumx sumy sumw =
    match pts with
    | [] -> make (sumx /. sumw) (sumy /. sumw)
    | (pt, w) :: tl ->
        aux tl ((w *. pt.x) +. sumx) ((w *. pt.y) +. sumy) (w +. sumw)
  in
  aux weighted_pts 0. 0. 0.

let sq_distance ({x= a; y= b} : t) ({x= c; y= d} : t) : distance =
  let diffX = a -. c and diffY = b -. d in
  (diffX *. diffX) +. (diffY *. diffY)

let distance ({x= a; y= b} : t) ({x= c; y= d} : t) : distance =
  hypot (a -. c) (b -. d)

let x_coord (p : t) = p.x

let y_coord (p : t) = p.y

let scale_x (p : t) f = {p with x= p.x *. f}

let scale_y (p : t) f = {p with y= p.y *. f}

let translate dx dy ({x; y} : t) : t = make (x +. dx) (y +. dy)

let transform aff p =
  let x, y = Affine.transform_point aff p.x p.y in
  {x; y}

let reflection center p =
  translate (center.x -. p.x) (center.y -. p.y) center

let rotate pivot p angle =
  let px =
    (cos angle *. (p.x -. pivot.x))
    -. (sin angle *. (p.y -. pivot.y))
    +. pivot.x
  and py =
    (sin angle *. (p.x -. pivot.x))
    +. (cos angle *. (p.y -. pivot.y))
    +. pivot.y
  in
  make px py

let rotate_angle pivot p angle = rotate pivot p (angle *. Math.deg_to_rad)

let print fmt {x; y} = Format.fprintf fmt "{x=%f; y=%f}" x y