2016-11-07 11:05:41 -08:00
|
|
|
from helpers import *
|
|
|
|
|
|
|
|
from mobject import Mobject
|
2017-06-05 21:35:09 -07:00
|
|
|
from mobject.vectorized_mobject import VGroup, VMobject, VectorizedPoint
|
2016-11-07 11:05:41 -08:00
|
|
|
from mobject.svg_mobject import SVGMobject
|
2018-02-12 22:33:41 +01:00
|
|
|
from mobject.tex_mobject import TextMobject, TexMobject, Brace
|
2016-11-07 11:05:41 -08:00
|
|
|
|
2016-12-26 07:10:38 -08:00
|
|
|
from animation import Animation
|
2018-03-08 13:55:01 -08:00
|
|
|
from animation.simple_animations import Rotating
|
|
|
|
from animation.compositions import LaggedStart, AnimationGroup
|
2018-01-29 21:29:36 -08:00
|
|
|
from animation.transform import ApplyMethod, FadeIn, GrowFromCenter
|
2016-12-26 07:10:38 -08:00
|
|
|
|
2017-06-14 21:54:21 -07:00
|
|
|
from topics.geometry import Circle, Line, Rectangle, Square, \
|
|
|
|
Arc, Polygon, SurroundingRectangle
|
2017-02-18 14:02:46 -08:00
|
|
|
from topics.three_dimensions import Cube
|
2016-11-07 11:05:41 -08:00
|
|
|
|
2017-06-30 15:41:45 -07:00
|
|
|
class Lightbulb(SVGMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "lightbulb",
|
|
|
|
"height" : 1,
|
|
|
|
"stroke_color" : YELLOW,
|
2017-12-03 16:22:24 -08:00
|
|
|
"stroke_width" : 3,
|
|
|
|
"fill_color" : YELLOW,
|
|
|
|
"fill_opacity" : 0,
|
2017-06-30 15:41:45 -07:00
|
|
|
}
|
|
|
|
|
2017-06-20 14:05:48 -07:00
|
|
|
class BitcoinLogo(SVGMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "Bitcoin_logo",
|
|
|
|
"height" : 1,
|
|
|
|
"fill_color" : "#f7931a",
|
|
|
|
"inner_color" : WHITE,
|
|
|
|
"fill_opacity" : 1,
|
|
|
|
"stroke_width" : 0,
|
|
|
|
}
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
SVGMobject.__init__(self, **kwargs)
|
|
|
|
self[0].set_fill(self.fill_color, self.fill_opacity)
|
|
|
|
self[1].set_fill(self.inner_color, 1)
|
|
|
|
|
2017-06-08 21:56:53 -07:00
|
|
|
class Guitar(SVGMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "guitar",
|
|
|
|
"height" : 2.5,
|
|
|
|
"fill_color" : DARK_GREY,
|
|
|
|
"fill_opacity" : 1,
|
|
|
|
"stroke_color" : WHITE,
|
|
|
|
"stroke_width" : 0.5,
|
|
|
|
}
|
|
|
|
|
2017-06-07 14:34:39 -07:00
|
|
|
class SunGlasses(SVGMobject):
|
2017-06-05 12:47:03 -07:00
|
|
|
CONFIG = {
|
2017-06-07 14:34:39 -07:00
|
|
|
"file_name" : "sunglasses",
|
|
|
|
"glasses_width_to_eyes_width" : 1.1,
|
2017-06-05 12:47:03 -07:00
|
|
|
}
|
2017-06-07 14:34:39 -07:00
|
|
|
def __init__(self, pi_creature, **kwargs):
|
|
|
|
SVGMobject.__init__(self, **kwargs)
|
|
|
|
self.set_stroke(WHITE, width = 0)
|
|
|
|
self.set_fill(GREY, 1)
|
|
|
|
self.scale_to_fit_width(
|
|
|
|
self.glasses_width_to_eyes_width*pi_creature.eyes.get_width()
|
2017-06-05 12:47:03 -07:00
|
|
|
)
|
2017-06-07 14:34:39 -07:00
|
|
|
self.move_to(pi_creature.eyes, UP)
|
2017-06-05 12:47:03 -07:00
|
|
|
|
2017-04-12 09:06:04 -07:00
|
|
|
class Speedometer(VMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"arc_angle" : 4*np.pi/3,
|
|
|
|
"num_ticks" : 8,
|
|
|
|
"tick_length" : 0.2,
|
|
|
|
"needle_width" : 0.1,
|
|
|
|
"needle_height" : 0.8,
|
|
|
|
"needle_color" : YELLOW,
|
|
|
|
}
|
|
|
|
def generate_points(self):
|
|
|
|
start_angle = np.pi/2 + self.arc_angle/2
|
|
|
|
end_angle = np.pi/2 - self.arc_angle/2
|
|
|
|
self.add(Arc(
|
|
|
|
start_angle = start_angle,
|
|
|
|
angle = -self.arc_angle
|
|
|
|
))
|
|
|
|
tick_angle_range = np.linspace(start_angle, end_angle, self.num_ticks)
|
|
|
|
for index, angle in enumerate(tick_angle_range):
|
|
|
|
vect = rotate_vector(RIGHT, angle)
|
|
|
|
tick = Line((1-self.tick_length)*vect, vect)
|
|
|
|
label = TexMobject(str(10*index))
|
|
|
|
label.scale_to_fit_height(self.tick_length)
|
|
|
|
label.shift((1+self.tick_length)*vect)
|
|
|
|
self.add(tick, label)
|
|
|
|
|
|
|
|
needle = Polygon(
|
|
|
|
LEFT, UP, RIGHT,
|
|
|
|
stroke_width = 0,
|
|
|
|
fill_opacity = 1,
|
|
|
|
fill_color = self.needle_color
|
|
|
|
)
|
|
|
|
needle.stretch_to_fit_width(self.needle_width)
|
|
|
|
needle.stretch_to_fit_height(self.needle_height)
|
2018-01-20 13:14:17 -08:00
|
|
|
needle.rotate(start_angle - np.pi/2, about_point = ORIGIN)
|
2017-04-12 09:06:04 -07:00
|
|
|
self.add(needle)
|
|
|
|
self.needle = needle
|
|
|
|
|
|
|
|
self.center_offset = self.get_center()
|
|
|
|
|
|
|
|
def get_center(self):
|
|
|
|
result = VMobject.get_center(self)
|
|
|
|
if hasattr(self, "center_offset"):
|
|
|
|
result -= self.center_offset
|
|
|
|
return result
|
|
|
|
|
|
|
|
def get_needle_tip(self):
|
|
|
|
return self.needle.get_anchors()[1]
|
|
|
|
|
|
|
|
def get_needle_angle(self):
|
|
|
|
return angle_of_vector(
|
|
|
|
self.get_needle_tip() - self.get_center()
|
|
|
|
)
|
|
|
|
|
|
|
|
def rotate_needle(self, angle):
|
|
|
|
self.needle.rotate(angle, about_point = self.get_center())
|
|
|
|
return self
|
|
|
|
|
|
|
|
def move_needle_to_velocity(self, velocity):
|
|
|
|
max_velocity = 10*(self.num_ticks-1)
|
|
|
|
proportion = float(velocity) / max_velocity
|
|
|
|
start_angle = np.pi/2 + self.arc_angle/2
|
|
|
|
target_angle = start_angle - self.arc_angle * proportion
|
|
|
|
self.rotate_needle(target_angle - self.get_needle_angle())
|
|
|
|
return self
|
|
|
|
|
2017-04-26 16:22:05 -07:00
|
|
|
class AoPSLogo(SVGMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "aops_logo",
|
|
|
|
"height" : 1.5,
|
|
|
|
}
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
SVGMobject.__init__(self, **kwargs)
|
|
|
|
self.set_stroke(WHITE, width = 0)
|
|
|
|
colors = [BLUE_E, "#008445", GREEN_B]
|
|
|
|
index_lists = [
|
|
|
|
(10, 11, 12, 13, 14, 21, 22, 23, 24, 27, 28, 29, 30),
|
|
|
|
(0, 1, 2, 3, 4, 15, 16, 17, 26),
|
|
|
|
(5, 6, 7, 8, 9, 18, 19, 20, 25)
|
|
|
|
]
|
|
|
|
for color, index_list in zip(colors, index_lists):
|
|
|
|
for i in index_list:
|
|
|
|
self.submobjects[i].set_fill(color, opacity = 1)
|
|
|
|
|
|
|
|
self.scale_to_fit_height(self.height)
|
|
|
|
self.center()
|
|
|
|
|
2017-02-23 14:13:30 -08:00
|
|
|
class PartyHat(SVGMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "party_hat",
|
|
|
|
"height" : 1.5,
|
|
|
|
"pi_creature" : None,
|
|
|
|
"stroke_width" : 0,
|
|
|
|
"fill_opacity" : 1,
|
2018-01-15 18:16:50 -08:00
|
|
|
"propagate_style_to_family" : True,
|
2017-02-23 18:14:34 -08:00
|
|
|
"frills_colors" : [MAROON_B, PURPLE],
|
2017-02-27 15:56:22 -08:00
|
|
|
"cone_color" : GREEN,
|
2017-02-23 18:14:34 -08:00
|
|
|
"dots_colors" : [YELLOW],
|
2017-02-23 14:13:30 -08:00
|
|
|
}
|
|
|
|
NUM_FRILLS = 7
|
|
|
|
NUM_DOTS = 6
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
SVGMobject.__init__(self, **kwargs)
|
|
|
|
self.scale_to_fit_height(self.height)
|
|
|
|
if self.pi_creature is not None:
|
|
|
|
self.next_to(self.pi_creature.eyes, UP, buff = 0)
|
|
|
|
|
|
|
|
self.frills = VGroup(*self[:self.NUM_FRILLS])
|
|
|
|
self.cone = self[self.NUM_FRILLS]
|
|
|
|
self.dots = VGroup(*self[self.NUM_FRILLS+1:])
|
|
|
|
|
2017-02-23 18:14:34 -08:00
|
|
|
self.frills.gradient_highlight(*self.frills_colors)
|
2018-03-30 11:51:31 -07:00
|
|
|
self.cone.set_color(self.cone_color)
|
2017-02-23 18:14:34 -08:00
|
|
|
self.dots.gradient_highlight(*self.dots_colors)
|
2017-02-23 14:13:30 -08:00
|
|
|
|
2017-02-18 14:02:46 -08:00
|
|
|
class Laptop(VGroup):
|
|
|
|
CONFIG = {
|
|
|
|
"width" : 3,
|
|
|
|
"body_dimensions" : [4, 3, 0.05],
|
|
|
|
"screen_thickness" : 0.01,
|
|
|
|
"keyboard_width_to_body_width" : 0.9,
|
|
|
|
"keyboard_height_to_body_height" : 0.5,
|
|
|
|
"screen_width_to_screen_plate_width" : 0.9,
|
|
|
|
"key_color_kwargs" : {
|
|
|
|
"stroke_width" : 0,
|
|
|
|
"fill_color" : BLACK,
|
|
|
|
"fill_opacity" : 1,
|
|
|
|
},
|
|
|
|
"body_color" : LIGHT_GREY,
|
|
|
|
"shaded_body_color" : GREY,
|
|
|
|
"open_angle" : np.pi/4,
|
|
|
|
}
|
|
|
|
def generate_points(self):
|
|
|
|
body = Cube(side_length = 1)
|
|
|
|
for dim, scale_factor in enumerate(self.body_dimensions):
|
|
|
|
body.stretch(scale_factor, dim = dim)
|
|
|
|
body.scale_to_fit_width(self.width)
|
|
|
|
body.set_fill(self.shaded_body_color, opacity = 1)
|
|
|
|
body.sort_submobjects(lambda p : p[2])
|
|
|
|
body[-1].set_fill(self.body_color)
|
|
|
|
keyboard = VGroup(*[
|
|
|
|
VGroup(*[
|
|
|
|
Square(**self.key_color_kwargs)
|
|
|
|
for x in range(12-y%2)
|
|
|
|
]).arrange_submobjects(RIGHT, buff = SMALL_BUFF)
|
|
|
|
for y in range(4)
|
|
|
|
]).arrange_submobjects(DOWN, buff = MED_SMALL_BUFF)
|
|
|
|
keyboard.stretch_to_fit_width(
|
|
|
|
self.keyboard_width_to_body_width*body.get_width(),
|
|
|
|
)
|
|
|
|
keyboard.stretch_to_fit_height(
|
|
|
|
self.keyboard_height_to_body_height*body.get_height(),
|
|
|
|
)
|
|
|
|
keyboard.next_to(body, OUT, buff = 0.1*SMALL_BUFF)
|
|
|
|
keyboard.shift(MED_SMALL_BUFF*UP)
|
|
|
|
body.add(keyboard)
|
|
|
|
|
|
|
|
screen_plate = body.copy()
|
|
|
|
screen_plate.stretch(self.screen_thickness/self.body_dimensions[2], dim = 2)
|
|
|
|
screen = Rectangle(
|
|
|
|
stroke_width = 0,
|
|
|
|
fill_color = BLACK,
|
|
|
|
fill_opacity = 1,
|
|
|
|
)
|
|
|
|
screen.replace(screen_plate, stretch = True)
|
|
|
|
screen.scale_in_place(self.screen_width_to_screen_plate_width)
|
|
|
|
screen.next_to(screen_plate, OUT, buff = 0.1*SMALL_BUFF)
|
|
|
|
screen_plate.add(screen)
|
|
|
|
screen_plate.next_to(body, UP, buff = 0)
|
|
|
|
screen_plate.rotate(
|
|
|
|
self.open_angle, RIGHT,
|
|
|
|
about_point = screen_plate.get_bottom()
|
|
|
|
)
|
|
|
|
self.screen_plate = screen_plate
|
|
|
|
self.screen = screen
|
|
|
|
|
|
|
|
axis = Line(
|
|
|
|
body.get_corner(UP+LEFT+OUT),
|
|
|
|
body.get_corner(UP+RIGHT+OUT),
|
|
|
|
color = BLACK,
|
|
|
|
stroke_width = 2
|
|
|
|
)
|
|
|
|
self.axis = axis
|
|
|
|
|
|
|
|
self.add(body, screen_plate, axis)
|
2018-01-20 13:14:17 -08:00
|
|
|
self.rotate(5*np.pi/12, LEFT, about_point = ORIGIN)
|
|
|
|
self.rotate(np.pi/6, DOWN, about_point = ORIGIN)
|
2017-02-18 14:02:46 -08:00
|
|
|
|
2017-02-08 13:29:32 -08:00
|
|
|
class PatreonLogo(SVGMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "patreon_logo",
|
2017-06-14 21:54:21 -07:00
|
|
|
"fill_color" : "#F96854",
|
|
|
|
# "fill_color" : WHITE,
|
2017-02-08 13:29:32 -08:00
|
|
|
"fill_opacity" : 1,
|
|
|
|
"stroke_width" : 0,
|
2017-06-14 21:54:21 -07:00
|
|
|
"width" : 4,
|
2018-01-15 18:16:50 -08:00
|
|
|
"propagate_style_to_family" : True
|
2017-02-08 13:29:32 -08:00
|
|
|
}
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
SVGMobject.__init__(self, **kwargs)
|
2017-06-14 21:54:21 -07:00
|
|
|
self.scale_to_fit_width(self.width)
|
2017-02-08 13:29:32 -08:00
|
|
|
self.center()
|
|
|
|
|
2016-11-07 11:05:41 -08:00
|
|
|
class VideoIcon(SVGMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "video_icon",
|
2018-03-30 11:25:37 -07:00
|
|
|
"width" : FRAME_WIDTH/12.,
|
2016-11-07 11:05:41 -08:00
|
|
|
}
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
SVGMobject.__init__(self, **kwargs)
|
|
|
|
self.center()
|
|
|
|
self.scale_to_fit_width(self.width)
|
|
|
|
self.set_stroke(color = WHITE, width = 0)
|
|
|
|
self.set_fill(color = WHITE, opacity = 1)
|
|
|
|
|
|
|
|
class VideoSeries(VGroup):
|
|
|
|
CONFIG = {
|
2017-03-21 15:53:41 -07:00
|
|
|
"num_videos" : 11,
|
2016-11-07 11:05:41 -08:00
|
|
|
"gradient_colors" : [BLUE_B, BLUE_D],
|
|
|
|
}
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
digest_config(self, kwargs)
|
|
|
|
videos = [VideoIcon() for x in range(self.num_videos)]
|
|
|
|
VGroup.__init__(self, *videos, **kwargs)
|
|
|
|
self.arrange_submobjects()
|
2018-03-30 11:25:37 -07:00
|
|
|
self.scale_to_fit_width(FRAME_WIDTH-MED_LARGE_BUFF)
|
2016-11-07 11:05:41 -08:00
|
|
|
self.gradient_highlight(*self.gradient_colors)
|
|
|
|
|
|
|
|
class Headphones(SVGMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "headphones",
|
|
|
|
"height" : 2,
|
|
|
|
"y_stretch_factor" : 0.5,
|
|
|
|
"color" : GREY,
|
|
|
|
}
|
2017-06-09 16:10:34 -07:00
|
|
|
def __init__(self, pi_creature = None, **kwargs):
|
2016-11-07 11:05:41 -08:00
|
|
|
digest_config(self, kwargs)
|
2016-12-07 18:37:56 -08:00
|
|
|
SVGMobject.__init__(self, file_name = self.file_name, **kwargs)
|
2016-11-07 11:05:41 -08:00
|
|
|
self.stretch(self.y_stretch_factor, 1)
|
|
|
|
self.scale_to_fit_height(self.height)
|
|
|
|
self.set_stroke(width = 0)
|
|
|
|
self.set_fill(color = self.color)
|
2017-06-09 16:10:34 -07:00
|
|
|
if pi_creature is not None:
|
|
|
|
eyes = pi_creature.eyes
|
|
|
|
self.scale_to_fit_height(3*eyes.get_height())
|
|
|
|
self.move_to(eyes, DOWN)
|
|
|
|
self.shift(DOWN*eyes.get_height()/4)
|
2016-11-07 11:05:41 -08:00
|
|
|
|
2016-12-26 07:10:38 -08:00
|
|
|
class Clock(VGroup):
|
|
|
|
CONFIG = {
|
2018-01-15 18:16:50 -08:00
|
|
|
"propagate_style_to_family" : True,
|
2016-12-26 07:10:38 -08:00
|
|
|
}
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
circle = Circle()
|
|
|
|
ticks = []
|
|
|
|
for x in range(12):
|
|
|
|
alpha = x/12.
|
|
|
|
point = complex_to_R3(
|
|
|
|
np.exp(2*np.pi*alpha*complex(0, 1))
|
|
|
|
)
|
|
|
|
length = 0.2 if x%3 == 0 else 0.1
|
|
|
|
ticks.append(
|
|
|
|
Line(point, (1-length)*point)
|
|
|
|
)
|
|
|
|
self.hour_hand = Line(ORIGIN, 0.3*UP)
|
|
|
|
self.minute_hand = Line(ORIGIN, 0.6*UP)
|
|
|
|
# for hand in self.hour_hand, self.minute_hand:
|
|
|
|
# #Balance out where the center is
|
|
|
|
# hand.add(VectorizedPoint(-hand.get_end()))
|
|
|
|
|
|
|
|
VGroup.__init__(
|
|
|
|
self, circle,
|
|
|
|
self.hour_hand, self.minute_hand,
|
|
|
|
*ticks
|
|
|
|
)
|
|
|
|
|
|
|
|
class ClockPassesTime(Animation):
|
|
|
|
CONFIG = {
|
|
|
|
"run_time" : 5,
|
|
|
|
"hours_passed" : 12,
|
|
|
|
"rate_func" : None,
|
|
|
|
}
|
|
|
|
def __init__(self, clock, **kwargs):
|
|
|
|
digest_config(self, kwargs)
|
|
|
|
assert(isinstance(clock, Clock))
|
|
|
|
rot_kwargs = {
|
|
|
|
"axis" : OUT,
|
|
|
|
"about_point" : clock.get_center()
|
|
|
|
}
|
|
|
|
hour_radians = -self.hours_passed*2*np.pi/12
|
|
|
|
self.hour_rotation = Rotating(
|
|
|
|
clock.hour_hand,
|
|
|
|
radians = hour_radians,
|
|
|
|
**rot_kwargs
|
|
|
|
)
|
|
|
|
self.minute_rotation = Rotating(
|
|
|
|
clock.minute_hand,
|
|
|
|
radians = 12*hour_radians,
|
|
|
|
**rot_kwargs
|
|
|
|
)
|
|
|
|
Animation.__init__(self, clock, **kwargs)
|
|
|
|
|
|
|
|
def update_mobject(self, alpha):
|
|
|
|
for rotation in self.hour_rotation, self.minute_rotation:
|
|
|
|
rotation.update_mobject(alpha)
|
2016-11-07 11:05:41 -08:00
|
|
|
|
|
|
|
class Bubble(SVGMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"direction" : LEFT,
|
|
|
|
"center_point" : ORIGIN,
|
|
|
|
"content_scale_factor" : 0.75,
|
|
|
|
"height" : 5,
|
|
|
|
"width" : 8,
|
|
|
|
"bubble_center_adjustment_factor" : 1./8,
|
|
|
|
"file_name" : None,
|
2018-01-15 18:16:50 -08:00
|
|
|
"propagate_style_to_family" : True,
|
2017-01-05 11:56:52 -08:00
|
|
|
"fill_color" : BLACK,
|
|
|
|
"fill_opacity" : 0.8,
|
2017-07-06 12:06:11 -07:00
|
|
|
"stroke_color" : WHITE,
|
|
|
|
"stroke_width" : 3,
|
2016-11-07 11:05:41 -08:00
|
|
|
}
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
digest_config(self, kwargs, locals())
|
|
|
|
if self.file_name is None:
|
|
|
|
raise Exception("Must invoke Bubble subclass")
|
2018-01-12 13:38:25 -08:00
|
|
|
try:
|
|
|
|
SVGMobject.__init__(self, **kwargs)
|
|
|
|
except IOError as err:
|
|
|
|
self.file_name = os.path.join(FILE_DIR, self.file_name)
|
|
|
|
SVGMobject.__init__(self, **kwargs)
|
2016-11-07 11:05:41 -08:00
|
|
|
self.center()
|
|
|
|
self.stretch_to_fit_height(self.height)
|
|
|
|
self.stretch_to_fit_width(self.width)
|
|
|
|
if self.direction[0] > 0:
|
|
|
|
Mobject.flip(self)
|
|
|
|
self.direction_was_specified = ("direction" in kwargs)
|
|
|
|
self.content = Mobject()
|
|
|
|
|
|
|
|
def get_tip(self):
|
|
|
|
#TODO, find a better way
|
|
|
|
return self.get_corner(DOWN+self.direction)-0.6*self.direction
|
|
|
|
|
|
|
|
def get_bubble_center(self):
|
|
|
|
factor = self.bubble_center_adjustment_factor
|
|
|
|
return self.get_center() + factor*self.get_height()*UP
|
|
|
|
|
|
|
|
def move_tip_to(self, point):
|
2016-11-18 21:23:52 -08:00
|
|
|
VGroup(self, self.content).shift(point - self.get_tip())
|
2016-11-07 11:05:41 -08:00
|
|
|
return self
|
|
|
|
|
|
|
|
def flip(self):
|
|
|
|
Mobject.flip(self)
|
|
|
|
self.direction = -np.array(self.direction)
|
|
|
|
return self
|
|
|
|
|
|
|
|
def pin_to(self, mobject):
|
|
|
|
mob_center = mobject.get_center()
|
|
|
|
want_to_filp = np.sign(mob_center[0]) != np.sign(self.direction[0])
|
|
|
|
can_flip = not self.direction_was_specified
|
|
|
|
if want_to_filp and can_flip:
|
|
|
|
self.flip()
|
|
|
|
boundary_point = mobject.get_critical_point(UP-self.direction)
|
|
|
|
vector_from_center = 1.0*(boundary_point-mob_center)
|
|
|
|
self.move_tip_to(mob_center+vector_from_center)
|
|
|
|
return self
|
|
|
|
|
|
|
|
def position_mobject_inside(self, mobject):
|
|
|
|
scaled_width = self.content_scale_factor*self.get_width()
|
|
|
|
if mobject.get_width() > scaled_width:
|
|
|
|
mobject.scale_to_fit_width(scaled_width)
|
|
|
|
mobject.shift(
|
|
|
|
self.get_bubble_center() - mobject.get_center()
|
|
|
|
)
|
|
|
|
return mobject
|
|
|
|
|
|
|
|
def add_content(self, mobject):
|
|
|
|
self.position_mobject_inside(mobject)
|
|
|
|
self.content = mobject
|
|
|
|
return self.content
|
|
|
|
|
|
|
|
def write(self, *text):
|
|
|
|
self.add_content(TextMobject(*text))
|
|
|
|
return self
|
|
|
|
|
|
|
|
def resize_to_content(self):
|
|
|
|
target_width = self.content.get_width()
|
2017-01-25 16:40:59 -08:00
|
|
|
target_width += max(MED_LARGE_BUFF, 2)
|
2016-11-07 11:05:41 -08:00
|
|
|
target_height = self.content.get_height()
|
|
|
|
target_height += 2.5*LARGE_BUFF
|
|
|
|
tip_point = self.get_tip()
|
|
|
|
self.stretch_to_fit_width(target_width)
|
|
|
|
self.stretch_to_fit_height(target_height)
|
|
|
|
self.move_tip_to(tip_point)
|
|
|
|
self.position_mobject_inside(self.content)
|
|
|
|
|
|
|
|
def clear(self):
|
|
|
|
self.add_content(VMobject())
|
|
|
|
return self
|
|
|
|
|
|
|
|
class SpeechBubble(Bubble):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "Bubbles_speech.svg",
|
|
|
|
"height" : 4
|
|
|
|
}
|
|
|
|
|
|
|
|
class DoubleSpeechBubble(Bubble):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "Bubbles_double_speech.svg",
|
|
|
|
"height" : 4
|
|
|
|
}
|
|
|
|
|
|
|
|
class ThoughtBubble(Bubble):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "Bubbles_thought.svg",
|
|
|
|
}
|
|
|
|
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
Bubble.__init__(self, **kwargs)
|
|
|
|
self.submobjects.sort(
|
|
|
|
lambda m1, m2 : int((m1.get_bottom()-m2.get_bottom())[1])
|
|
|
|
)
|
|
|
|
|
|
|
|
def make_green_screen(self):
|
|
|
|
self.submobjects[-1].set_fill(GREEN_SCREEN, opacity = 1)
|
|
|
|
return self
|
2018-01-11 18:13:35 -08:00
|
|
|
|
2018-02-13 21:38:10 -08:00
|
|
|
class Car(SVGMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"file_name" : "Car",
|
|
|
|
"height" : 1,
|
|
|
|
"color" : LIGHT_GREY,
|
|
|
|
"light_colors" : [BLACK, BLACK],
|
|
|
|
}
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
SVGMobject.__init__(self, **kwargs)
|
|
|
|
self.scale_to_fit_height(self.height)
|
|
|
|
self.set_stroke(color = WHITE, width = 0)
|
|
|
|
self.set_fill(self.color, opacity = 1)
|
|
|
|
|
|
|
|
from topics.characters import Randolph
|
|
|
|
randy = Randolph(mode = "happy")
|
|
|
|
randy.scale_to_fit_height(0.6*self.get_height())
|
|
|
|
randy.stretch(0.8, 0)
|
|
|
|
randy.look(RIGHT)
|
|
|
|
randy.move_to(self)
|
|
|
|
randy.shift(0.07*self.height*(RIGHT+UP))
|
|
|
|
self.randy = self.pi_creature = randy
|
|
|
|
self.add_to_back(randy)
|
|
|
|
|
|
|
|
orientation_line = Line(self.get_left(), self.get_right())
|
|
|
|
orientation_line.set_stroke(width = 0)
|
|
|
|
self.add(orientation_line)
|
|
|
|
self.orientation_line = orientation_line
|
|
|
|
|
|
|
|
for light, color in zip(self.get_lights(), self.light_colors):
|
|
|
|
light.set_fill(color, 1)
|
|
|
|
light.is_subpath = False
|
|
|
|
|
|
|
|
self.add_treds_to_tires()
|
|
|
|
|
|
|
|
def move_to(self, point_or_mobject):
|
|
|
|
vect = rotate_vector(
|
|
|
|
UP+LEFT, self.orientation_line.get_angle()
|
|
|
|
)
|
|
|
|
self.next_to(point_or_mobject, vect, buff = 0)
|
|
|
|
return self
|
|
|
|
|
|
|
|
def get_front_line(self):
|
|
|
|
return DashedLine(
|
|
|
|
self.get_corner(UP+RIGHT),
|
|
|
|
self.get_corner(DOWN+RIGHT),
|
|
|
|
color = DISTANCE_COLOR,
|
|
|
|
dashed_segment_length = 0.05,
|
|
|
|
)
|
|
|
|
|
|
|
|
def add_treds_to_tires(self):
|
|
|
|
for tire in self.get_tires():
|
|
|
|
radius = tire.get_width()/2
|
|
|
|
center = tire.get_center()
|
|
|
|
tred = Line(
|
|
|
|
0.9*radius*RIGHT, 1.4*radius*RIGHT,
|
|
|
|
stroke_width = 2,
|
|
|
|
color = BLACK
|
|
|
|
)
|
|
|
|
tred.rotate_in_place(np.pi/4)
|
|
|
|
for theta in np.arange(0, 2*np.pi, np.pi/4):
|
|
|
|
new_tred = tred.copy()
|
|
|
|
new_tred.rotate(theta, about_point = ORIGIN)
|
|
|
|
new_tred.shift(center)
|
|
|
|
tire.add(new_tred)
|
|
|
|
return self
|
|
|
|
|
|
|
|
def get_tires(self):
|
|
|
|
return VGroup(self[1][1], self[1][3])
|
|
|
|
|
|
|
|
def get_lights(self):
|
|
|
|
return VGroup(self.get_front_light(), self.get_rear_light())
|
|
|
|
|
|
|
|
def get_front_light(self):
|
|
|
|
return self[1][5]
|
|
|
|
|
|
|
|
def get_rear_light(self):
|
|
|
|
return self[1][8]
|
|
|
|
|
|
|
|
class MoveCar(ApplyMethod):
|
|
|
|
CONFIG = {
|
|
|
|
"moving_forward" : True,
|
|
|
|
}
|
|
|
|
def __init__(self, car, target_point, **kwargs):
|
|
|
|
ApplyMethod.__init__(self, car.move_to, target_point, **kwargs)
|
|
|
|
displacement = self.target_mobject.get_right()-self.starting_mobject.get_right()
|
|
|
|
distance = np.linalg.norm(displacement)
|
|
|
|
if not self.moving_forward:
|
|
|
|
distance *= -1
|
|
|
|
tire_radius = car.get_tires()[0].get_width()/2
|
|
|
|
self.total_tire_radians = -distance/tire_radius
|
|
|
|
|
|
|
|
def update_mobject(self, alpha):
|
|
|
|
ApplyMethod.update_mobject(self, alpha)
|
|
|
|
if alpha == 0:
|
|
|
|
return
|
|
|
|
radians = alpha*self.total_tire_radians
|
|
|
|
for tire in self.mobject.get_tires():
|
|
|
|
tire.rotate_in_place(radians)
|
2018-01-11 18:13:35 -08:00
|
|
|
|
|
|
|
#TODO: Where should this live?
|
|
|
|
class Broadcast(LaggedStart):
|
|
|
|
CONFIG = {
|
|
|
|
"small_radius" : 0.0,
|
|
|
|
"big_radius" : 5,
|
|
|
|
"n_circles" : 5,
|
2018-01-21 23:32:25 -08:00
|
|
|
"start_stroke_width" : 8,
|
|
|
|
"color" : WHITE,
|
2018-01-11 18:13:35 -08:00
|
|
|
"remover" : True,
|
|
|
|
"lag_ratio" : 0.7,
|
|
|
|
"run_time" : 3,
|
|
|
|
"remover" : True,
|
|
|
|
}
|
|
|
|
def __init__(self, focal_point, **kwargs):
|
|
|
|
digest_config(self, kwargs)
|
|
|
|
circles = VGroup()
|
|
|
|
for x in range(self.n_circles):
|
|
|
|
circle = Circle(
|
|
|
|
radius = self.big_radius,
|
|
|
|
stroke_color = BLACK,
|
|
|
|
stroke_width = 0,
|
|
|
|
)
|
|
|
|
circle.move_to(focal_point)
|
|
|
|
circle.save_state()
|
|
|
|
circle.scale_to_fit_width(self.small_radius*2)
|
2018-01-21 23:32:25 -08:00
|
|
|
circle.set_stroke(self.color, self.start_stroke_width)
|
2018-01-11 18:13:35 -08:00
|
|
|
circles.add(circle)
|
|
|
|
LaggedStart.__init__(
|
|
|
|
self, ApplyMethod, circles,
|
|
|
|
lambda c : (c.restore,),
|
|
|
|
**kwargs
|
2018-01-12 13:38:25 -08:00
|
|
|
|
|
|
|
)
|
|
|
|
|
2018-01-28 16:53:17 +01:00
|
|
|
class BraceLabel(VMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"label_constructor" : TexMobject,
|
|
|
|
"label_scale" : 1,
|
|
|
|
}
|
|
|
|
def __init__(self, obj, text, brace_direction = DOWN, **kwargs):
|
|
|
|
VMobject.__init__(self, **kwargs)
|
|
|
|
self.brace_direction = brace_direction
|
|
|
|
if isinstance(obj, list): obj = VMobject(*obj)
|
|
|
|
self.brace = Brace(obj, brace_direction, **kwargs)
|
|
|
|
|
|
|
|
if isinstance(text, tuple) or isinstance(text, list):
|
|
|
|
self.label = self.label_constructor(*text, **kwargs)
|
|
|
|
else: self.label = self.label_constructor(str(text))
|
|
|
|
if self.label_scale != 1: self.label.scale(self.label_scale)
|
|
|
|
|
|
|
|
self.brace.put_at_tip(self.label)
|
|
|
|
self.submobjects = [self.brace, self.label]
|
|
|
|
|
|
|
|
def creation_anim(self, label_anim = FadeIn, brace_anim = GrowFromCenter):
|
|
|
|
return AnimationGroup(brace_anim(self.brace), label_anim(self.label))
|
|
|
|
|
|
|
|
def shift_brace(self, obj, **kwargs):
|
|
|
|
if isinstance(obj, list): obj = VMobject(*obj)
|
|
|
|
self.brace = Brace(obj, self.brace_direction, **kwargs)
|
|
|
|
self.brace.put_at_tip(self.label)
|
|
|
|
self.submobjects[0] = self.brace
|
|
|
|
return self
|
2018-01-12 13:38:25 -08:00
|
|
|
|
2018-01-28 16:53:17 +01:00
|
|
|
def change_label(self, *text, **kwargs):
|
|
|
|
self.label = self.label_constructor(*text, **kwargs)
|
|
|
|
if self.label_scale != 1: self.label.scale(self.label_scale)
|
2018-01-12 13:38:25 -08:00
|
|
|
|
2018-01-28 16:53:17 +01:00
|
|
|
self.brace.put_at_tip(self.label)
|
|
|
|
self.submobjects[1] = self.label
|
|
|
|
return self
|
2018-01-12 13:38:25 -08:00
|
|
|
|
2018-01-28 16:53:17 +01:00
|
|
|
def change_brace_label(self, obj, *text):
|
|
|
|
self.shift_brace(obj)
|
|
|
|
self.change_label(*text)
|
|
|
|
return self
|
2018-01-12 13:38:25 -08:00
|
|
|
|
2018-01-28 16:53:17 +01:00
|
|
|
def copy(self):
|
|
|
|
copy_mobject = copy.copy(self)
|
|
|
|
copy_mobject.brace = self.brace.copy()
|
|
|
|
copy_mobject.label = self.label.copy()
|
|
|
|
copy_mobject.submobjects = [copy_mobject.brace, copy_mobject.label]
|
2018-01-12 13:38:25 -08:00
|
|
|
|
2018-01-28 16:53:17 +01:00
|
|
|
return copy_mobject
|
2018-01-12 13:38:25 -08:00
|
|
|
|
2018-01-28 16:53:17 +01:00
|
|
|
class BraceText(BraceLabel):
|
|
|
|
CONFIG = {
|
|
|
|
"label_constructor" : TextMobject
|
|
|
|
}
|
2018-01-28 20:31:55 +01:00
|
|
|
|
|
|
|
class DashedMobject(VMobject):
|
|
|
|
CONFIG = {
|
|
|
|
"dashes_num" : 15,
|
|
|
|
"spacing" : 0.5,
|
|
|
|
"color" : WHITE
|
|
|
|
}
|
|
|
|
def __init__(self, mob, **kwargs):
|
|
|
|
digest_locals(self)
|
|
|
|
VMobject.__init__(self, **kwargs)
|
|
|
|
|
|
|
|
buff = float(self.spacing) / self.dashes_num
|
|
|
|
|
|
|
|
for i in range(self.dashes_num):
|
|
|
|
a = ((1+buff) * i)/self.dashes_num
|
|
|
|
b = 1-((1+buff) * (self.dashes_num-1-i)) / self.dashes_num
|
|
|
|
dash = VMobject(color = self.color)
|
|
|
|
dash.pointwise_become_partial(mob, a, b)
|
|
|
|
self.submobjects.append(dash)
|