mirror of
https://github.com/3b1b/manim.git
synced 2025-08-01 17:29:06 +08:00
Merge branch 'master' of https://github.com/3b1b/manim into WindingNumber
This commit is contained in:
@ -1,4 +1,4 @@
|
|||||||
#!/usr/bin/env python
|
#!/usr/bin/env python2
|
||||||
|
|
||||||
import sys
|
import sys
|
||||||
# import getopt
|
# import getopt
|
||||||
@ -85,6 +85,7 @@ def get_configuration():
|
|||||||
#If -t is passed in (for transparent), this will be RGBA
|
#If -t is passed in (for transparent), this will be RGBA
|
||||||
"saved_image_mode": "RGBA" if args.transparent else "RGB",
|
"saved_image_mode": "RGBA" if args.transparent else "RGB",
|
||||||
"quiet" : args.quiet or args.write_all,
|
"quiet" : args.quiet or args.write_all,
|
||||||
|
"ignore_waits" : args.preview,
|
||||||
"write_all" : args.write_all,
|
"write_all" : args.write_all,
|
||||||
"output_name" : args.output_name,
|
"output_name" : args.output_name,
|
||||||
"skip_to_animation_number" : args.skip_to_animation_number,
|
"skip_to_animation_number" : args.skip_to_animation_number,
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
from xml.dom import minidom
|
from xml.dom import minidom
|
||||||
import warnings
|
import warnings
|
||||||
|
|
||||||
from vectorized_mobject import VMobject
|
from vectorized_mobject import VMobject, VGroup
|
||||||
from topics.geometry import Rectangle, Circle
|
from topics.geometry import Rectangle, Circle
|
||||||
from helpers import *
|
from helpers import *
|
||||||
|
|
||||||
@ -21,6 +21,7 @@ class SVGMobject(VMobject):
|
|||||||
"width" : None,
|
"width" : None,
|
||||||
#Must be filled in in a subclass, or when called
|
#Must be filled in in a subclass, or when called
|
||||||
"file_name" : None,
|
"file_name" : None,
|
||||||
|
"unpack_groups" : True, # if False, creates a hierarchy of VGroups
|
||||||
"stroke_width" : 0,
|
"stroke_width" : 0,
|
||||||
"fill_opacity" : 1,
|
"fill_opacity" : 1,
|
||||||
# "fill_color" : LIGHT_GREY,
|
# "fill_color" : LIGHT_GREY,
|
||||||
@ -50,7 +51,9 @@ class SVGMobject(VMobject):
|
|||||||
doc = minidom.parse(self.file_path)
|
doc = minidom.parse(self.file_path)
|
||||||
self.ref_to_element = {}
|
self.ref_to_element = {}
|
||||||
for svg in doc.getElementsByTagName("svg"):
|
for svg in doc.getElementsByTagName("svg"):
|
||||||
self.add(*self.get_mobjects_from(svg))
|
mobjects = self.get_mobjects_from(svg)
|
||||||
|
if self.unpack_groups: self.add(*mobjects)
|
||||||
|
else: self.add(*mobjects[0].submobjects)
|
||||||
doc.unlink()
|
doc.unlink()
|
||||||
|
|
||||||
def get_mobjects_from(self, element):
|
def get_mobjects_from(self, element):
|
||||||
@ -76,6 +79,8 @@ class SVGMobject(VMobject):
|
|||||||
result.append(self.rect_to_mobject(element))
|
result.append(self.rect_to_mobject(element))
|
||||||
elif element.tagName == 'circle':
|
elif element.tagName == 'circle':
|
||||||
result.append(self.circle_to_mobject(element))
|
result.append(self.circle_to_mobject(element))
|
||||||
|
elif element.tagName == 'ellipse':
|
||||||
|
result.append(self.ellipse_to_mobject(element))
|
||||||
elif element.tagName in ['polygon', 'polyline']:
|
elif element.tagName in ['polygon', 'polyline']:
|
||||||
result.append(self.polygon_to_mobject(element))
|
result.append(self.polygon_to_mobject(element))
|
||||||
else:
|
else:
|
||||||
@ -83,6 +88,9 @@ class SVGMobject(VMobject):
|
|||||||
# warnings.warn("Unknown element type: " + element.tagName)
|
# warnings.warn("Unknown element type: " + element.tagName)
|
||||||
result = filter(lambda m : m is not None, result)
|
result = filter(lambda m : m is not None, result)
|
||||||
self.handle_transforms(element, VMobject(*result))
|
self.handle_transforms(element, VMobject(*result))
|
||||||
|
if len(result) > 1 and not self.unpack_groups:
|
||||||
|
result = [VGroup(*result)]
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
def g_to_mobjects(self, g_element):
|
def g_to_mobjects(self, g_element):
|
||||||
@ -122,6 +130,15 @@ class SVGMobject(VMobject):
|
|||||||
]
|
]
|
||||||
return Circle(radius = r).shift(x*RIGHT+y*DOWN)
|
return Circle(radius = r).shift(x*RIGHT+y*DOWN)
|
||||||
|
|
||||||
|
def ellipse_to_mobject(self, circle_element):
|
||||||
|
x, y, rx, ry = [
|
||||||
|
float(circle_element.getAttribute(key))
|
||||||
|
if circle_element.hasAttribute(key)
|
||||||
|
else 0.0
|
||||||
|
for key in "cx", "cy", "rx", "ry"
|
||||||
|
]
|
||||||
|
return Circle().scale(rx*RIGHT + ry*UP).shift(x*RIGHT+y*DOWN)
|
||||||
|
|
||||||
def rect_to_mobject(self, rect_element):
|
def rect_to_mobject(self, rect_element):
|
||||||
if rect_element.hasAttribute("fill"):
|
if rect_element.hasAttribute("fill"):
|
||||||
if Color(str(rect_element.getAttribute("fill"))) == Color(WHITE):
|
if Color(str(rect_element.getAttribute("fill"))) == Color(WHITE):
|
||||||
@ -154,7 +171,7 @@ class SVGMobject(VMobject):
|
|||||||
transform = string_to_numbers(transform)
|
transform = string_to_numbers(transform)
|
||||||
transform = np.array(transform).reshape([3,2])
|
transform = np.array(transform).reshape([3,2])
|
||||||
x += transform[2][0]
|
x += transform[2][0]
|
||||||
y += transform[2][1]
|
y -= transform[2][1]
|
||||||
matrix = np.identity(self.dim)
|
matrix = np.identity(self.dim)
|
||||||
matrix[:2,:2] = transform[:2,:]
|
matrix[:2,:2] = transform[:2,:]
|
||||||
t_matrix = np.transpose(matrix)
|
t_matrix = np.transpose(matrix)
|
||||||
@ -227,16 +244,25 @@ class VMobjectFromSVGPathstring(VMobject):
|
|||||||
#list. This variable may get modified in the conditionals below.
|
#list. This variable may get modified in the conditionals below.
|
||||||
points = self.growing_path.points
|
points = self.growing_path.points
|
||||||
new_points = self.string_to_points(coord_string)
|
new_points = self.string_to_points(coord_string)
|
||||||
|
|
||||||
|
if command == "M": #moveto
|
||||||
|
if isLower and len(points) > 0:
|
||||||
|
new_points[0] += points[-1]
|
||||||
|
if len(points) > 0:
|
||||||
|
self.growing_path = self.add_subpath(new_points[:1])
|
||||||
|
else:
|
||||||
|
self.growing_path.start_at(new_points[0])
|
||||||
|
|
||||||
|
if len(new_points) <= 1: return
|
||||||
|
|
||||||
|
points = self.growing_path.points
|
||||||
|
new_points = new_points[1:]
|
||||||
|
command = "L"
|
||||||
|
|
||||||
if isLower and len(points) > 0:
|
if isLower and len(points) > 0:
|
||||||
new_points += points[-1]
|
new_points += points[-1]
|
||||||
if command == "M": #moveto
|
|
||||||
if len(points) > 0:
|
if command in ["L", "H", "V"]: #lineto
|
||||||
self.growing_path = self.add_subpath(new_points)
|
|
||||||
else:
|
|
||||||
if isLower: self.growing_path.start_at(np.sum(new_points, axis=0))
|
|
||||||
else: self.growing_path.start_at(new_points[-1])
|
|
||||||
return
|
|
||||||
elif command in ["L", "H", "V"]: #lineto
|
|
||||||
if command == "H":
|
if command == "H":
|
||||||
new_points[0,1] = points[-1,1]
|
new_points[0,1] = points[-1,1]
|
||||||
elif command == "V":
|
elif command == "V":
|
||||||
|
@ -414,6 +414,9 @@ class VMobject(Mobject):
|
|||||||
b_residue = (num_cubics*b)%1
|
b_residue = (num_cubics*b)%1
|
||||||
if b == 1:
|
if b == 1:
|
||||||
b_residue = 1
|
b_residue = 1
|
||||||
|
elif lower_index == upper_index:
|
||||||
|
b_residue = (b_residue - a_residue)/(1-a_residue)
|
||||||
|
|
||||||
points[:4] = partial_bezier_points(
|
points[:4] = partial_bezier_points(
|
||||||
points[:4], a_residue, 1
|
points[:4], a_residue, 1
|
||||||
)
|
)
|
||||||
@ -424,8 +427,18 @@ class VMobject(Mobject):
|
|||||||
return self
|
return self
|
||||||
|
|
||||||
class VGroup(VMobject):
|
class VGroup(VMobject):
|
||||||
#Alternate name to improve readability during use
|
def __init__(self, *args, **kwargs):
|
||||||
pass
|
if len(args) == 1 and isinstance(args[0], (tuple, list)):
|
||||||
|
args = args[0]
|
||||||
|
|
||||||
|
packed_args = []
|
||||||
|
for arg in args:
|
||||||
|
if isinstance(arg, (tuple, list)):
|
||||||
|
packed_args.append(VGroup(arg))
|
||||||
|
else: packed_args.append(arg)
|
||||||
|
|
||||||
|
VMobject.__init__(self, *packed_args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
class VectorizedPoint(VMobject):
|
class VectorizedPoint(VMobject):
|
||||||
CONFIG = {
|
CONFIG = {
|
||||||
|
@ -29,6 +29,12 @@ IMAGE_MAP_DATA_FILE = os.path.join(NN_DIRECTORY, "image_map")
|
|||||||
# DEFAULT_LAYER_SIZES = [28**2, 80, 10]
|
# DEFAULT_LAYER_SIZES = [28**2, 80, 10]
|
||||||
DEFAULT_LAYER_SIZES = [28**2, 16, 16, 10]
|
DEFAULT_LAYER_SIZES = [28**2, 16, 16, 10]
|
||||||
|
|
||||||
|
try:
|
||||||
|
xrange # Python 2
|
||||||
|
except NameError:
|
||||||
|
xrange = range # Python 3
|
||||||
|
|
||||||
|
|
||||||
class Network(object):
|
class Network(object):
|
||||||
def __init__(self, sizes, non_linearity = "sigmoid"):
|
def __init__(self, sizes, non_linearity = "sigmoid"):
|
||||||
"""The list ``sizes`` contains the number of neurons in the
|
"""The list ``sizes`` contains the number of neurons in the
|
||||||
|
@ -1,6 +1,10 @@
|
|||||||
import numpy as np
|
import numpy as np
|
||||||
from animation.transform import Transform
|
from animation.transform import ApplyMethod, Transform
|
||||||
|
from constants import RIGHT, SPACE_WIDTH, UP
|
||||||
|
from helpers import counterclockwise_path, straight_path
|
||||||
from point_cloud_mobject import Point
|
from point_cloud_mobject import Point
|
||||||
|
from scene import Scene
|
||||||
|
from topics.geometry import Line
|
||||||
from topics.number_line import NumberLine
|
from topics.number_line import NumberLine
|
||||||
|
|
||||||
class NumberLineScene(Scene):
|
class NumberLineScene(Scene):
|
||||||
@ -78,11 +82,3 @@ class NumberLineScene(Scene):
|
|||||||
ApplyMethod(mob.shift, (num-1)*mob.get_center()[0]*RIGHT, **kwargs)
|
ApplyMethod(mob.shift, (num-1)*mob.get_center()[0]*RIGHT, **kwargs)
|
||||||
for mob in self.number_mobs
|
for mob in self.number_mobs
|
||||||
])
|
])
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -29,6 +29,7 @@ class Scene(Container):
|
|||||||
"frame_duration" : LOW_QUALITY_FRAME_DURATION,
|
"frame_duration" : LOW_QUALITY_FRAME_DURATION,
|
||||||
"construct_args" : [],
|
"construct_args" : [],
|
||||||
"skip_animations" : False,
|
"skip_animations" : False,
|
||||||
|
"ignore_waits" : False,
|
||||||
"write_to_movie" : False,
|
"write_to_movie" : False,
|
||||||
"save_frames" : False,
|
"save_frames" : False,
|
||||||
"save_pngs" : False,
|
"save_pngs" : False,
|
||||||
@ -50,6 +51,7 @@ class Scene(Container):
|
|||||||
self.saved_frames = []
|
self.saved_frames = []
|
||||||
self.shared_locals = {}
|
self.shared_locals = {}
|
||||||
self.frame_num = 0
|
self.frame_num = 0
|
||||||
|
self.current_scene_time = 0
|
||||||
if self.name is None:
|
if self.name is None:
|
||||||
self.name = self.__class__.__name__
|
self.name = self.__class__.__name__
|
||||||
if self.random_seed is not None:
|
if self.random_seed is not None:
|
||||||
@ -455,6 +457,17 @@ class Scene(Container):
|
|||||||
|
|
||||||
return self
|
return self
|
||||||
|
|
||||||
|
def wait_to(self, time, assert_positive = True):
|
||||||
|
if self.ignore_waits:
|
||||||
|
return
|
||||||
|
time -= self.current_scene_time
|
||||||
|
if assert_positive:
|
||||||
|
assert(time >= 0)
|
||||||
|
elif time < 0:
|
||||||
|
return
|
||||||
|
|
||||||
|
self.wait(time)
|
||||||
|
|
||||||
def force_skipping(self):
|
def force_skipping(self):
|
||||||
self.original_skipping_status = self.skip_animations
|
self.original_skipping_status = self.skip_animations
|
||||||
self.skip_animations = True
|
self.skip_animations = True
|
||||||
@ -466,6 +479,7 @@ class Scene(Container):
|
|||||||
return self
|
return self
|
||||||
|
|
||||||
def add_frames(self, *frames):
|
def add_frames(self, *frames):
|
||||||
|
self.current_scene_time += len(frames)*self.frame_duration
|
||||||
if self.write_to_movie:
|
if self.write_to_movie:
|
||||||
for frame in frames:
|
for frame in frames:
|
||||||
if self.save_pngs:
|
if self.save_pngs:
|
||||||
@ -551,7 +565,3 @@ class Scene(Container):
|
|||||||
shutil.move(*self.args_to_rename_file)
|
shutil.move(*self.args_to_rename_file)
|
||||||
else:
|
else:
|
||||||
os.rename(*self.args_to_rename_file)
|
os.rename(*self.args_to_rename_file)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -7,7 +7,7 @@ from mobject.tex_mobject import TextMobject, TexMobject
|
|||||||
|
|
||||||
from animation import Animation
|
from animation import Animation
|
||||||
from animation.simple_animations import Rotating, LaggedStart
|
from animation.simple_animations import Rotating, LaggedStart
|
||||||
from animation.transform import ApplyMethod
|
from animation.transform import ApplyMethod, FadeIn, GrowFromCenter
|
||||||
|
|
||||||
from topics.geometry import Circle, Line, Rectangle, Square, \
|
from topics.geometry import Circle, Line, Rectangle, Square, \
|
||||||
Arc, Polygon, SurroundingRectangle
|
Arc, Polygon, SurroundingRectangle
|
||||||
@ -521,19 +521,76 @@ class Broadcast(LaggedStart):
|
|||||||
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
def change_label(self, *text, **kwargs):
|
||||||
|
self.label = self.label_constructor(*text, **kwargs)
|
||||||
|
if self.label_scale != 1: self.label.scale(self.label_scale)
|
||||||
|
|
||||||
|
self.brace.put_at_tip(self.label)
|
||||||
|
self.submobjects[1] = self.label
|
||||||
|
return self
|
||||||
|
|
||||||
|
def change_brace_label(self, obj, *text):
|
||||||
|
self.shift_brace(obj)
|
||||||
|
self.change_label(*text)
|
||||||
|
return self
|
||||||
|
|
||||||
|
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]
|
||||||
|
|
||||||
|
return copy_mobject
|
||||||
|
|
||||||
|
class BraceText(BraceLabel):
|
||||||
|
CONFIG = {
|
||||||
|
"label_constructor" : TextMobject
|
||||||
|
}
|
||||||
|
|
||||||
|
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)
|
||||||
|
@ -27,10 +27,10 @@ class CameraWithPerspective(Camera):
|
|||||||
class ThreeDCamera(CameraWithPerspective):
|
class ThreeDCamera(CameraWithPerspective):
|
||||||
CONFIG = {
|
CONFIG = {
|
||||||
"sun_vect" : 5*UP+LEFT,
|
"sun_vect" : 5*UP+LEFT,
|
||||||
"shading_factor" : 0.5,
|
"shading_factor" : 0.2,
|
||||||
"distance" : 5,
|
"distance" : 5,
|
||||||
"phi" : 0, #Angle off z axis
|
"phi" : 0, #Angle off z axis
|
||||||
"theta" : -np.pi/2, #Rotation about z axis
|
"theta" : -TAU/4, #Rotation about z axis
|
||||||
}
|
}
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
Camera.__init__(self, *args, **kwargs)
|
Camera.__init__(self, *args, **kwargs)
|
||||||
@ -78,19 +78,26 @@ class ThreeDCamera(CameraWithPerspective):
|
|||||||
return normal/length
|
return normal/length
|
||||||
|
|
||||||
def display_multiple_vectorized_mobjects(self, vmobjects):
|
def display_multiple_vectorized_mobjects(self, vmobjects):
|
||||||
|
camera_point = self.spherical_coords_to_point(
|
||||||
|
*self.get_spherical_coords()
|
||||||
|
)
|
||||||
def z_cmp(*vmobs):
|
def z_cmp(*vmobs):
|
||||||
#Compare to three dimensional mobjects based on their
|
#Compare to three dimensional mobjects based on
|
||||||
#z value, otherwise don't compare.
|
#how close they are to the camera
|
||||||
three_d_status = map(should_shade_in_3d, vmobs)
|
|
||||||
has_points = [vm.get_num_points() > 0 for vm in vmobs]
|
|
||||||
if all(three_d_status) and all(has_points):
|
|
||||||
cmp_vect = self.get_unit_normal_vect(vmobs[1])
|
|
||||||
return cmp(*[
|
return cmp(*[
|
||||||
np.dot(vm.get_center(), cmp_vect)
|
-np.linalg.norm(vm.get_center()-camera_point)
|
||||||
for vm in vmobs
|
for vm in vmobs
|
||||||
])
|
])
|
||||||
else:
|
# three_d_status = map(should_shade_in_3d, vmobs)
|
||||||
return 0
|
# has_points = [vm.get_num_points() > 0 for vm in vmobs]
|
||||||
|
# if all(three_d_status) and all(has_points):
|
||||||
|
# cmp_vect = self.get_unit_normal_vect(vmobs[1])
|
||||||
|
# return cmp(*[
|
||||||
|
# np.dot(vm.get_center(), cmp_vect)
|
||||||
|
# for vm in vmobs
|
||||||
|
# ])
|
||||||
|
# else:
|
||||||
|
# return 0
|
||||||
Camera.display_multiple_vectorized_mobjects(
|
Camera.display_multiple_vectorized_mobjects(
|
||||||
self, sorted(vmobjects, cmp = z_cmp)
|
self, sorted(vmobjects, cmp = z_cmp)
|
||||||
)
|
)
|
||||||
@ -176,9 +183,10 @@ class ThreeDScene(Scene):
|
|||||||
self.add(self.ambient_camera_rotation)
|
self.add(self.ambient_camera_rotation)
|
||||||
|
|
||||||
def get_moving_mobjects(self, *animations):
|
def get_moving_mobjects(self, *animations):
|
||||||
if self.camera.rotation_mobject in moving:
|
moving_mobjects = Scene.get_moving_mobjects(self, *animations)
|
||||||
return self.mobjects
|
if self.camera.rotation_mobject in moving_mobjects:
|
||||||
return Scene.get_moving_mobjects(self, *animations)
|
return list_update(self.mobjects, moving_mobjects)
|
||||||
|
return moving_mobjects
|
||||||
|
|
||||||
##############
|
##############
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user