summaryrefslogtreecommitdiff
path: root/lib/triangle.h
blob: e43065373e20cfbe5657f4a59b8aeb032222e1ec (plain)
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
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
#pragma once

#include "config/types.h"
#include "maths.h"
#include <glm/glm.hpp>

template<glm::length_t Dim, Arithmetic T, glm::qualifier Q = glm::defaultp>
struct Triangle : public glm::vec<3, glm::vec<Dim, T, Q>> {
	using Point = glm::vec<Dim, T, Q>;
	using Base = glm::vec<3, glm::vec<Dim, T, Q>>;
	using Base::Base;

	[[nodiscard]] constexpr Point
	operator*(BaryPosition bari) const
	{
		return p(0) + (sideDifference(1) * bari.x) + (sideDifference(2) * bari.y);
	}

	[[nodiscard]] constexpr Point
	centroid() const
	{
		return [this]<glm::length_t... Axis>(std::integer_sequence<glm::length_t, Axis...>) {
			return Point {(p(0)[Axis] + p(1)[Axis] + p(2)[Axis]) / 3 ...};
		}(std::make_integer_sequence<glm::length_t, Dim>());
	}

	[[nodiscard]] constexpr auto
	area() const
		requires(Dim == 3)
	{
		return glm::length(crossProduct(sideDifference(1), sideDifference(2))) / T {2};
	}

	[[nodiscard]] constexpr auto
	area() const
		requires(Dim == 2)
	{
		return std::abs((sideDifference(1).x * sideDifference(2).y) - (sideDifference(2).x * sideDifference(1).y)) / 2;
	}

	[[nodiscard]] constexpr Normal3D
	normal() const
		requires(Dim == 3)
	{
		return crossProduct(sideDifference(1), sideDifference(2));
	}

	[[nodiscard]] constexpr auto
	height()
	{
		return (area() * 2) / ::distance(p(0), p(1));
	}

	[[nodiscard]] constexpr Normal3D
	nnormal() const
		requires(Dim == 3)
	{
		return glm::normalize(normal());
	}

	[[nodiscard]] constexpr auto
	sideDifference(glm::length_t side) const
	{
		return difference(p(side), p(0));
	}

	[[nodiscard]] constexpr auto
	angle(glm::length_t corner) const
	{
		return Arc {P(corner), P(corner + 2), P(corner + 1)}.length();
	}

	template<glm::length_t D = Dim>
	[[nodiscard]] constexpr auto
	angleAt(const glm::vec<D, T, Q> pos) const
		requires(D <= Dim)
	{
		for (glm::length_t i {}; i < 3; ++i) {
			if (glm::vec<D, T, Q> {p(i)} == pos) {
				return angle(i);
			}
		}
		return 0.F;
	}

	[[nodiscard]] constexpr auto
	isUp() const
	{
		const auto edgeAB = sideDifference(1);
		const auto edgeAC = sideDifference(2);
		return edgeAB.x * edgeAC.y >= edgeAB.y * edgeAC.x;
	}

	[[nodiscard]] constexpr auto
	p(const glm::length_t idx) const
	{
		return Base::operator[](idx);
	}

	[[nodiscard]] constexpr auto
	P(const glm::length_t idx) const
	{
		return Base::operator[](idx % 3);
	}

	[[nodiscard]] constexpr Point *
	begin()
	{
		return &(Base::x);
	}

	[[nodiscard]] constexpr const Point *
	begin() const
	{
		return &(Base::x);
	}

	[[nodiscard]] constexpr Point *
	end()
	{
		return begin() + 3;
	}

	[[nodiscard]] constexpr const Point *
	end() const
	{
		return begin() + 3;
	}
};