forked from Dan-Piker/K2Goals
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathHinge.cs
104 lines (89 loc) · 3.31 KB
/
Hinge.cs
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
using System.Collections.Generic;
using Rhino.Geometry;
using System;
namespace KangarooSolver.Goals
{
/// <summary>
/// Bending resistance between a pair of triangles
/// Based on ideas from
/// "Discrete Shells" by Grinspun et al
/// http://www.cs.columbia.edu/cg/pdfs/10_ds.pdf
/// and
/// "Interactive Form-Finding of Elastic Origami" by Tachi
/// http://www.tsg.ne.jp/TT/cg/ElasticOrigami_Tachi_IASS2013.pdf
/// </summary>
public class Hinge : GoalObject
{
public double Strength;
public double RestAngle;
public Hinge()
{
}
public Hinge(int P0, int P1, int P2, int P3, double RA, double k)
{
PIndex = new int[4] { P0, P1, P2, P3 };
Move = new Vector3d[4];
Weighting = new double[4];
Strength = k;
RestAngle = RA;
}
public Hinge(Point3d P0, Point3d P1, Point3d P2, Point3d P3, double RA, double k)
{
PPos = new Point3d[4] { P0, P1, P2, P3 };
PIndex = new int[4];
Move = new Vector3d[4];
Weighting = new double[4];
Strength = k;
RestAngle = RA;
}
public override void Calculate(List<KangarooSolver.Particle> p)
{
Point3d P0 = p[PIndex[0]].Position;
Point3d P1 = p[PIndex[1]].Position;
Point3d P2 = p[PIndex[2]].Position;
Point3d P3 = p[PIndex[3]].Position;
Vector3d V01 = P1 - P0;
Vector3d V02 = P2 - P0;
Vector3d V03 = P3 - P0;
Vector3d V21 = P1 - P2;
Vector3d V31 = P1 - P3;
double L01 = V01.Length;
double invL01 = 1.0 / L01;
//get heights
//there is some re-use possible here to speed up
double H0 = (V02 - V02 * invL01 * invL01 * V01 * V01).Length;
double H1 = (V03 - V03 * invL01 * invL01 * V01 * V01).Length;
double H = 0.5 / (H0 + H1);
double Dot0201 = V02 * V01;
double Dot0301 = V03 * V01;
double Dot2101 = V21 * V01;
double Dot3101 = V31 * V01;
//Cot is dot over mag of cross
//Get normals
Vector3d Cross0 = Vector3d.CrossProduct(V02, V01);
Vector3d Cross1 = Vector3d.CrossProduct(V01, V03);
double CurrentAngle = Vector3d.VectorAngle(Cross0, Cross1, new Plane(P0, V01));
if (CurrentAngle > Math.PI) { CurrentAngle = CurrentAngle - 2 * Math.PI; }
double AngleError = CurrentAngle - RestAngle;
double InvL = 1.0 / Cross0.Length;
double Cot0u = Dot0201 * InvL;
double Cot0v = Dot2101 * InvL;
InvL = 1.0 / Cross1.Length;
double Cot1u = Dot0301 * InvL;
double Cot1v = Dot3101 * InvL;
double D = AngleError * H * 0.5;
Vector3d A = Cross0 * D;
Move[0] = Cot0v * A;
Move[1] = Cot0u * A;
Move[2] = -(Cot0v + Cot0u) * A;
A = Cross1 * D;
Move[0] += Cot1v * A;
Move[1] += Cot1u * A;
Move[3] = -(Cot1v + Cot1u) * A;
Weighting[0] = Strength;
Weighting[1] = Strength;
Weighting[2] = Strength;
Weighting[3] = Strength;
}
}
}