forked from sugarlabs/tictactoe
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathanim.py
139 lines (121 loc) · 4.65 KB
/
anim.py
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
130
131
132
133
134
135
136
137
138
139
# Copyright (C) 2022 Sparsh Goenka
#
# 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 <https://www.gnu.org/licenses/>.
import pygame as pg
from math import sqrt
import g
class Animate:
LINEAR = lambda x: x
EASE_OUT_QUART = lambda x: 1 - pow(1 - x, 4)
def EASE_IO_QUART(x):
return 8 * pow(x, 4) if x < 0.5 else 1 - pow(-2 * x + 2, 4) / 2
def __init__(self, main, dur=500, color=g.WHITE, fn=EASE_OUT_QUART):
self.main = main
self.color = color
self.function = fn
self.remove = False
self.dur = dur
self.start_time = pg.time.get_ticks()
self.final_time = self.start_time + dur
self.type = None
self.sub_animations = None
def line(self, p1, p2, width=g.LINE_WIDTH):
self.type = "line"
self.finished = False
self.width = width
self.p1 = pg.Vector2(p1)
self.p2 = pg.Vector2(p2)
self.p = self.p2 - self.p1
self.length = sqrt(self.p.x ** 2 + self.p.y ** 2)
return self
def circle(self, center, radius=g.CIRCLE_RADIUS, width=g.CIRCLE_WIDTH):
self.type = "circle"
self.finished = False
self.width = width
self.radius = radius
self.r = radius
self.center = pg.Vector2(center)
self.rect = pg.Rect(
(center - pg.Vector2(radius, radius)), (2 * radius, 2 * radius)
)
return self
def cross(self, center, length=g.CROSS_LENGTH, width=g.CROSS_WIDTH):
self.type = "cross"
points = [
pg.Vector2(-length, 0),
pg.Vector2(length, 0),
pg.Vector2(0, length),
pg.Vector2(0, -length),
]
for i in range(2):
points[i] = points[i].rotate(40) + pg.Vector2(center)
points[i + 2] = points[i + 2].rotate(50) + pg.Vector2(center)
self.sub_animations = [
Animate(self.main, self.dur + i * 150, self.color, self.function).line(
points[i], points[i + 1], width
)
for i in range(0, len(points), 2)
]
return self
def setup_remove(self, dur=500):
self.finished = True
self.remove = True
self.dur = dur
self.start_time = pg.time.get_ticks()
self.final_time = self.start_time + self.dur
if self.sub_animations is not None:
for i in self.sub_animations:
i.setup_remove(dur)
def update(self, skip=False):
# Call method recursively
if self.type == "cross":
for animation in self.sub_animations:
animation.update()
return
# Calculate the animation
if not self.finished or self.remove:
if pg.time.get_ticks() < self.final_time and not skip:
fraction = (pg.time.get_ticks() - self.start_time) / self.dur
if fraction < 0.01:
fraction += 0.008
if self.remove:
fraction = 1 - fraction
if self.type == "line":
self.p.scale_to_length(self.length * self.function(fraction))
elif self.type == "circle":
self.r = self.radius - self.width * self.function(fraction)
else:
if self.type == "line":
self.p = self.p2 - self.p1
if self.remove:
self.p = [0, 0]
if self.type == "circle":
self.r = self.radius - self.width
if self.remove:
self.r = self.radius
self.remove = False
self.finished = True
# Draw stuff
if self.type == "line":
pg.draw.line(g.WIN, self.color, self.p1, self.p + self.p1, self.width)
elif self.type == "circle":
pg.draw.circle(
g.WIN,
self.color,
(int(self.center.x), int(self.center.y)),
int(self.radius),
)
pg.draw.circle(
g.WIN, g.BLACK, (int(self.center.x), int(self.center.y)), int(self.r)
)