-
Notifications
You must be signed in to change notification settings - Fork 3
/
constraint.go
64 lines (54 loc) · 1.54 KB
/
constraint.go
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
package main
import (
"image/color"
"math"
"gioui.org/layout"
)
type Constraint struct {
p1, p2 *Particle
length float64
color color.NRGBA
}
// NewConstraint creates a new constraint between two points/particles.
// The constraint actually is a stick which connects two points.
func NewConstraint(p1, p2 *Particle, length float64, col color.NRGBA) *Constraint {
return &Constraint{
p1, p2, length, col,
}
}
// Update updates the stick between two points by resolving the constraints between them.
func (c *Constraint) Update(gtx layout.Context, cloth *Cloth, mouse *Mouse) {
dx := c.p1.x - c.p2.x
dy := c.p1.y - c.p2.y
dist := math.Sqrt(dx*dx + dy*dy)
if dist < c.length {
return
}
// Tear up the cloth under the mouse position if the applied force exceeds a certain threshold.
// The threshold is the distance between the two points.
if mouse.getDragging() {
if dist > 150 {
c.removeConstraint(cloth)
}
}
diff := (c.length - dist) / dist
mul := diff * 0.35 * (1 - c.length/dist)
offsetX, offsetY := dx*mul, dy*mul
if !c.p1.pinX {
c.p1.x += offsetX
c.p1.y += offsetY
}
if !c.p2.pinX {
c.p2.x -= offsetX
c.p2.y -= offsetY
}
}
// removeConstraint removes a specific constraint (stick) from the collection, stored into a slice.
func (c *Constraint) removeConstraint(cloth *Cloth) {
for idx, constraint := range cloth.constraints {
if c == constraint {
cloth.constraints = append(cloth.constraints[:idx], cloth.constraints[idx+1:]...)
break
}
}
}