| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 12:56:29 -07:00
										 |  |  | from big_ol_pile_of_manim_imports import * | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | import mpmath | 
					
						
							|  |  |  | mpmath.mp.dps = 7 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def zeta(z): | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |     max_norm = FRAME_X_RADIUS | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |     try: | 
					
						
							|  |  |  |         return np.complex(mpmath.zeta(z)) | 
					
						
							|  |  |  |     except: | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         return np.complex(max_norm, 0) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  | def d_zeta(z): | 
					
						
							|  |  |  |     epsilon = 0.01 | 
					
						
							|  |  |  |     return (zeta(z + epsilon) - zeta(z))/epsilon | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | class ZetaTransformationScene(ComplexTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         "anchor_density" : 35, | 
					
						
							|  |  |  |         "min_added_anchors" : 10, | 
					
						
							|  |  |  |         "max_added_anchors" : 300, | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         "num_anchors_to_add_per_line" : 75, | 
					
						
							| 
									
										
										
										
											2017-01-11 12:04:06 -08:00
										 |  |  |         "post_transformation_stroke_width" : 2, | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         "default_apply_complex_function_kwargs" : { | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |             "run_time" : 5, | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         }, | 
					
						
							|  |  |  |         "x_min" : 1, | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |         "x_max" : int(FRAME_X_RADIUS+2), | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         "extra_lines_x_min" : -2, | 
					
						
							|  |  |  |         "extra_lines_x_max" : 4, | 
					
						
							|  |  |  |         "extra_lines_y_min" : -2, | 
					
						
							|  |  |  |         "extra_lines_y_max" : 2, | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |     def prepare_for_transformation(self, mob): | 
					
						
							|  |  |  |         for line in mob.family_members_with_points(): | 
					
						
							|  |  |  |             #Find point of line cloest to 1 on C | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |             if not isinstance(line, Line): | 
					
						
							|  |  |  |                 line.insert_n_anchor_points(self.min_added_anchors) | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             p1 = line.get_start()+LEFT | 
					
						
							|  |  |  |             p2 = line.get_end()+LEFT | 
					
						
							| 
									
										
										
										
											2018-08-15 17:30:24 -07:00
										 |  |  |             t = (-np.dot(p1, p2-p1))/(get_norm(p2-p1)**2) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             closest_to_one = interpolate( | 
					
						
							|  |  |  |                 line.get_start(), line.get_end(), t | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             #See how big this line will become | 
					
						
							|  |  |  |             diameter = abs(zeta(complex(*closest_to_one[:2]))) | 
					
						
							|  |  |  |             target_num_anchors = np.clip( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |                 int(self.anchor_density*np.pi*diameter), | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |                 self.min_added_anchors, | 
					
						
							|  |  |  |                 self.max_added_anchors, | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             num_anchors = line.get_num_anchor_points() | 
					
						
							|  |  |  |             if num_anchors < target_num_anchors: | 
					
						
							|  |  |  |                 line.insert_n_anchor_points(target_num_anchors-num_anchors) | 
					
						
							|  |  |  |             line.make_smooth() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_extra_plane_lines_for_zeta(self, animate = False, **kwargs): | 
					
						
							|  |  |  |         dense_grid = self.get_dense_grid(**kwargs) | 
					
						
							|  |  |  |         if animate: | 
					
						
							|  |  |  |             self.play(ShowCreation(dense_grid)) | 
					
						
							|  |  |  |         self.plane.add(dense_grid) | 
					
						
							|  |  |  |         self.add(self.plane) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_dense_grid(self, step_size = 1./16): | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         epsilon = 0.1 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         x_range = np.arange( | 
					
						
							|  |  |  |             max(self.x_min, self.extra_lines_x_min), | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             min(self.x_max, self.extra_lines_x_max), | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             step_size | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         y_range = np.arange( | 
					
						
							|  |  |  |             max(self.y_min, self.extra_lines_y_min), | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             min(self.y_max, self.extra_lines_y_max), | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             step_size | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         vert_lines = VGroup(*[ | 
					
						
							|  |  |  |             Line( | 
					
						
							|  |  |  |                 self.y_min*UP, | 
					
						
							|  |  |  |                 self.y_max*UP, | 
					
						
							|  |  |  |             ).shift(x*RIGHT) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             for x in x_range | 
					
						
							|  |  |  |             if abs(x-1) > epsilon | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         ]) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:59:39 -07:00
										 |  |  |         vert_lines.set_color_by_gradient( | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             self.vert_start_color, self.vert_end_color | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         horiz_lines = VGroup(*[ | 
					
						
							|  |  |  |             Line( | 
					
						
							|  |  |  |                 self.x_min*RIGHT, | 
					
						
							|  |  |  |                 self.x_max*RIGHT, | 
					
						
							|  |  |  |             ).shift(y*UP) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             for y in y_range | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             if abs(y) > epsilon | 
					
						
							|  |  |  |         ]) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:59:39 -07:00
										 |  |  |         horiz_lines.set_color_by_gradient( | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             self.horiz_start_color, self.horiz_end_color | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         dense_grid = VGroup(horiz_lines, vert_lines) | 
					
						
							|  |  |  |         dense_grid.set_stroke(width = 1) | 
					
						
							|  |  |  |         return dense_grid | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_reflected_plane(self, animate = False): | 
					
						
							|  |  |  |         reflected_plane = self.get_reflected_plane() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         if animate: | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             self.play(ShowCreation(reflected_plane, run_time = 5)) | 
					
						
							|  |  |  |         self.plane.add(reflected_plane) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.add(self.plane) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |     def get_reflected_plane(self): | 
					
						
							|  |  |  |         reflected_plane = self.plane.copy() | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         reflected_plane.rotate(np.pi, UP, about_point = RIGHT) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         for mob in reflected_plane.family_members_with_points(): | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             mob.set_color( | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |                 Color(rgb = 1-0.5*color_to_rgb(mob.get_color())) | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.prepare_for_transformation(reflected_plane) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         reflected_plane.submobjects = list(reversed( | 
					
						
							|  |  |  |             reflected_plane.family_members_with_points() | 
					
						
							|  |  |  |         )) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         return reflected_plane | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |     def apply_zeta_function(self, **kwargs): | 
					
						
							|  |  |  |         transform_kwargs = dict(self.default_apply_complex_function_kwargs) | 
					
						
							|  |  |  |         transform_kwargs.update(kwargs) | 
					
						
							|  |  |  |         self.apply_complex_function(zeta, **kwargs) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-06 13:01:09 -08:00
										 |  |  | class TestZetaOnHalfPlane(ZetaTransformationScene): | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |     CONFIG = { | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         "anchor_density" : 15, | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.add_extra_plane_lines_for_zeta() | 
					
						
							|  |  |  |         self.prepare_for_transformation(self.plane) | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |         print(sum([ | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             mob.get_num_points() | 
					
						
							|  |  |  |             for mob in self.plane.family_members_with_points() | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |         ])) | 
					
						
							|  |  |  |         print(len(self.plane.family_members_with_points())) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.apply_zeta_function() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-06 13:01:09 -08:00
										 |  |  | class TestZetaOnFullPlane(ZetaTransformationScene): | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_transformable_plane(animate = True) | 
					
						
							|  |  |  |         self.add_extra_plane_lines_for_zeta(animate = True) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.add_reflected_plane(animate = True) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.apply_zeta_function() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 12:04:06 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | class TestZetaOnLine(ZetaTransformationScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         line = Line(UP+20*LEFT, UP+20*RIGHT) | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.plane.submobjects = [line] | 
					
						
							|  |  |  |         self.apply_zeta_function() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play(ShowCreation(line, run_time = 10)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(3) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | ###################### | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class IntroduceZeta(ZetaTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "default_apply_complex_function_kwargs" : { | 
					
						
							|  |  |  |             "run_time" : 8, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         title = TextMobject("Riemann zeta function") | 
					
						
							|  |  |  |         title.add_background_rectangle() | 
					
						
							|  |  |  |         title.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         func_mob = VGroup( | 
					
						
							|  |  |  |             TexMobject("\\zeta(s) = "), | 
					
						
							|  |  |  |             TexMobject("\\sum_{n=1}^\\infty \\frac{1}{n^s}") | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         func_mob.arrange_submobjects(RIGHT, buff = 0) | 
					
						
							|  |  |  |         for submob in func_mob: | 
					
						
							|  |  |  |             submob.add_background_rectangle() | 
					
						
							|  |  |  |         func_mob.next_to(title, DOWN) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         randy = Randolph().flip() | 
					
						
							|  |  |  |         randy.to_corner(DOWN+RIGHT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add_foreground_mobjects(title, func_mob) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.add_extra_plane_lines_for_zeta() | 
					
						
							|  |  |  |         self.play(ShowCreation(self.plane, run_time = 2)) | 
					
						
							|  |  |  |         reflected_plane = self.get_reflected_plane() | 
					
						
							|  |  |  |         self.play(ShowCreation(reflected_plane, run_time = 2)) | 
					
						
							|  |  |  |         self.plane.add(reflected_plane) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.apply_zeta_function() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play(FadeIn(randy)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             randy.change_mode, "confused", | 
					
						
							|  |  |  |             randy.look_at, func_mob, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class WhyPeopleMayKnowIt(TeacherStudentsScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         title = TextMobject("Riemann zeta function") | 
					
						
							|  |  |  |         title.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         func_mob = TexMobject( | 
					
						
							|  |  |  |             "\\zeta(s) = \\sum_{n=1}^\\infty \\frac{1}{n^s}" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         func_mob.next_to(title, DOWN, aligned_edge = LEFT) | 
					
						
							|  |  |  |         self.add(title, func_mob) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         mercenary_thought = VGroup( | 
					
						
							| 
									
										
										
										
											2018-03-30 11:59:39 -07:00
										 |  |  |             TexMobject("\\$1{,}000{,}000").set_color_by_gradient(GREEN_B, GREEN_D), | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             TexMobject("\\zeta(s) = 0") | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         mercenary_thought.arrange_submobjects(DOWN) | 
					
						
							|  |  |  |         divergent_sum = VGroup( | 
					
						
							|  |  |  |             TexMobject("1+2+3+4+\\cdots = -\\frac{1}{12}"), | 
					
						
							|  |  |  |             TexMobject("\\zeta(-1) = -\\frac{1}{12}") | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         divergent_sum.arrange_submobjects(DOWN) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:59:39 -07:00
										 |  |  |         divergent_sum[0].set_color_by_gradient(YELLOW, MAROON_B) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         divergent_sum[1].set_color(BLACK) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         #Thoughts | 
					
						
							|  |  |  |         self.play(*it.chain(*[ | 
					
						
							|  |  |  |             [pi.change_mode, "pondering", pi.look_at, func_mob] | 
					
						
							| 
									
										
										
										
											2017-02-14 14:00:50 -08:00
										 |  |  |             for pi in self.get_pi_creatures() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         ])) | 
					
						
							|  |  |  |         self.random_blink() | 
					
						
							|  |  |  |         self.student_thinks( | 
					
						
							|  |  |  |             mercenary_thought, student_index = 2, | 
					
						
							|  |  |  |             target_mode = "surprised", | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         student = self.get_students()[2] | 
					
						
							|  |  |  |         self.random_blink() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.student_thinks( | 
					
						
							|  |  |  |             divergent_sum, student_index = 1, | 
					
						
							|  |  |  |             added_anims = [student.change_mode, "plain"] | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         student = self.get_students()[1] | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             student.change_mode, "confused", | 
					
						
							|  |  |  |             student.look_at, divergent_sum, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.random_blink() | 
					
						
							|  |  |  |         self.play(*it.chain(*[ | 
					
						
							|  |  |  |             [pi.change_mode, "confused", pi.look_at, divergent_sum] | 
					
						
							| 
									
										
										
										
											2017-02-14 14:00:50 -08:00
										 |  |  |             for pi in self.get_pi_creatures() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         ])) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.random_blink() | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         divergent_sum[1].set_color(WHITE) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         self.play(Write(divergent_sum[1])) | 
					
						
							|  |  |  |         self.random_blink() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         #Ask about continuation | 
					
						
							|  |  |  |         self.student_says( | 
					
						
							|  |  |  |             TextMobject("Can you explain \\\\" , "``analytic continuation''?"), | 
					
						
							|  |  |  |             student_index = 1, | 
					
						
							|  |  |  |             target_mode = "raise_right_hand" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.change_student_modes( | 
					
						
							|  |  |  |             "raise_left_hand", | 
					
						
							|  |  |  |             "raise_right_hand", | 
					
						
							|  |  |  |             "raise_left_hand", | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             self.get_teacher().change_mode, "happy", | 
					
						
							|  |  |  |             self.get_teacher().look_at, student.eyes, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.random_blink() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.random_blink() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | class ComplexValuedFunctions(ComplexTransformationScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         title = TextMobject("Complex-valued function") | 
					
						
							|  |  |  |         title.scale(1.5) | 
					
						
							|  |  |  |         title.add_background_rectangle() | 
					
						
							|  |  |  |         title.to_edge(UP) | 
					
						
							|  |  |  |         self.add(title) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         z_in = Dot(UP+RIGHT, color = YELLOW) | 
					
						
							|  |  |  |         z_out = Dot(4*RIGHT + 2*UP, color = MAROON_B) | 
					
						
							|  |  |  |         arrow = Arrow(z_in, z_out, buff = 0.1) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         arrow.set_color(WHITE) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         z = TexMobject("z").next_to(z_in, DOWN+LEFT, buff = SMALL_BUFF) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         z.set_color(z_in.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         f_z = TexMobject("f(z)").next_to(z_out, UP+RIGHT, buff = SMALL_BUFF) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         f_z.set_color(z_out.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.add(z_in, z) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         self.play(ShowCreation(arrow)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(z_out), | 
					
						
							|  |  |  |             Write(f_z) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | class PreviewZetaAndContinuation(ZetaTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "default_apply_complex_function_kwargs" : { | 
					
						
							|  |  |  |             "run_time" : 4, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.add_extra_plane_lines_for_zeta() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         reflected_plane = self.get_reflected_plane() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         titles = [ | 
					
						
							|  |  |  |             TextMobject( | 
					
						
							|  |  |  |                 "What does", "%s"%s, | 
					
						
							|  |  |  |                 "look like?", | 
					
						
							|  |  |  |                 alignment = "", | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             for s in [ | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |                 "$\\displaystyle \\sum_{n=1}^\\infty \\frac{1}{n^s}$", | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |                 "analytic continuation" | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         for mob in titles: | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             mob[1].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             mob.to_corner(UP+LEFT, buff = 0.7) | 
					
						
							|  |  |  |             mob.add_background_rectangle() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.remove(self.plane) | 
					
						
							|  |  |  |         self.play(Write(titles[0], run_time = 2)) | 
					
						
							|  |  |  |         self.add_foreground_mobjects(titles[0]) | 
					
						
							|  |  |  |         self.play(FadeIn(self.plane)) | 
					
						
							|  |  |  |         self.apply_zeta_function() | 
					
						
							|  |  |  |         reflected_plane.apply_complex_function(zeta) | 
					
						
							|  |  |  |         reflected_plane.make_smooth() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         reflected_plane.set_stroke(width = 2) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play(Transform(*titles)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play(ShowCreation( | 
					
						
							|  |  |  |             reflected_plane, | 
					
						
							|  |  |  |             submobject_mode = "all_at_once", | 
					
						
							|  |  |  |             run_time = 2 | 
					
						
							|  |  |  |         )) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class AssumeKnowledgeOfComplexNumbers(ComplexTransformationScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         z = complex(5, 2) | 
					
						
							|  |  |  |         dot = Dot(z.real*RIGHT + z.imag*UP, color = YELLOW) | 
					
						
							|  |  |  |         line = Line(ORIGIN, dot.get_center(), color = dot.get_color()) | 
					
						
							|  |  |  |         x_line = Line(ORIGIN, z.real*RIGHT, color = GREEN_B) | 
					
						
							|  |  |  |         y_line = Line(ORIGIN, z.imag*UP, color = RED) | 
					
						
							|  |  |  |         y_line.shift(z.real*RIGHT) | 
					
						
							|  |  |  |         complex_number_label = TexMobject( | 
					
						
							|  |  |  |             "%d+%di"%(int(z.real), int(z.imag)) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         complex_number_label[0].set_color(x_line.get_color()) | 
					
						
							|  |  |  |         complex_number_label[2].set_color(y_line.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         complex_number_label.next_to(dot, UP) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         text = VGroup( | 
					
						
							|  |  |  |             TextMobject("Assumed knowledge:"), | 
					
						
							|  |  |  |             TextMobject("1) What complex numbers are."), | 
					
						
							|  |  |  |             TextMobject("2) How to work with them."), | 
					
						
							|  |  |  |             TextMobject("3) Maybe derivatives?"), | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         text.arrange_submobjects(DOWN, aligned_edge = LEFT) | 
					
						
							|  |  |  |         for words in text: | 
					
						
							|  |  |  |             words.add_background_rectangle() | 
					
						
							|  |  |  |         text[0].shift(LEFT) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         text[-1].set_color(PINK) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         text.to_corner(UP+LEFT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(Write(text[0])) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play(FadeIn(text[1])) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(x_line), | 
					
						
							|  |  |  |             ShowCreation(y_line), | 
					
						
							|  |  |  |             ShowCreation(VGroup(line, dot)), | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             Write(complex_number_label), | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         ) | 
					
						
							|  |  |  |         self.play(Write(text[2])) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play(Write(text[3])) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play(text[3].fade) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class DefineForRealS(PiCreatureScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         zeta_def, s_group = self.get_definition("s") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.initial_definition(zeta_def) | 
					
						
							|  |  |  |         self.plug_in_two(zeta_def) | 
					
						
							|  |  |  |         self.plug_in_three_and_four(zeta_def) | 
					
						
							|  |  |  |         self.plug_in_negative_values(zeta_def) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def initial_definition(self, zeta_def): | 
					
						
							|  |  |  |         zeta_s, sum_terms, brace, sigma = zeta_def | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.say("Let's define $\\zeta(s)$") | 
					
						
							|  |  |  |         self.blink() | 
					
						
							|  |  |  |         pre_zeta_s = VGroup( | 
					
						
							|  |  |  |             *self.pi_creature.bubble.content.copy()[-4:] | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         pre_zeta_s.add(VectorizedPoint(pre_zeta_s.get_right())) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             Transform(pre_zeta_s, zeta_s), | 
					
						
							|  |  |  |             *self.get_bubble_fade_anims() | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.remove(pre_zeta_s) | 
					
						
							|  |  |  |         self.add(zeta_s) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for count, term in enumerate(sum_terms): | 
					
						
							|  |  |  |             self.play(FadeIn(term), run_time = 0.5) | 
					
						
							|  |  |  |             if count%2 == 0: | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |                 self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             GrowFromCenter(brace), | 
					
						
							|  |  |  |             Write(sigma), | 
					
						
							|  |  |  |             self.pi_creature.change_mode, "pondering" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def plug_in_two(self, zeta_def): | 
					
						
							|  |  |  |         two_def = self.get_definition("2")[0] | 
					
						
							|  |  |  |         number_line = NumberLine( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             x_min = 0, | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             x_max = 3, | 
					
						
							|  |  |  |             tick_frequency = 0.25, | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |             numbers_with_elongated_ticks = list(range(4)), | 
					
						
							| 
									
										
										
										
											2017-05-23 13:17:31 -07:00
										 |  |  |             unit_size = 3, | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         ) | 
					
						
							|  |  |  |         number_line.add_numbers() | 
					
						
							|  |  |  |         number_line.next_to(self.pi_creature, LEFT) | 
					
						
							|  |  |  |         number_line.to_edge(LEFT) | 
					
						
							|  |  |  |         self.number_line = number_line | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         lines, braces, dots, pi_dot = self.get_sum_lines(2) | 
					
						
							|  |  |  |         fracs = VGroup(*[ | 
					
						
							|  |  |  |             TexMobject("\\frac{1}{%d}"%((d+1)**2)).scale(0.7) | 
					
						
							|  |  |  |             for d, brace in enumerate(braces) | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         for frac, brace, line in zip(fracs, braces, lines): | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             frac.set_color(line.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             frac.next_to(brace, UP, buff = SMALL_BUFF) | 
					
						
							|  |  |  |             if frac is fracs[-1]: | 
					
						
							|  |  |  |                 frac.shift(0.5*RIGHT + 0.2*UP) | 
					
						
							|  |  |  |                 arrow = Arrow( | 
					
						
							|  |  |  |                     frac.get_bottom(), brace.get_top(), | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |                     tip_length = 0.1, | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |                     buff = 0.1 | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |                 arrow.set_color(line.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |                 frac.add(arrow) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pi_term = TexMobject("= \\frac{\\pi^2}{6}") | 
					
						
							|  |  |  |         pi_term.next_to(zeta_def[1], RIGHT) | 
					
						
							|  |  |  |         pi_arrow = Arrow( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             pi_term[-1].get_bottom(), pi_dot, | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             color = pi_dot.get_color() | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         approx = TexMobject("\\approx 1.645") | 
					
						
							|  |  |  |         approx.next_to(pi_term) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(Transform(zeta_def, two_def)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play(ShowCreation(number_line)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for frac, brace, line in zip(fracs, braces, lines): | 
					
						
							|  |  |  |             self.play( | 
					
						
							|  |  |  |                 Write(frac), | 
					
						
							|  |  |  |                 GrowFromCenter(brace), | 
					
						
							|  |  |  |                 ShowCreation(line), | 
					
						
							|  |  |  |                 run_time = 0.7 | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait(0.7) | 
					
						
							|  |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(VGroup(*lines[4:])), | 
					
						
							|  |  |  |             Write(dots) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             Write(pi_term), | 
					
						
							|  |  |  |             ShowCreation(VGroup(pi_arrow, pi_dot)), | 
					
						
							|  |  |  |             self.pi_creature.change_mode, "hooray" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             Write(approx), | 
					
						
							|  |  |  |             self.pi_creature.change_mode, "happy" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(3) | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(FadeOut, [ | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             fracs, pi_arrow, pi_dot, approx, | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         ]))) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.lines = lines | 
					
						
							|  |  |  |         self.braces = braces | 
					
						
							|  |  |  |         self.dots = dots | 
					
						
							|  |  |  |         self.final_dot = pi_dot | 
					
						
							|  |  |  |         self.final_sum = pi_term | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def plug_in_three_and_four(self, zeta_def): | 
					
						
							|  |  |  |         final_sums = ["1.202\\dots", "\\frac{\\pi^4}{90}"] | 
					
						
							|  |  |  |         sum_terms, brace, sigma = zeta_def[1:] | 
					
						
							|  |  |  |         for exponent, final_sum in zip([3, 4], final_sums): | 
					
						
							|  |  |  |             self.transition_to_new_input(zeta_def, exponent, final_sum) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         arrow = Arrow(sum_terms.get_left(), sum_terms.get_right()) | 
					
						
							|  |  |  |         arrow.next_to(sum_terms, DOWN) | 
					
						
							|  |  |  |         smaller_words = TextMobject("Getting smaller") | 
					
						
							|  |  |  |         smaller_words.next_to(arrow, DOWN) | 
					
						
							|  |  |  |         self.arrow, self.smaller_words = arrow, smaller_words | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(arrow), | 
					
						
							|  |  |  |             Write(smaller_words) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.change_mode("happy") | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def plug_in_negative_values(self, zeta_def): | 
					
						
							|  |  |  |         zeta_s, sum_terms, brace, sigma = zeta_def | 
					
						
							|  |  |  |         arrow = self.arrow | 
					
						
							|  |  |  |         smaller_words = self.smaller_words | 
					
						
							|  |  |  |         bigger_words = TextMobject("Getting \\emph{bigger}?") | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         bigger_words.move_to(self.smaller_words) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         #plug in -1 | 
					
						
							|  |  |  |         self.transition_to_new_input(zeta_def, -1, "-\\frac{1}{12}") | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             Transform(self.smaller_words, bigger_words), | 
					
						
							|  |  |  |             self.pi_creature.change_mode, "confused" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         new_sum_terms = TexMobject( | 
					
						
							|  |  |  |             list("1+2+3+4+") + ["\\cdots"] | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         new_sum_terms.move_to(sum_terms, LEFT) | 
					
						
							|  |  |  |         arrow.target = arrow.copy().next_to(new_sum_terms, DOWN) | 
					
						
							|  |  |  |         arrow.target.stretch_to_fit_width(new_sum_terms.get_width()) | 
					
						
							|  |  |  |         bigger_words.next_to(arrow.target, DOWN) | 
					
						
							|  |  |  |         new_brace = Brace(new_sum_terms, UP) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             Transform(sum_terms, new_sum_terms), | 
					
						
							|  |  |  |             Transform(brace, new_brace), | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             sigma.next_to, new_brace, UP, | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             MoveToTarget(arrow), | 
					
						
							|  |  |  |             Transform(smaller_words, bigger_words), | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             self.final_sum.next_to, new_sum_terms, RIGHT | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(3) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         #plug in -2 | 
					
						
							|  |  |  |         new_sum_terms = TexMobject( | 
					
						
							|  |  |  |             list("1+4+9+16+") + ["\\cdots"] | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         new_sum_terms.move_to(sum_terms, LEFT) | 
					
						
							|  |  |  |         new_zeta_def, ignore = self.get_definition("-2") | 
					
						
							|  |  |  |         zeta_minus_two, ignore, ignore, new_sigma = new_zeta_def | 
					
						
							|  |  |  |         new_sigma.next_to(brace, UP) | 
					
						
							|  |  |  |         new_final_sum = TexMobject("=0") | 
					
						
							|  |  |  |         new_final_sum.next_to(new_sum_terms) | 
					
						
							|  |  |  |         lines, braces, dots, final_dot = self.get_sum_lines(-2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             Transform(zeta_s, zeta_minus_two), | 
					
						
							|  |  |  |             Transform(sum_terms, new_sum_terms), | 
					
						
							|  |  |  |             Transform(sigma, new_sigma), | 
					
						
							|  |  |  |             Transform(self.final_sum, new_final_sum), | 
					
						
							|  |  |  |             Transform(self.lines, lines), | 
					
						
							|  |  |  |             Transform(self.braces, braces), | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         self.change_mode("pleading") | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def get_definition(self, input_string, input_color = YELLOW): | 
					
						
							|  |  |  |         inputs = VGroup() | 
					
						
							|  |  |  |         num_shown_terms = 4 | 
					
						
							|  |  |  |         n_input_chars = len(input_string) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         zeta_s_eq = TexMobject("\\zeta(%s) = "%input_string) | 
					
						
							|  |  |  |         zeta_s_eq.to_edge(LEFT, buff = LARGE_BUFF) | 
					
						
							|  |  |  |         zeta_s_eq.shift(0.5*UP) | 
					
						
							|  |  |  |         inputs.add(*zeta_s_eq[2:2+n_input_chars]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         sum_terms = TexMobject(*it.chain(*list(zip( | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |             [ | 
					
						
							|  |  |  |                 "\\frac{1}{%d^{%s}}"%(d, input_string) | 
					
						
							|  |  |  |                 for d in range(1, 1+num_shown_terms) | 
					
						
							|  |  |  |             ], | 
					
						
							|  |  |  |             it.cycle(["+"]) | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         )))) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         sum_terms.add(TexMobject("\\cdots").next_to(sum_terms)) | 
					
						
							|  |  |  |         sum_terms.next_to(zeta_s_eq, RIGHT) | 
					
						
							|  |  |  |         for x in range(num_shown_terms): | 
					
						
							|  |  |  |             inputs.add(*sum_terms[2*x][-n_input_chars:]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         brace = Brace(sum_terms, UP) | 
					
						
							|  |  |  |         sigma = TexMobject( | 
					
						
							|  |  |  |             "\\sum_{n=1}^\\infty \\frac{1}{n^{%s}}"%input_string | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         sigma.next_to(brace, UP) | 
					
						
							|  |  |  |         inputs.add(*sigma[-n_input_chars:]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         inputs.set_color(input_color) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         group = VGroup(zeta_s_eq, sum_terms, brace, sigma) | 
					
						
							|  |  |  |         return group, inputs | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_sum_lines(self, exponent, line_thickness = 6): | 
					
						
							|  |  |  |         num_lines = 100 if exponent > 0 else 6 | 
					
						
							|  |  |  |         powers = [0] + [x**(-exponent) for x in range(1, num_lines)] | 
					
						
							|  |  |  |         power_sums = np.cumsum(powers) | 
					
						
							|  |  |  |         lines = VGroup(*[ | 
					
						
							|  |  |  |             Line( | 
					
						
							|  |  |  |                 self.number_line.number_to_point(s1), | 
					
						
							|  |  |  |                 self.number_line.number_to_point(s2), | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             for s1, s2 in zip(power_sums, power_sums[1:]) | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         lines.set_stroke(width = line_thickness) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:59:39 -07:00
										 |  |  |         # VGroup(*lines[:4]).set_color_by_gradient(RED, GREEN_B) | 
					
						
							|  |  |  |         # VGroup(*lines[4:]).set_color_by_gradient(GREEN_B, MAROON_B) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(*lines[::2]).set_color(MAROON_B) | 
					
						
							|  |  |  |         VGroup(*lines[1::2]).set_color(RED) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         braces = VGroup(*[ | 
					
						
							|  |  |  |             Brace(line, UP) | 
					
						
							|  |  |  |             for line in lines[:4] | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         dots = TexMobject("...") | 
					
						
							|  |  |  |         dots.stretch_to_fit_width( | 
					
						
							|  |  |  |             0.8 * VGroup(*lines[4:]).get_width() | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         dots.next_to(braces, RIGHT, buff = SMALL_BUFF) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         final_dot = Dot( | 
					
						
							|  |  |  |             self.number_line.number_to_point(power_sums[-1]), | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             color = GREEN_B | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return lines, braces, dots, final_dot | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def transition_to_new_input(self, zeta_def, exponent, final_sum): | 
					
						
							|  |  |  |         new_zeta_def = self.get_definition(str(exponent))[0] | 
					
						
							|  |  |  |         lines, braces, dots, final_dot = self.get_sum_lines(exponent) | 
					
						
							|  |  |  |         final_sum = TexMobject("=" + final_sum) | 
					
						
							|  |  |  |         final_sum.next_to(new_zeta_def[1][-1]) | 
					
						
							|  |  |  |         final_sum.shift(SMALL_BUFF*UP) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             Transform(zeta_def, new_zeta_def), | 
					
						
							|  |  |  |             Transform(self.lines, lines), | 
					
						
							|  |  |  |             Transform(self.braces, braces), | 
					
						
							|  |  |  |             Transform(self.dots, dots), | 
					
						
							|  |  |  |             Transform(self.final_dot, final_dot), | 
					
						
							|  |  |  |             Transform(self.final_sum, final_sum), | 
					
						
							|  |  |  |             self.pi_creature.change_mode, "pondering" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | class ReadIntoZetaFunction(Scene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "statement" : "$\\zeta(-1) = -\\frac{1}{12}$", | 
					
						
							|  |  |  |         "target_mode" : "frustrated", | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         randy = Randolph(mode = "pondering") | 
					
						
							|  |  |  |         randy.shift(3*LEFT+DOWN) | 
					
						
							|  |  |  |         paper = Rectangle(width = 4, height = 5) | 
					
						
							|  |  |  |         paper.next_to(randy, RIGHT, aligned_edge = DOWN) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         paper.set_color(WHITE) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         max_width = 0.8*paper.get_width() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         title = TextMobject("$\\zeta(s)$ manual") | 
					
						
							|  |  |  |         title.next_to(paper.get_top(), DOWN) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         title.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         paper.add(title) | 
					
						
							|  |  |  |         paragraph_lines = VGroup( | 
					
						
							|  |  |  |             Line(LEFT, RIGHT), | 
					
						
							|  |  |  |             Line(LEFT, RIGHT).shift(0.2*DOWN), | 
					
						
							|  |  |  |             Line(LEFT, ORIGIN).shift(0.4*DOWN) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         paragraph_lines.set_width(max_width) | 
					
						
							| 
									
										
										
										
											2017-01-25 16:40:59 -08:00
										 |  |  |         paragraph_lines.next_to(title, DOWN, MED_LARGE_BUFF) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         paper.add(paragraph_lines) | 
					
						
							|  |  |  |         max_height = 1.5*paragraph_lines.get_height() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         statement = TextMobject(self.statement) | 
					
						
							|  |  |  |         if statement.get_width() > max_width: | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |             statement.set_width(max_width) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         if statement.get_height() > max_height: | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |             statement.set_height(max_height) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         statement.next_to(paragraph_lines, DOWN) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         statement.set_color(GREEN_B) | 
					
						
							| 
									
										
										
										
											2017-01-25 16:40:59 -08:00
										 |  |  |         paper.add(paragraph_lines.copy().next_to(statement, DOWN, MED_LARGE_BUFF)) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         randy.look_at(statement) | 
					
						
							|  |  |  |         self.add(randy, paper) | 
					
						
							|  |  |  |         self.play(Write(statement)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             randy.change_mode, self.target_mode, | 
					
						
							|  |  |  |             randy.look_at, title | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							|  |  |  |         self.play(randy.look_at, statement) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class ReadIntoZetaFunctionTrivialZero(ReadIntoZetaFunction): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "statement" : "$\\zeta(-2n) = 0$" | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ReadIntoZetaFunctionAnalyticContinuation(ReadIntoZetaFunction): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "statement" : "...analytic \\\\ continuation...", | 
					
						
							|  |  |  |         "target_mode" : "confused", | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | class IgnoreNegatives(TeacherStudentsScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         definition = TexMobject("""
 | 
					
						
							|  |  |  |             \\zeta(s) = \\sum_{n=1}^{\\infty} \\frac{1}{n^s} | 
					
						
							|  |  |  |         """)
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(definition[2], definition[-1]).set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         definition.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         self.add(definition) | 
					
						
							|  |  |  |         brace = Brace(definition, DOWN) | 
					
						
							|  |  |  |         only_s_gt_1 = brace.get_text("""
 | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             Only defined | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             for $s > 1$ | 
					
						
							|  |  |  |         """)
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         only_s_gt_1[-3].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.change_student_modes(*["confused"]*3) | 
					
						
							|  |  |  |         words = TextMobject( | 
					
						
							|  |  |  |             "Ignore $s \\le 1$ \\dots \\\\", | 
					
						
							|  |  |  |             "For now." | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         words[0][6].set_color(YELLOW) | 
					
						
							|  |  |  |         words[1].set_color(BLACK) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.teacher_says(words) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         self.play(words[1].set_color, WHITE) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.change_student_modes(*["happy"]*3) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             GrowFromCenter(brace), | 
					
						
							|  |  |  |             Write(only_s_gt_1), | 
					
						
							|  |  |  |             *it.chain(*[ | 
					
						
							|  |  |  |                 [pi.look_at, definition] | 
					
						
							| 
									
										
										
										
											2017-02-14 14:00:50 -08:00
										 |  |  |                 for pi in self.get_pi_creatures() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             ]) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.random_blink(3) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class RiemannFatherOfComplex(ComplexTransformationScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         name = TextMobject( | 
					
						
							|  |  |  |             "Bernhard Riemann $\\rightarrow$ Complex analysis" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         name.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         name.shift(0.25*DOWN) | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |         name.add_background_rectangle() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         # photo = Square() | 
					
						
							|  |  |  |         photo = ImageMobject("Riemann", invert = False) | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         photo.set_width(5) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         photo.next_to(name, DOWN, aligned_edge = LEFT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add(photo) | 
					
						
							|  |  |  |         self.play(Write(name)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         input_dot = Dot(2*RIGHT+UP, color = YELLOW) | 
					
						
							|  |  |  |         arc = Arc(-2*np.pi/3) | 
					
						
							|  |  |  |         arc.rotate(-np.pi) | 
					
						
							|  |  |  |         arc.add_tip() | 
					
						
							|  |  |  |         arc.shift(input_dot.get_top()-arc.points[0]+SMALL_BUFF*UP) | 
					
						
							|  |  |  |         output_dot = Dot( | 
					
						
							|  |  |  |             arc.points[-1] + SMALL_BUFF*(2*RIGHT+DOWN), | 
					
						
							|  |  |  |             color = MAROON_B | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         for dot, tex in (input_dot, "z"), (output_dot, "f(z)"): | 
					
						
							|  |  |  |             dot.label = TexMobject(tex) | 
					
						
							|  |  |  |             dot.label.add_background_rectangle() | 
					
						
							|  |  |  |             dot.label.next_to(dot, DOWN+RIGHT, buff = SMALL_BUFF) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             dot.label.set_color(dot.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(input_dot), | 
					
						
							|  |  |  |             Write(input_dot.label) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(ShowCreation(arc)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(output_dot), | 
					
						
							|  |  |  |             Write(output_dot.label) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class FromRealToComplex(ComplexTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "plane_config" : { | 
					
						
							|  |  |  |             "space_unit_to_x_unit" : 2, | 
					
						
							|  |  |  |             "space_unit_to_y_unit" : 2, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         "background_label_scale_val" : 0.7, | 
					
						
							|  |  |  |         "output_color" : GREEN_B, | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         "num_lines_in_spiril_sum" : 1000, | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.handle_background() | 
					
						
							|  |  |  |         self.show_real_to_real() | 
					
						
							|  |  |  |         self.transition_to_complex() | 
					
						
							|  |  |  |         self.single_out_complex_exponent() | 
					
						
							|  |  |  |         ##Fade to several scenes defined below | 
					
						
							|  |  |  |         self.show_s_equals_two_lines() | 
					
						
							|  |  |  |         self.transition_to_spiril_sum() | 
					
						
							|  |  |  |         self.vary_complex_input() | 
					
						
							|  |  |  |         self.show_domain_of_convergence() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.ask_about_visualizing_all() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def handle_background(self): | 
					
						
							|  |  |  |         self.remove(self.background) | 
					
						
							|  |  |  |         #Oh yeah, this is great practice... | 
					
						
							|  |  |  |         self.background[-1].remove(*self.background[-1][-3:]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def show_real_to_real(self): | 
					
						
							|  |  |  |         zeta = self.get_zeta_definition("2",  "\\frac{\\pi^2}{6}") | 
					
						
							|  |  |  |         number_line = NumberLine( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             unit_size = 2, | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             tick_frequency = 0.5, | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |             numbers_with_elongated_ticks = list(range(-2, 3)) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         ) | 
					
						
							|  |  |  |         number_line.add_numbers() | 
					
						
							|  |  |  |         input_dot = Dot(number_line.number_to_point(2)) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         input_dot.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         output_dot = Dot(number_line.number_to_point(np.pi**2/6)) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         output_dot.set_color(self.output_color) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         arc = Arc( | 
					
						
							|  |  |  |             2*np.pi/3, start_angle = np.pi/6, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         arc.stretch_to_fit_width( | 
					
						
							|  |  |  |             (input_dot.get_center()-output_dot.get_center())[0] | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         arc.stretch_to_fit_height(0.5) | 
					
						
							|  |  |  |         arc.next_to(input_dot.get_center(), UP, aligned_edge = RIGHT) | 
					
						
							|  |  |  |         arc.add_tip() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         two = zeta[1][2].copy() | 
					
						
							|  |  |  |         sum_term = zeta[-1] | 
					
						
							|  |  |  |         self.add(number_line, *zeta[:-1]) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.play(Transform(two, input_dot)) | 
					
						
							|  |  |  |         self.remove(two) | 
					
						
							|  |  |  |         self.add(input_dot) | 
					
						
							|  |  |  |         self.play(ShowCreation(arc)) | 
					
						
							|  |  |  |         self.play(ShowCreation(output_dot)) | 
					
						
							|  |  |  |         self.play(Transform(output_dot.copy(), sum_term)) | 
					
						
							|  |  |  |         self.remove(*self.get_mobjects_from_last_animation()) | 
					
						
							|  |  |  |         self.add(sum_term) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation( | 
					
						
							|  |  |  |                 self.background, | 
					
						
							|  |  |  |                 run_time = 2 | 
					
						
							|  |  |  |             ), | 
					
						
							|  |  |  |             FadeOut(VGroup(arc, output_dot, number_line)), | 
					
						
							|  |  |  |             Animation(zeta), | 
					
						
							|  |  |  |             Animation(input_dot) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.zeta = zeta | 
					
						
							|  |  |  |         self.input_dot = input_dot | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def transition_to_complex(self): | 
					
						
							|  |  |  |         complex_zeta = self.get_zeta_definition("2+i", "???") | 
					
						
							|  |  |  |         input_dot = self.input_dot | 
					
						
							|  |  |  |         input_dot.generate_target() | 
					
						
							|  |  |  |         input_dot.target.move_to( | 
					
						
							|  |  |  |             self.background.num_pair_to_point((2, 1)) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         input_label = TexMobject("2+i") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         input_label.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         input_label.next_to(input_dot.target, DOWN+RIGHT, buff = SMALL_BUFF) | 
					
						
							|  |  |  |         input_label.add_background_rectangle() | 
					
						
							|  |  |  |         input_label.save_state() | 
					
						
							|  |  |  |         input_label.replace(VGroup(*complex_zeta[1][2:5])) | 
					
						
							|  |  |  |         input_label.background_rectangle.scale_in_place(0.01) | 
					
						
							|  |  |  |         self.input_label = input_label | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(Transform(self.zeta, complex_zeta)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             input_label.restore, | 
					
						
							|  |  |  |             MoveToTarget(input_dot) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def single_out_complex_exponent(self): | 
					
						
							|  |  |  |         frac_scale_factor = 1.2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         randy = Randolph() | 
					
						
							|  |  |  |         randy.to_corner() | 
					
						
							|  |  |  |         bubble = randy.get_bubble(height = 4) | 
					
						
							|  |  |  |         bubble.set_fill(BLACK, opacity = 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         frac = VGroup( | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             VectorizedPoint(self.zeta[2][3].get_left()), | 
					
						
							|  |  |  |             self.zeta[2][3], | 
					
						
							|  |  |  |             VectorizedPoint(self.zeta[2][3].get_right()), | 
					
						
							|  |  |  |             self.zeta[2][4], | 
					
						
							|  |  |  |         ).copy() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         frac.generate_target() | 
					
						
							|  |  |  |         frac.target.scale(frac_scale_factor) | 
					
						
							|  |  |  |         bubble.add_content(frac.target) | 
					
						
							|  |  |  |         new_frac = TexMobject( | 
					
						
							|  |  |  |             "\\Big(", "\\frac{1}{2}", "\\Big)", "^{2+i}" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         new_frac[-1].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         new_frac.scale(frac_scale_factor) | 
					
						
							|  |  |  |         new_frac.move_to(frac.target) | 
					
						
							|  |  |  |         new_frac.shift(LEFT+0.2*UP) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         words = TextMobject("Not repeated \\\\", " multiplication") | 
					
						
							|  |  |  |         words.scale(0.8) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         words.set_color(RED) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         words.next_to(new_frac, RIGHT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         new_words = TextMobject("Not \\emph{super} \\\\", "crucial to know...") | 
					
						
							|  |  |  |         new_words.replace(words) | 
					
						
							|  |  |  |         new_words.scale_in_place(1.3) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(FadeIn(randy)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             randy.change_mode, "confused", | 
					
						
							|  |  |  |             randy.look_at, bubble, | 
					
						
							|  |  |  |             ShowCreation(bubble), | 
					
						
							|  |  |  |             MoveToTarget(frac) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							|  |  |  |         self.play(Transform(frac, new_frac)) | 
					
						
							|  |  |  |         self.play(Write(words)) | 
					
						
							|  |  |  |         for x in range(2): | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             self.play(Blink(randy)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             Transform(words, new_words), | 
					
						
							|  |  |  |             randy.change_mode, "maybe" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.play(Blink(randy)) | 
					
						
							|  |  |  |         self.play(randy.change_mode, "happy") | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(FadeOut, [randy, bubble, frac, words]))) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def show_s_equals_two_lines(self): | 
					
						
							|  |  |  |         self.input_label.save_state() | 
					
						
							|  |  |  |         zeta = self.get_zeta_definition("2", "\\frac{\\pi^2}{6}") | 
					
						
							|  |  |  |         lines, output_dot = self.get_sum_lines(2) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         sum_terms = self.zeta[2][:-1:3] | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         dots_copy = zeta[2][-1].copy() | 
					
						
							|  |  |  |         pi_copy = zeta[3].copy() | 
					
						
							|  |  |  |         def transform_and_replace(m1, m2): | 
					
						
							|  |  |  |             self.play(Transform(m1, m2)) | 
					
						
							|  |  |  |             self.remove(m1) | 
					
						
							|  |  |  |             self.add(m2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             self.input_dot.shift, 2*DOWN, | 
					
						
							|  |  |  |             self.input_label.fade, 0.7, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Transform(self.zeta, zeta)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for term, line in zip(sum_terms, lines): | 
					
						
							|  |  |  |             line.save_state() | 
					
						
							|  |  |  |             line.next_to(term, DOWN) | 
					
						
							|  |  |  |             term_copy = term.copy() | 
					
						
							|  |  |  |             transform_and_replace(term_copy, line) | 
					
						
							|  |  |  |             self.play(line.restore) | 
					
						
							|  |  |  |         later_lines = VGroup(*lines[4:]) | 
					
						
							|  |  |  |         transform_and_replace(dots_copy, later_lines) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         transform_and_replace(pi_copy, output_dot) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.lines = lines | 
					
						
							|  |  |  |         self.output_dot = output_dot | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |     def transition_to_spiril_sum(self): | 
					
						
							|  |  |  |         zeta = self.get_zeta_definition("2+i", "1.15 - 0.44i") | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         zeta.set_width(FRAME_WIDTH-1) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         zeta.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         lines, output_dot = self.get_sum_lines(complex(2, 1)) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             self.input_dot.shift, 2*UP, | 
					
						
							|  |  |  |             self.input_label.restore, | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.play(Transform(self.zeta, zeta)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             Transform(self.lines, lines), | 
					
						
							|  |  |  |             Transform(self.output_dot, output_dot), | 
					
						
							|  |  |  |             run_time = 2, | 
					
						
							|  |  |  |             path_arc = -np.pi/6, | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |     def vary_complex_input(self): | 
					
						
							|  |  |  |         zeta = self.get_zeta_definition("s", "") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         zeta[3].set_color(BLACK) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         self.play(Transform(self.zeta, zeta)) | 
					
						
							|  |  |  |         self.play(FadeOut(self.input_label)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         inputs = [ | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |             complex(1.5, 1.8), | 
					
						
							|  |  |  |             complex(1.5, -1), | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             complex(3, -1), | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |             complex(1.5, 1.8), | 
					
						
							|  |  |  |             complex(1.5, -1.8), | 
					
						
							|  |  |  |             complex(1.4, -1.8), | 
					
						
							|  |  |  |             complex(1.5, 0), | 
					
						
							|  |  |  |             complex(2, 1), | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         ] | 
					
						
							|  |  |  |         for s in inputs: | 
					
						
							|  |  |  |             input_point = self.z_to_point(s) | 
					
						
							|  |  |  |             lines, output_dot = self.get_sum_lines(s) | 
					
						
							|  |  |  |             self.play( | 
					
						
							|  |  |  |                 self.input_dot.move_to, input_point, | 
					
						
							|  |  |  |                 Transform(self.lines, lines), | 
					
						
							|  |  |  |                 Transform(self.output_dot, output_dot), | 
					
						
							|  |  |  |                 run_time = 2 | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() | 
					
						
							|  |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |     def show_domain_of_convergence(self, opacity = 0.2): | 
					
						
							|  |  |  |         domain = Rectangle( | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             width = FRAME_X_RADIUS-2, | 
					
						
							|  |  |  |             height = FRAME_HEIGHT, | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |             stroke_width = 0, | 
					
						
							|  |  |  |             fill_color = YELLOW, | 
					
						
							|  |  |  |             fill_opacity = opacity, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         domain.to_edge(RIGHT, buff = 0) | 
					
						
							|  |  |  |         anti_domain = Rectangle( | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             width = FRAME_X_RADIUS+2, | 
					
						
							|  |  |  |             height = FRAME_HEIGHT, | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |             stroke_width = 0, | 
					
						
							|  |  |  |             fill_color = RED, | 
					
						
							|  |  |  |             fill_opacity = opacity, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         anti_domain.to_edge(LEFT, buff = 0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         domain_words = TextMobject("""
 | 
					
						
							|  |  |  |             $\\zeta(s)$ happily | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             converges and | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |             makes sense | 
					
						
							|  |  |  |         """)
 | 
					
						
							| 
									
										
										
										
											2017-01-25 16:40:59 -08:00
										 |  |  |         domain_words.to_corner(UP+RIGHT, buff = MED_LARGE_BUFF) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         anti_domain_words = TextMobject("""
 | 
					
						
							|  |  |  |             Not so much... | 
					
						
							|  |  |  |         """)
 | 
					
						
							|  |  |  |         anti_domain_words.next_to(ORIGIN, LEFT, buff = LARGE_BUFF) | 
					
						
							|  |  |  |         anti_domain_words.shift(1.5*DOWN) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(FadeIn(domain)) | 
					
						
							|  |  |  |         self.play(Write(domain_words)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play(FadeIn(anti_domain)) | 
					
						
							|  |  |  |         self.play(Write(anti_domain_words)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(FadeOut, [ | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |             anti_domain, anti_domain_words, | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         ]))) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.domain_words = domain_words | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def ask_about_visualizing_all(self): | 
					
						
							|  |  |  |         morty = Mortimer().flip() | 
					
						
							|  |  |  |         morty.scale(0.7) | 
					
						
							|  |  |  |         morty.to_corner(DOWN+LEFT) | 
					
						
							| 
									
										
										
										
											2017-02-14 14:00:50 -08:00
										 |  |  |         bubble = morty.get_bubble(SpeechBubble, height = 4) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         bubble.set_fill(BLACK, opacity = 0.5) | 
					
						
							|  |  |  |         bubble.write("""
 | 
					
						
							|  |  |  |             How can we visualize | 
					
						
							|  |  |  |             this for all inputs? | 
					
						
							|  |  |  |         """)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(FadeIn(morty)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             morty.change_mode, "speaking", | 
					
						
							|  |  |  |             ShowCreation(bubble), | 
					
						
							|  |  |  |             Write(bubble.content) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(morty)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(3) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             morty.change_mode, "pondering", | 
					
						
							|  |  |  |             morty.look_at, self.input_dot, | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |             *list(map(FadeOut, [ | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |                 bubble, bubble.content, self.domain_words | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |             ])) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         ) | 
					
						
							|  |  |  |         arrow = Arrow(self.input_dot, self.output_dot, buff = SMALL_BUFF) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         arrow.set_color(WHITE) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play(ShowCreation(arrow)) | 
					
						
							|  |  |  |         self.play(Blink(morty)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |     def get_zeta_definition(self, input_string, output_string, input_color = YELLOW): | 
					
						
							|  |  |  |         inputs = VGroup() | 
					
						
							|  |  |  |         num_shown_terms = 4 | 
					
						
							|  |  |  |         n_input_chars = len(input_string) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         zeta_s_eq = TexMobject("\\zeta(%s) = "%input_string) | 
					
						
							|  |  |  |         zeta_s_eq.to_edge(LEFT, buff = LARGE_BUFF) | 
					
						
							|  |  |  |         zeta_s_eq.shift(0.5*UP) | 
					
						
							|  |  |  |         inputs.add(*zeta_s_eq[2:2+n_input_chars]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         raw_sum_terms = TexMobject(*[ | 
					
						
							|  |  |  |             "\\frac{1}{%d^{%s}} + "%(d, input_string) | 
					
						
							|  |  |  |             for d in range(1, 1+num_shown_terms) | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         sum_terms = VGroup(*it.chain(*[ | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             [ | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |                 VGroup(*term[:3]), | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |                 VGroup(*term[3:-1]), | 
					
						
							|  |  |  |                 term[-1], | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |             for term in raw_sum_terms | 
					
						
							|  |  |  |         ])) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         sum_terms.add(TexMobject("\\cdots").next_to(sum_terms[-1])) | 
					
						
							|  |  |  |         sum_terms.next_to(zeta_s_eq, RIGHT) | 
					
						
							|  |  |  |         for x in range(num_shown_terms): | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             inputs.add(*sum_terms[3*x+1]) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         output = TexMobject("= \\," + output_string) | 
					
						
							|  |  |  |         output.next_to(sum_terms, RIGHT) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         output.set_color(self.output_color) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         inputs.set_color(input_color) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         group = VGroup(zeta_s_eq, sum_terms, output) | 
					
						
							|  |  |  |         group.to_edge(UP) | 
					
						
							|  |  |  |         group.add_to_back(BackgroundRectangle(group)) | 
					
						
							|  |  |  |         return group | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_sum_lines(self, exponent, line_thickness = 6): | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         powers = [0] + [ | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             x**(-exponent) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |             for x in range(1, self.num_lines_in_spiril_sum) | 
					
						
							|  |  |  |         ] | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         power_sums = np.cumsum(powers) | 
					
						
							|  |  |  |         lines = VGroup(*[ | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |             Line(*list(map(self.z_to_point, z_pair))) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             for z_pair in zip(power_sums, power_sums[1:]) | 
					
						
							|  |  |  |         ]) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         widths = np.linspace(line_thickness, 0, len(list(lines))) | 
					
						
							|  |  |  |         for line, width in zip(lines, widths): | 
					
						
							|  |  |  |             line.set_stroke(width = width) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(*lines[::2]).set_color(MAROON_B) | 
					
						
							|  |  |  |         VGroup(*lines[1::2]).set_color(RED) | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         final_dot = Dot( | 
					
						
							| 
									
										
										
										
											2016-12-26 07:10:38 -08:00
										 |  |  |             # self.z_to_point(power_sums[-1]), | 
					
						
							|  |  |  |             self.z_to_point(zeta(exponent)), | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |             color = self.output_color | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-02 13:12:58 -08:00
										 |  |  |         return lines, final_dot | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | class TerritoryOfExponents(ComplexTransformationScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_title() | 
					
						
							|  |  |  |         familiar_territory = TextMobject("Familiar territory") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         familiar_territory.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         familiar_territory.next_to(ORIGIN, UP+RIGHT) | 
					
						
							|  |  |  |         familiar_territory.shift(2*UP) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |         real_line = Line(LEFT, RIGHT).scale(FRAME_X_RADIUS) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         real_line.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         arrow1 = Arrow(familiar_territory.get_bottom(), real_line.get_left()) | 
					
						
							|  |  |  |         arrow2 = Arrow(familiar_territory.get_bottom(), real_line.get_right()) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(arrow1, arrow2).set_color(WHITE) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         extended_realm = TextMobject("Extended realm") | 
					
						
							|  |  |  |         extended_realm.move_to(familiar_territory) | 
					
						
							|  |  |  |         full_plane = Rectangle( | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             width = FRAME_WIDTH, | 
					
						
							|  |  |  |             height = FRAME_HEIGHT, | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |             fill_color = YELLOW, | 
					
						
							|  |  |  |             fill_opacity = 0.3 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add(familiar_territory) | 
					
						
							|  |  |  |         self.play(ShowCreation(arrow1)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             Transform(arrow1, arrow2), | 
					
						
							|  |  |  |             ShowCreation(real_line) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(FadeOut(arrow1)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             FadeIn(full_plane), | 
					
						
							|  |  |  |             Transform(familiar_territory, extended_realm), | 
					
						
							|  |  |  |             Animation(real_line) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_title(self): | 
					
						
							|  |  |  |         exponent = TexMobject( | 
					
						
							|  |  |  |             "\\left(\\frac{1}{2}\\right)^s" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         exponent[-1].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2017-01-25 16:40:59 -08:00
										 |  |  |         exponent.next_to(ORIGIN, LEFT, MED_LARGE_BUFF).to_edge(UP) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         self.add_foreground_mobjects(exponent) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | class ComplexExponentiation(Scene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.extract_pure_imaginary_part() | 
					
						
							|  |  |  |         self.add_on_planes() | 
					
						
							|  |  |  |         self.show_imaginary_powers() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def extract_pure_imaginary_part(self): | 
					
						
							|  |  |  |         original = TexMobject( | 
					
						
							|  |  |  |             "\\left(\\frac{1}{2}\\right)", "^{2+i}" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         split = TexMobject( | 
					
						
							|  |  |  |              "\\left(\\frac{1}{2}\\right)", "^{2}", | 
					
						
							|  |  |  |              "\\left(\\frac{1}{2}\\right)", "^{i}", | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(original[-1], split[1], split[3]).set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         VGroup(original, split).shift(UP) | 
					
						
							|  |  |  |         real_part = VGroup(*split[:2]) | 
					
						
							|  |  |  |         imag_part = VGroup(*split[2:]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         brace = Brace(real_part) | 
					
						
							|  |  |  |         we_understand = brace.get_text( | 
					
						
							|  |  |  |             "We understand this" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(brace, we_understand).set_color(GREEN_B) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.add(original) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play(*[ | 
					
						
							|  |  |  |             Transform(*pair) | 
					
						
							|  |  |  |             for pair in [ | 
					
						
							|  |  |  |                 (original[0], split[0]), | 
					
						
							|  |  |  |                 (original[1][0], split[1]), | 
					
						
							|  |  |  |                 (original[0].copy(), split[2]), | 
					
						
							|  |  |  |                 (VGroup(*original[1][1:]), split[3]), | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         self.remove(*self.get_mobjects_from_last_animation()) | 
					
						
							|  |  |  |         self.add(real_part, imag_part) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             GrowFromCenter(brace), | 
					
						
							|  |  |  |             FadeIn(we_understand), | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             real_part.set_color, GREEN_B | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             imag_part.move_to, imag_part.get_left(), | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |             *list(map(FadeOut, [brace, we_understand, real_part])) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.imag_exponent = imag_part | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_on_planes(self): | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |         left_plane = NumberPlane(x_radius = (FRAME_X_RADIUS-1)/2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         left_plane.to_edge(LEFT, buff = 0) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |         imag_line = Line(DOWN, UP).scale(FRAME_Y_RADIUS) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         imag_line.set_color(YELLOW).fade(0.3) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         imag_line.move_to(left_plane.get_center()) | 
					
						
							|  |  |  |         left_plane.add(imag_line) | 
					
						
							|  |  |  |         left_title = TextMobject("Input space") | 
					
						
							|  |  |  |         left_title.add_background_rectangle() | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         left_title.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         left_title.next_to(left_plane.get_top(), DOWN) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |         right_plane = NumberPlane(x_radius = (FRAME_X_RADIUS-1)/2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         right_plane.to_edge(RIGHT, buff = 0) | 
					
						
							|  |  |  |         unit_circle = Circle() | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         unit_circle.set_color(MAROON_B).fade(0.3) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         unit_circle.shift(right_plane.get_center()) | 
					
						
							|  |  |  |         right_plane.add(unit_circle) | 
					
						
							|  |  |  |         right_title = TextMobject("Output space") | 
					
						
							|  |  |  |         right_title.add_background_rectangle() | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         right_title.set_color(MAROON_B) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         right_title.next_to(right_plane.get_top(), DOWN) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for plane in left_plane, right_plane: | 
					
						
							|  |  |  |             labels = VGroup() | 
					
						
							|  |  |  |             for x in range(-2, 3): | 
					
						
							|  |  |  |                 label = TexMobject(str(x)) | 
					
						
							|  |  |  |                 label.move_to(plane.num_pair_to_point((x, 0))) | 
					
						
							|  |  |  |                 labels.add(label) | 
					
						
							|  |  |  |             for y in range(-3, 4): | 
					
						
							|  |  |  |                 if y == 0: | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 label = TexMobject(str(y) + "i") | 
					
						
							|  |  |  |                 label.move_to(plane.num_pair_to_point((0, y))) | 
					
						
							|  |  |  |                 labels.add(label) | 
					
						
							|  |  |  |             for label in labels: | 
					
						
							|  |  |  |                 label.scale_in_place(0.5) | 
					
						
							|  |  |  |                 label.next_to( | 
					
						
							|  |  |  |                     label.get_center(), DOWN+RIGHT, | 
					
						
							|  |  |  |                     buff = SMALL_BUFF | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |             plane.add(labels) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         arrow = Arrow(LEFT, RIGHT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(left_plane), | 
					
						
							|  |  |  |             Write(left_title), | 
					
						
							|  |  |  |             run_time = 3 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(right_plane), | 
					
						
							|  |  |  |             Write(right_title), | 
					
						
							|  |  |  |             run_time = 3 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(ShowCreation(arrow)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.left_plane = left_plane | 
					
						
							|  |  |  |         self.right_plane = right_plane | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def show_imaginary_powers(self): | 
					
						
							|  |  |  |         i = complex(0, 1) | 
					
						
							|  |  |  |         input_dot = Dot(self.z_to_point(i)) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         input_dot.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         output_dot = Dot(self.z_to_point(0.5**(i), is_input = False)) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         output_dot.set_color(MAROON_B) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         output_dot.save_state() | 
					
						
							|  |  |  |         output_dot.move_to(input_dot) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         output_dot.set_color(input_dot.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         curr_base = 0.5 | 
					
						
							|  |  |  |         def output_dot_update(ouput_dot): | 
					
						
							|  |  |  |             y = input_dot.get_center()[1] | 
					
						
							|  |  |  |             output_dot.move_to(self.z_to_point( | 
					
						
							|  |  |  |                 curr_base**complex(0, y), is_input = False | 
					
						
							|  |  |  |             )) | 
					
						
							|  |  |  |             return output_dot | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def walk_up_and_down(): | 
					
						
							|  |  |  |             for vect in 3*DOWN, 5*UP, 5*DOWN, 2*UP: | 
					
						
							|  |  |  |                 self.play( | 
					
						
							|  |  |  |                     input_dot.shift, vect, | 
					
						
							|  |  |  |                     UpdateFromFunc(output_dot, output_dot_update), | 
					
						
							|  |  |  |                     run_time = 3 | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         exp = self.imag_exponent[-1] | 
					
						
							|  |  |  |         new_exp = TexMobject("ti") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         new_exp.set_color(exp.get_color()) | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         new_exp.set_height(exp.get_height()) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         new_exp.move_to(exp, LEFT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         nine = TexMobject("9") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         nine.set_color(BLUE) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         denom = self.imag_exponent[0][3] | 
					
						
							|  |  |  |         denom.save_state() | 
					
						
							|  |  |  |         nine.replace(denom) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(Transform(exp, new_exp)) | 
					
						
							|  |  |  |         self.play(input_dot.shift, 2*UP) | 
					
						
							|  |  |  |         self.play(input_dot.shift, 2*DOWN) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play(output_dot.restore) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         walk_up_and_down() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |         curr_base = 1./9 | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play(Transform(denom, nine)) | 
					
						
							|  |  |  |         walk_up_and_down() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def z_to_point(self, z, is_input = True): | 
					
						
							|  |  |  |         if is_input: | 
					
						
							|  |  |  |             plane = self.left_plane | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             plane = self.right_plane | 
					
						
							|  |  |  |         return plane.num_pair_to_point((z.real, z.imag)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | class SizeAndRotationBreakdown(Scene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         original = TexMobject( | 
					
						
							|  |  |  |             "\\left(\\frac{1}{2}\\right)", "^{2+i}" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         split = TexMobject( | 
					
						
							|  |  |  |              "\\left(\\frac{1}{2}\\right)", "^{2}", | 
					
						
							|  |  |  |              "\\left(\\frac{1}{2}\\right)", "^{i}", | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(original[-1], split[1], split[3]).set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         VGroup(original, split).shift(UP) | 
					
						
							|  |  |  |         real_part = VGroup(*split[:2]) | 
					
						
							|  |  |  |         imag_part = VGroup(*split[2:]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         size_brace = Brace(real_part) | 
					
						
							|  |  |  |         size = size_brace.get_text("Size") | 
					
						
							|  |  |  |         rotation_brace = Brace(imag_part, UP) | 
					
						
							|  |  |  |         rotation = rotation_brace.get_text("Rotation") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add(original) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         self.play(*[ | 
					
						
							|  |  |  |             Transform(*pair) | 
					
						
							|  |  |  |             for pair in [ | 
					
						
							|  |  |  |                 (original[0], split[0]), | 
					
						
							|  |  |  |                 (original[1][0], split[1]), | 
					
						
							|  |  |  |                 (original[0].copy(), split[2]), | 
					
						
							|  |  |  |                 (VGroup(*original[1][1:]), split[3]), | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             GrowFromCenter(size_brace), | 
					
						
							|  |  |  |             Write(size) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             GrowFromCenter(rotation_brace), | 
					
						
							|  |  |  |             Write(rotation) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | class SeeLinksInDescription(TeacherStudentsScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.teacher_says("""
 | 
					
						
							|  |  |  |             See links in the | 
					
						
							|  |  |  |             description for more. | 
					
						
							|  |  |  |         """)
 | 
					
						
							|  |  |  |         self.play(*it.chain(*[ | 
					
						
							|  |  |  |             [pi.change_mode, "hooray", pi.look, DOWN] | 
					
						
							|  |  |  |             for pi in self.get_students() | 
					
						
							|  |  |  |         ])) | 
					
						
							|  |  |  |         self.random_blink(3) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ShowMultiplicationOfRealAndImaginaryExponentialParts(FromRealToComplex): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.break_up_exponent() | 
					
						
							|  |  |  |         self.show_multiplication() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def break_up_exponent(self): | 
					
						
							|  |  |  |         original = TexMobject( | 
					
						
							|  |  |  |             "\\left(\\frac{1}{2}\\right)", "^{2+i}" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         split = TexMobject( | 
					
						
							|  |  |  |              "\\left(\\frac{1}{2}\\right)", "^{2}", | 
					
						
							|  |  |  |              "\\left(\\frac{1}{2}\\right)", "^{i}", | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(original[-1], split[1], split[3]).set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         VGroup(original, split).to_corner(UP+LEFT) | 
					
						
							|  |  |  |         rect = BackgroundRectangle(split) | 
					
						
							|  |  |  |         real_part = VGroup(*split[:2]) | 
					
						
							|  |  |  |         imag_part = VGroup(*split[2:]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add(rect, original) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play(*[ | 
					
						
							|  |  |  |             Transform(*pair) | 
					
						
							|  |  |  |             for pair in [ | 
					
						
							|  |  |  |                 (original[0], split[0]), | 
					
						
							|  |  |  |                 (original[1][0], split[1]), | 
					
						
							|  |  |  |                 (original[0].copy(), split[2]), | 
					
						
							|  |  |  |                 (VGroup(*original[1][1:]), split[3]), | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         self.remove(*self.get_mobjects_from_last_animation()) | 
					
						
							|  |  |  |         self.add(real_part, imag_part) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.real_part = real_part | 
					
						
							|  |  |  |         self.imag_part = imag_part | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def show_multiplication(self): | 
					
						
							|  |  |  |         real_part = self.real_part.copy() | 
					
						
							|  |  |  |         imag_part = self.imag_part.copy() | 
					
						
							|  |  |  |         for part in real_part, imag_part: | 
					
						
							|  |  |  |             part.add_to_back(BackgroundRectangle(part)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fourth_point = self.z_to_point(0.25) | 
					
						
							|  |  |  |         fourth_line = Line(ORIGIN, fourth_point) | 
					
						
							|  |  |  |         brace = Brace(fourth_line, UP, buff = SMALL_BUFF) | 
					
						
							|  |  |  |         fourth_dot = Dot(fourth_point) | 
					
						
							|  |  |  |         fourth_group = VGroup(fourth_line, brace, fourth_dot) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         fourth_group.set_color(RED) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         circle = Circle(radius = 2, color = MAROON_B) | 
					
						
							|  |  |  |         circle.fade(0.3) | 
					
						
							|  |  |  |         imag_power_point = self.z_to_point(0.5**complex(0, 1)) | 
					
						
							|  |  |  |         imag_power_dot = Dot(imag_power_point) | 
					
						
							|  |  |  |         imag_power_line = Line(ORIGIN, imag_power_point) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(imag_power_dot, imag_power_line).set_color(MAROON_B) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         full_power_tex = TexMobject( | 
					
						
							|  |  |  |             "\\left(\\frac{1}{2}\\right)", "^{2+i}" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         full_power_tex[-1].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         full_power_tex.add_background_rectangle() | 
					
						
							|  |  |  |         full_power_tex.scale(0.7) | 
					
						
							|  |  |  |         full_power_tex.next_to( | 
					
						
							|  |  |  |             0.5*self.z_to_point(0.5**complex(2, 1)), | 
					
						
							|  |  |  |             UP+RIGHT | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             real_part.scale, 0.7, | 
					
						
							|  |  |  |             real_part.next_to, brace, UP, SMALL_BUFF, LEFT, | 
					
						
							|  |  |  |             ShowCreation(fourth_dot) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             GrowFromCenter(brace), | 
					
						
							|  |  |  |             ShowCreation(fourth_line), | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             imag_part.scale, 0.7, | 
					
						
							|  |  |  |             imag_part.next_to, imag_power_dot, DOWN+RIGHT, SMALL_BUFF, | 
					
						
							|  |  |  |             ShowCreation(imag_power_dot) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(ShowCreation(circle), Animation(imag_power_dot)) | 
					
						
							|  |  |  |         self.play(ShowCreation(imag_power_line)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             fourth_group.rotate, imag_power_line.get_angle() | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         real_part.generate_target() | 
					
						
							|  |  |  |         imag_part.generate_target() | 
					
						
							|  |  |  |         real_part.target.next_to(brace, UP+RIGHT, buff = 0) | 
					
						
							|  |  |  |         imag_part.target.next_to(real_part.target, buff = 0) | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(MoveToTarget, [real_part, imag_part]))) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | class ComplexFunctionsAsTransformations(ComplexTransformationScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_title() | 
					
						
							|  |  |  |         input_dots, output_dots, arrows = self.get_dots() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(FadeIn( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             input_dots, | 
					
						
							|  |  |  |             run_time = 2, | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |             submobject_mode = "lagged_start" | 
					
						
							|  |  |  |         )) | 
					
						
							|  |  |  |         for in_dot, out_dot, arrow in zip(input_dots, output_dots, arrows): | 
					
						
							|  |  |  |             self.play( | 
					
						
							|  |  |  |                 Transform(in_dot.copy(), out_dot), | 
					
						
							|  |  |  |                 ShowCreation(arrow) | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() | 
					
						
							|  |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_title(self): | 
					
						
							|  |  |  |         title = TextMobject("Complex functions as transformations") | 
					
						
							|  |  |  |         title.add_background_rectangle() | 
					
						
							|  |  |  |         title.to_edge(UP) | 
					
						
							|  |  |  |         self.add(title) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_dots(self): | 
					
						
							|  |  |  |         input_points = [ | 
					
						
							|  |  |  |             RIGHT+2*UP, | 
					
						
							|  |  |  |             4*RIGHT+DOWN, | 
					
						
							|  |  |  |             2*LEFT+2*UP, | 
					
						
							|  |  |  |             LEFT+DOWN, | 
					
						
							|  |  |  |             6*LEFT+DOWN, | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         output_nudges = [ | 
					
						
							|  |  |  |             DOWN+RIGHT, | 
					
						
							|  |  |  |             2*UP+RIGHT, | 
					
						
							|  |  |  |             2*RIGHT+2*DOWN, | 
					
						
							|  |  |  |             2*RIGHT+DOWN, | 
					
						
							|  |  |  |             RIGHT+2*UP, | 
					
						
							|  |  |  |         ] | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         input_dots = VGroup(*list(map(Dot, input_points))) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         input_dots.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         output_dots = VGroup(*[ | 
					
						
							|  |  |  |             Dot(ip + on) | 
					
						
							|  |  |  |             for ip, on in zip(input_points, output_nudges) | 
					
						
							|  |  |  |         ]) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         output_dots.set_color(MAROON_B) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         arrows = VGroup(*[ | 
					
						
							|  |  |  |             Arrow(in_dot, out_dot, buff = 0.1, color = WHITE) | 
					
						
							|  |  |  |             for in_dot, out_dot, in zip(input_dots, output_dots) | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         for i, dot in enumerate(input_dots): | 
					
						
							|  |  |  |             label = TexMobject("s_%d"%i) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             label.set_color(dot.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |             label.next_to(dot, DOWN+LEFT, buff = SMALL_BUFF) | 
					
						
							|  |  |  |             dot.add(label) | 
					
						
							|  |  |  |         for i, dot in enumerate(output_dots): | 
					
						
							|  |  |  |             label = TexMobject("f(s_%d)"%i) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             label.set_color(dot.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |             label.next_to(dot, UP+RIGHT, buff = SMALL_BUFF) | 
					
						
							|  |  |  |             dot.add(label) | 
					
						
							|  |  |  |         return input_dots, output_dots, arrows | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | class VisualizingSSquared(ComplexTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "num_anchors_to_add_per_line" : 100, | 
					
						
							|  |  |  |         "horiz_end_color" : GOLD, | 
					
						
							|  |  |  |         "y_min" : 0, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_title() | 
					
						
							|  |  |  |         self.plug_in_specific_values() | 
					
						
							|  |  |  |         self.show_transformation() | 
					
						
							|  |  |  |         self.comment_on_two_dimensions() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_title(self): | 
					
						
							|  |  |  |         title = TexMobject("f(", "s", ") = ", "s", "^2") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         title.set_color_by_tex("s", YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         title.add_background_rectangle() | 
					
						
							|  |  |  |         title.scale(1.5) | 
					
						
							|  |  |  |         title.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         self.play(Write(title)) | 
					
						
							|  |  |  |         self.add_foreground_mobject(title) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.title = title | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def plug_in_specific_values(self): | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         inputs = list(map(complex, [2, -1, complex(0, 1)])) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         input_dots  = VGroup(*[ | 
					
						
							|  |  |  |             Dot(self.z_to_point(z), color = YELLOW) | 
					
						
							|  |  |  |             for z in inputs | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         output_dots = VGroup(*[ | 
					
						
							|  |  |  |             Dot(self.z_to_point(z**2), color = BLUE) | 
					
						
							|  |  |  |             for z in inputs | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         arrows = VGroup() | 
					
						
							|  |  |  |         VGroup(*[ | 
					
						
							|  |  |  |             ParametricFunction( | 
					
						
							|  |  |  |                 lambda t : self.z_to_point(z**(1.1+0.8*t)) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             for z in inputs | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         for z, dot in zip(inputs, input_dots): | 
					
						
							|  |  |  |             path = ParametricFunction( | 
					
						
							|  |  |  |                 lambda t : self.z_to_point(z**(1+t)) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             dot.path = path | 
					
						
							|  |  |  |             arrow = ParametricFunction( | 
					
						
							|  |  |  |                 lambda t : self.z_to_point(z**(1.1+0.8*t)) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             stand_in_arrow = Arrow( | 
					
						
							|  |  |  |                 arrow.points[-2], arrow.points[-1], | 
					
						
							|  |  |  |                 tip_length = 0.2 | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             arrow.add(stand_in_arrow.tip) | 
					
						
							|  |  |  |             arrows.add(arrow) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         arrows.set_color(WHITE) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for input_dot, output_dot, arrow in zip(input_dots, output_dots, arrows): | 
					
						
							|  |  |  |             input_dot.save_state() | 
					
						
							|  |  |  |             input_dot.move_to(self.title[1][1]) | 
					
						
							|  |  |  |             input_dot.set_fill(opacity = 0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             self.play(input_dot.restore) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |             self.play(ShowCreation(arrow)) | 
					
						
							|  |  |  |             self.play(ShowCreation(output_dot)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.add_foreground_mobjects(arrows, output_dots, input_dots) | 
					
						
							|  |  |  |         self.input_dots = input_dots | 
					
						
							|  |  |  |         self.output_dots = output_dots | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |     def add_transformable_plane(self, **kwargs): | 
					
						
							|  |  |  |         ComplexTransformationScene.add_transformable_plane(self, **kwargs) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.plane.next_to(ORIGIN, UP, buff = 0.01) | 
					
						
							|  |  |  |         self.plane.add(self.plane.copy().rotate(np.pi, RIGHT)) | 
					
						
							|  |  |  |         self.plane.add( | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             Line(ORIGIN, FRAME_X_RADIUS*RIGHT, color = self.horiz_end_color), | 
					
						
							|  |  |  |             Line(ORIGIN, FRAME_X_RADIUS*LEFT, color = self.horiz_end_color), | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.add(self.plane) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def show_transformation(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play(ShowCreation(self.plane, run_time = 3)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.apply_complex_homotopy( | 
					
						
							|  |  |  |             lambda z, t : z**(1+t), | 
					
						
							|  |  |  |             added_anims = [ | 
					
						
							|  |  |  |                 MoveAlongPath(dot, dot.path, run_time = 5) | 
					
						
							|  |  |  |                 for dot in self.input_dots | 
					
						
							|  |  |  |             ], | 
					
						
							|  |  |  |             run_time = 5 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def comment_on_two_dimensions(self): | 
					
						
							|  |  |  |         morty = Mortimer().flip() | 
					
						
							|  |  |  |         morty.scale(0.7) | 
					
						
							|  |  |  |         morty.to_corner(DOWN+LEFT) | 
					
						
							| 
									
										
										
										
											2017-02-14 14:00:50 -08:00
										 |  |  |         bubble = morty.get_bubble(SpeechBubble, height = 2, width = 4) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         bubble.set_fill(BLACK, opacity = 0.9) | 
					
						
							|  |  |  |         bubble.write("""
 | 
					
						
							|  |  |  |             It all happens | 
					
						
							|  |  |  |             in two dimensions! | 
					
						
							|  |  |  |         """)
 | 
					
						
							|  |  |  |         self.foreground_mobjects = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(FadeIn(morty)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             morty.change_mode, "hooray", | 
					
						
							|  |  |  |             ShowCreation(bubble), | 
					
						
							|  |  |  |             Write(bubble.content), | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(morty)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class ShowZetaOnHalfPlane(ZetaTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "x_min" : 1, | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |         "x_max" : int(FRAME_X_RADIUS+2), | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_title() | 
					
						
							|  |  |  |         self.initial_transformation() | 
					
						
							|  |  |  |         self.react_to_transformation() | 
					
						
							|  |  |  |         self.show_cutoff() | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         self.set_color_i_line() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.show_continuation() | 
					
						
							|  |  |  |         self.emphsize_sum_doesnt_make_sense() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_title(self): | 
					
						
							|  |  |  |         zeta = TexMobject( | 
					
						
							|  |  |  |             "\\zeta(", "s", ")=", | 
					
						
							|  |  |  |             *[ | 
					
						
							|  |  |  |                 "\\frac{1}{%d^s} + "%d | 
					
						
							|  |  |  |                 for d in range(1, 5) | 
					
						
							|  |  |  |             ] + ["\\cdots"] | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         zeta[1].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         for mob in zeta[3:3+4]: | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             mob[-2].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         zeta.add_background_rectangle() | 
					
						
							|  |  |  |         zeta.scale(0.8) | 
					
						
							|  |  |  |         zeta.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         self.add_foreground_mobjects(zeta) | 
					
						
							|  |  |  |         self.zeta = zeta | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def initial_transformation(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.add_extra_plane_lines_for_zeta(animate = True) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.plane.save_state() | 
					
						
							|  |  |  |         self.apply_zeta_function() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def react_to_transformation(self): | 
					
						
							|  |  |  |         morty = Mortimer().flip() | 
					
						
							|  |  |  |         morty.to_corner(DOWN+LEFT) | 
					
						
							| 
									
										
										
										
											2017-02-14 14:00:50 -08:00
										 |  |  |         bubble = morty.get_bubble(SpeechBubble) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         bubble.set_fill(BLACK, 0.5) | 
					
						
							|  |  |  |         bubble.write("\\emph{Damn}!") | 
					
						
							|  |  |  |         bubble.resize_to_content() | 
					
						
							|  |  |  |         bubble.pin_to(morty) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(FadeIn(morty)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             morty.change_mode, "surprised", | 
					
						
							|  |  |  |             ShowCreation(bubble), | 
					
						
							|  |  |  |             Write(bubble.content) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(morty)) | 
					
						
							|  |  |  |         self.play(morty.look_at, self.plane.get_top()) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             morty.look_at, self.plane.get_bottom(), | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |             *list(map(FadeOut, [bubble, bubble.content])) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(morty)) | 
					
						
							|  |  |  |         self.play(FadeOut(morty)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def show_cutoff(self): | 
					
						
							|  |  |  |         words = TextMobject("Such an abrupt stop...") | 
					
						
							|  |  |  |         words.add_background_rectangle() | 
					
						
							|  |  |  |         words.next_to(ORIGIN, UP+LEFT) | 
					
						
							|  |  |  |         words.shift(LEFT+UP) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         line = Line(*list(map(self.z_to_point, [ | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             complex(np.euler_gamma, u*FRAME_Y_RADIUS) | 
					
						
							| 
									
										
										
										
											2018-06-02 08:59:26 -04:00
										 |  |  |             for u in (1, -1) | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         ]))) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         line.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         arrows = [ | 
					
						
							|  |  |  |             Arrow(words.get_right(), point) | 
					
						
							|  |  |  |             for point in line.get_start_and_end() | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(Write(words, run_time = 2)) | 
					
						
							|  |  |  |         self.play(ShowCreation(arrows[0])) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             Transform(*arrows), | 
					
						
							|  |  |  |             ShowCreation(line), | 
					
						
							|  |  |  |             run_time = 2 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(FadeOut(arrows[0])) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(FadeOut, [words, line]))) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |     def set_color_i_line(self): | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         right_i_lines, left_i_lines = [ | 
					
						
							|  |  |  |             VGroup(*[ | 
					
						
							|  |  |  |                 Line( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |                     vert_vect+RIGHT, | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |                     vert_vect+(FRAME_X_RADIUS+1)*horiz_vect | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2018-06-02 08:59:26 -04:00
										 |  |  |                 for vert_vect in (UP, DOWN) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |             ]) | 
					
						
							| 
									
										
										
										
											2018-06-02 08:59:26 -04:00
										 |  |  |             for horiz_vect in (RIGHT, LEFT) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         ] | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         right_i_lines.set_color(YELLOW) | 
					
						
							|  |  |  |         left_i_lines.set_color(BLUE) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         for lines in right_i_lines, left_i_lines: | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             self.prepare_for_transformation(lines) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.restore_mobjects(self.plane) | 
					
						
							|  |  |  |         self.plane.add(*right_i_lines) | 
					
						
							|  |  |  |         colored_plane = self.plane.copy() | 
					
						
							|  |  |  |         right_i_lines.set_stroke(width = 0) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             self.plane.set_stroke, GREY, 1, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         right_i_lines.set_stroke(YELLOW, width = 3) | 
					
						
							|  |  |  |         self.play(ShowCreation(right_i_lines)) | 
					
						
							|  |  |  |         self.plane.save_state() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.apply_zeta_function() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         left_i_lines.save_state() | 
					
						
							|  |  |  |         left_i_lines.apply_complex_function(zeta) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play(ShowCreation(left_i_lines, run_time = 5)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.restore_mobjects(self.plane, left_i_lines) | 
					
						
							|  |  |  |         self.play(Transform(self.plane, colored_plane)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         self.left_i_lines = left_i_lines | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def show_continuation(self): | 
					
						
							|  |  |  |         reflected_plane = self.get_reflected_plane() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play(ShowCreation(reflected_plane, run_time = 2)) | 
					
						
							|  |  |  |         self.plane.add(reflected_plane) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         self.remove(self.left_i_lines) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.apply_zeta_function() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play(ShowCreation( | 
					
						
							|  |  |  |             reflected_plane, | 
					
						
							|  |  |  |             run_time = 6, | 
					
						
							|  |  |  |             rate_func = lambda t : 1-there_and_back(t) | 
					
						
							|  |  |  |         )) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def emphsize_sum_doesnt_make_sense(self): | 
					
						
							|  |  |  |         brace = Brace(VGroup(*self.zeta[1][3:])) | 
					
						
							|  |  |  |         words = brace.get_text("""
 | 
					
						
							|  |  |  |             Still fails to converge | 
					
						
							|  |  |  |             when Re$(s) < 1$ | 
					
						
							|  |  |  |         """, buff = SMALL_BUFF)
 | 
					
						
							|  |  |  |         words.add_background_rectangle() | 
					
						
							|  |  |  |         words.scale_in_place(0.8) | 
					
						
							|  |  |  |         divergent_sum = TexMobject("1+2+3+4+\\cdots") | 
					
						
							|  |  |  |         divergent_sum.next_to(ORIGIN, UP) | 
					
						
							|  |  |  |         divergent_sum.to_edge(LEFT) | 
					
						
							|  |  |  |         divergent_sum.add_background_rectangle() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             GrowFromCenter(brace), | 
					
						
							|  |  |  |             Write(words) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play(Write(divergent_sum)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def restore_mobjects(self, *mobjects): | 
					
						
							|  |  |  |         self.play(*it.chain(*[ | 
					
						
							|  |  |  |             [m.restore, m.make_smooth] | 
					
						
							|  |  |  |             for m in  mobjects | 
					
						
							|  |  |  |         ]), run_time = 2) | 
					
						
							|  |  |  |         for m in mobjects: | 
					
						
							|  |  |  |             self.remove(m) | 
					
						
							|  |  |  |             m.restore() | 
					
						
							|  |  |  |             self.add(m) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ShowConditionalDefinition(Scene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         zeta = TexMobject("\\zeta(s)=") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         zeta[2].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         sigma = TexMobject("\\sum_{n=1}^\\infty \\frac{1}{n^s}") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         sigma[-1].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         something_else = TextMobject("Something else...") | 
					
						
							|  |  |  |         conditions = VGroup(*[ | 
					
						
							|  |  |  |             TextMobject("if Re$(s) %s 1$"%s) | 
					
						
							| 
									
										
										
										
											2018-06-02 08:59:26 -04:00
										 |  |  |             for s in (">", "\\le") | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         ]) | 
					
						
							|  |  |  |         definitions = VGroup(sigma, something_else) | 
					
						
							| 
									
										
										
										
											2017-01-25 16:40:59 -08:00
										 |  |  |         definitions.arrange_submobjects(DOWN, buff = MED_LARGE_BUFF, aligned_edge = LEFT) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         conditions.arrange_submobjects(DOWN, buff = LARGE_BUFF) | 
					
						
							|  |  |  |         definitions.shift(2*LEFT+2*UP) | 
					
						
							|  |  |  |         conditions.next_to(definitions, RIGHT, buff = LARGE_BUFF, aligned_edge = DOWN) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         brace = Brace(definitions, LEFT) | 
					
						
							|  |  |  |         zeta.next_to(brace, LEFT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         sigma.save_state() | 
					
						
							|  |  |  |         sigma.next_to(zeta) | 
					
						
							|  |  |  |         self.add(zeta, sigma) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             sigma.restore, | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |             GrowFromCenter(brace), | 
					
						
							|  |  |  |             FadeIn(something_else) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         self.play(Write(conditions)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         underbrace = Brace(something_else) | 
					
						
							|  |  |  |         question = underbrace.get_text("""
 | 
					
						
							|  |  |  |             What to put here? | 
					
						
							|  |  |  |         """)
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(underbrace, question).set_color(GREEN_B) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             GrowFromCenter(underbrace), | 
					
						
							|  |  |  |             Write(question), | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             something_else.set_color, GREEN_B | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class SquiggleOnExtensions(ZetaTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "x_min" : 1, | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |         "x_max" : int(FRAME_X_RADIUS+2), | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.show_negative_one() | 
					
						
							|  |  |  |         self.cycle_through_options() | 
					
						
							|  |  |  |         self.lock_into_place() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def show_negative_one(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         thin_plane = self.plane.copy() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         thin_plane.add(self.get_reflected_plane()) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.remove(self.plane) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         self.add_extra_plane_lines_for_zeta() | 
					
						
							|  |  |  |         reflected_plane = self.get_reflected_plane() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.plane.add(reflected_plane) | 
					
						
							|  |  |  |         self.remove(self.plane) | 
					
						
							|  |  |  |         self.add(thin_plane) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         dot = self.note_point(-1, "-1") | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(self.plane, run_time = 2), | 
					
						
							|  |  |  |             Animation(dot), | 
					
						
							|  |  |  |             run_time = 2 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.remove(thin_plane) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         self.apply_zeta_function(added_anims = [ | 
					
						
							|  |  |  |             ApplyMethod( | 
					
						
							|  |  |  |                 dot.move_to, self.z_to_point(-1./12), | 
					
						
							|  |  |  |                 run_time = 5 | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |         ]) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         dot_to_remove = self.note_point(-1./12, "-\\frac{1}{12}") | 
					
						
							|  |  |  |         self.remove(dot_to_remove) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         self.left_plane = reflected_plane | 
					
						
							|  |  |  |         self.dot = dot | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def note_point(self, z, label_tex): | 
					
						
							|  |  |  |         dot = Dot(self.z_to_point(z)) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         dot.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         label = TexMobject(label_tex) | 
					
						
							|  |  |  |         label.add_background_rectangle() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         label.next_to(dot, UP+LEFT, buff = SMALL_BUFF) | 
					
						
							|  |  |  |         label.shift(LEFT) | 
					
						
							|  |  |  |         arrow = Arrow(label.get_right(), dot, buff = SMALL_BUFF) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.play(Write(label, run_time = 1)) | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(ShowCreation, [arrow, dot]))) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(FadeOut, [arrow, label]))) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         return dot | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def cycle_through_options(self): | 
					
						
							|  |  |  |         gamma = np.euler_gamma | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         def shear(point): | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |             x, y, z = point | 
					
						
							|  |  |  |             return np.array([ | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |                 x, | 
					
						
							|  |  |  |                 y+0.25*(1-x)**2, | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |                 0 | 
					
						
							|  |  |  |             ]) | 
					
						
							|  |  |  |         def mixed_scalar_func(point): | 
					
						
							|  |  |  |             x, y, z = point | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             scalar = 1 + (gamma-x)/(gamma+FRAME_X_RADIUS) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |             return np.array([ | 
					
						
							|  |  |  |                 (scalar**2)*x, | 
					
						
							|  |  |  |                 (scalar**3)*y, | 
					
						
							|  |  |  |                 0 | 
					
						
							|  |  |  |             ]) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         def alt_mixed_scalar_func(point): | 
					
						
							|  |  |  |             x, y, z = point | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             scalar = 1 + (gamma-x)/(gamma+FRAME_X_RADIUS) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             return np.array([ | 
					
						
							|  |  |  |                 (scalar**5)*x, | 
					
						
							|  |  |  |                 (scalar**2)*y, | 
					
						
							|  |  |  |                 0 | 
					
						
							|  |  |  |             ]) | 
					
						
							|  |  |  |         def sinusoidal_func(point): | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |             x, y, z = point | 
					
						
							|  |  |  |             freq = np.pi/gamma | 
					
						
							|  |  |  |             return np.array([ | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |                 x-0.2*np.sin(x*freq)*np.sin(y), | 
					
						
							|  |  |  |                 y-0.2*np.sin(x*freq)*np.sin(y), | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |                 0 | 
					
						
							|  |  |  |             ]) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         funcs = [ | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             shear, | 
					
						
							|  |  |  |             mixed_scalar_func, | 
					
						
							|  |  |  |             alt_mixed_scalar_func, | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             sinusoidal_func, | 
					
						
							|  |  |  |         ] | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         for mob in self.left_plane.family_members_with_points(): | 
					
						
							|  |  |  |             if np.all(np.abs(mob.points[:,1]) < 0.1): | 
					
						
							|  |  |  |                 self.left_plane.remove(mob) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         new_left_planes = [ | 
					
						
							|  |  |  |             self.left_plane.copy().apply_function(func) | 
					
						
							|  |  |  |             for func in funcs | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         new_dots = [ | 
					
						
							|  |  |  |             self.dot.copy().move_to(func(self.dot.get_center())) | 
					
						
							|  |  |  |             for func in funcs | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         self.left_plane.save_state() | 
					
						
							|  |  |  |         for plane, dot in zip(new_left_planes, new_dots): | 
					
						
							|  |  |  |             self.play( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |                 Transform(self.left_plane, plane), | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |                 Transform(self.dot, dot), | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |                 run_time = 3 | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play(FadeOut(self.dot)) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         #Squiggle on example | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         self.play(FadeOut(self.left_plane)) | 
					
						
							|  |  |  |         self.play(ShowCreation( | 
					
						
							|  |  |  |             self.left_plane, | 
					
						
							|  |  |  |             run_time = 5, | 
					
						
							|  |  |  |             rate_func = None | 
					
						
							|  |  |  |         )) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def lock_into_place(self): | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         words = TextMobject( | 
					
						
							|  |  |  |             """Only one extension
 | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             has a """,
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             "\\emph{derivative}", | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             "everywhere", | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             alignment = "" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         words.to_corner(UP+LEFT) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         words.set_color_by_tex("\\emph{derivative}", YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-03 19:06:50 -08:00
										 |  |  |         words.add_background_rectangle() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(Write(words)) | 
					
						
							|  |  |  |         self.add_foreground_mobjects(words) | 
					
						
							|  |  |  |         self.play(self.left_plane.restore) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  | class DontKnowDerivatives(TeacherStudentsScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.student_says( | 
					
						
							|  |  |  |             """
 | 
					
						
							|  |  |  |             You said we don't | 
					
						
							|  |  |  |             need derivatives! | 
					
						
							|  |  |  |             """,
 | 
					
						
							|  |  |  |             target_mode = "pleading" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.random_blink(2) | 
					
						
							|  |  |  |         self.student_says( | 
					
						
							|  |  |  |             """
 | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             I get $\\frac{df}{dx}$, just not | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             for complex functions | 
					
						
							|  |  |  |             """,
 | 
					
						
							|  |  |  |             target_mode = "confused", | 
					
						
							|  |  |  |             student_index = 2 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.random_blink(2) | 
					
						
							|  |  |  |         self.teacher_says( | 
					
						
							|  |  |  |             """
 | 
					
						
							|  |  |  |             Luckily, there's a purely | 
					
						
							|  |  |  |             geometric intuition here. | 
					
						
							|  |  |  |             """,
 | 
					
						
							|  |  |  |             target_mode = "hooray" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.change_student_modes(*["happy"]*3) | 
					
						
							|  |  |  |         self.random_blink(3) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class IntroduceAnglePreservation(VisualizingSSquared): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "num_anchors_to_add_per_line" : 50, | 
					
						
							|  |  |  |         "use_homotopy" : True, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_title() | 
					
						
							|  |  |  |         self.show_initial_transformation() | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         self.talk_about_derivative() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.cycle_through_line_pairs() | 
					
						
							|  |  |  |         self.note_grid_lines() | 
					
						
							|  |  |  |         self.name_analytic() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_title(self): | 
					
						
							|  |  |  |         title = TexMobject("f(", "s", ")=", "s", "^2") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         title.set_color_by_tex("s", YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         title.scale(1.5) | 
					
						
							|  |  |  |         title.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         title.add_background_rectangle() | 
					
						
							|  |  |  |         self.title = title | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.play(Write(title)) | 
					
						
							|  |  |  |         self.add_foreground_mobjects(title) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def show_initial_transformation(self): | 
					
						
							|  |  |  |         self.apply_function() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.reset() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |     def talk_about_derivative(self): | 
					
						
							|  |  |  |         randy = Randolph().scale(0.8) | 
					
						
							|  |  |  |         randy.to_corner(DOWN+LEFT) | 
					
						
							|  |  |  |         morty = Mortimer() | 
					
						
							|  |  |  |         morty.to_corner(DOWN+RIGHT) | 
					
						
							|  |  |  |         randy.make_eye_contact(morty) | 
					
						
							|  |  |  |         for pi, words in (randy, "$f'(s) = 2s$"), (morty, "Here's some \\\\ related geometry..."): | 
					
						
							| 
									
										
										
										
											2017-02-14 14:00:50 -08:00
										 |  |  |             pi.bubble = pi.get_bubble(SpeechBubble) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |             pi.bubble.set_fill(BLACK, opacity = 0.7) | 
					
						
							|  |  |  |             pi.bubble.write(words) | 
					
						
							|  |  |  |             pi.bubble.resize_to_content() | 
					
						
							|  |  |  |             pi.bubble.pin_to(pi) | 
					
						
							|  |  |  |         for index in 3, 7: | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             randy.bubble.content[index].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(FadeIn, [randy, morty]))) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             randy.change_mode, "speaking", | 
					
						
							|  |  |  |             ShowCreation(randy.bubble), | 
					
						
							|  |  |  |             Write(randy.bubble.content) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(morty)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             morty.change_mode, "speaking", | 
					
						
							|  |  |  |             randy.change_mode, "pondering", | 
					
						
							|  |  |  |             ShowCreation(morty.bubble), | 
					
						
							|  |  |  |             Write(morty.bubble.content), | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(FadeOut, [ | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             randy, morty, | 
					
						
							|  |  |  |             randy.bubble, randy.bubble.content, | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |             morty.bubble, morty.bubble.content, | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         ]))) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |     def cycle_through_line_pairs(self): | 
					
						
							|  |  |  |         line_pairs = [ | 
					
						
							|  |  |  |             ( | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |                 Line(3*DOWN+3*RIGHT, 2*UP), | 
					
						
							|  |  |  |                 Line(DOWN+RIGHT, 3*UP+4*RIGHT) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             ), | 
					
						
							|  |  |  |             ( | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |                 Line(RIGHT+3.5*DOWN, RIGHT+2.5*UP), | 
					
						
							|  |  |  |                 Line(3*LEFT+0.5*UP, 3*RIGHT+0.5*UP), | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             ), | 
					
						
							|  |  |  |             ( | 
					
						
							|  |  |  |                 Line(4*RIGHT+4*DOWN, RIGHT+2*UP), | 
					
						
							|  |  |  |                 Line(4*DOWN+RIGHT, 2*UP+2*RIGHT) | 
					
						
							|  |  |  |             ), | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         for lines in line_pairs: | 
					
						
							|  |  |  |             self.show_angle_preservation_between_lines(*lines) | 
					
						
							|  |  |  |             self.reset() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def note_grid_lines(self): | 
					
						
							|  |  |  |         intersection_inputs = [ | 
					
						
							|  |  |  |             complex(x, y) | 
					
						
							|  |  |  |             for x in np.arange(-5, 5, 0.5) | 
					
						
							|  |  |  |             for y in np.arange(0, 3, 0.5) | 
					
						
							|  |  |  |             if not (x <= 0 and y == 0) | 
					
						
							|  |  |  |         ] | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         brackets = VGroup(*list(map( | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             self.get_right_angle_bracket, | 
					
						
							|  |  |  |             intersection_inputs | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         ))) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.apply_function() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(brackets, run_time = 5), | 
					
						
							|  |  |  |             Animation(self.plane) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |     def name_analytic(self): | 
					
						
							|  |  |  |         equiv = TextMobject("``Analytic'' $\\Leftrightarrow$ Angle-preserving") | 
					
						
							|  |  |  |         kind_of = TextMobject("...kind of") | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |         for text in equiv, kind_of: | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             text.scale(1.2) | 
					
						
							|  |  |  |             text.add_background_rectangle() | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         equiv.set_color(YELLOW) | 
					
						
							|  |  |  |         kind_of.set_color(RED) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         kind_of.next_to(equiv, RIGHT) | 
					
						
							|  |  |  |         VGroup(equiv, kind_of).next_to(ORIGIN, UP, buff = 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(Write(equiv)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play(Write(kind_of, run_time = 1)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def reset(self, faded = True): | 
					
						
							|  |  |  |         self.play(FadeOut(self.plane)) | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         if faded: | 
					
						
							|  |  |  |             self.plane.fade() | 
					
						
							|  |  |  |         self.play(FadeIn(self.plane)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def apply_function(self, **kwargs): | 
					
						
							|  |  |  |         if self.use_homotopy: | 
					
						
							|  |  |  |             self.apply_complex_homotopy( | 
					
						
							|  |  |  |                 lambda z, t : z**(1+t), | 
					
						
							|  |  |  |                 run_time = 5, | 
					
						
							|  |  |  |                 **kwargs | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.apply_complex_function( | 
					
						
							|  |  |  |                 lambda z : z**2, | 
					
						
							|  |  |  |                 **kwargs | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def show_angle_preservation_between_lines(self, *lines): | 
					
						
							|  |  |  |         R2_endpoints = [ | 
					
						
							|  |  |  |             [l.get_start()[:2], l.get_end()[:2]] | 
					
						
							|  |  |  |             for l in lines | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         R2_intersection_point = intersection(*R2_endpoints) | 
					
						
							|  |  |  |         intersection_point = np.array(list(R2_intersection_point) + [0]) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         angle1, angle2 = [l.get_angle() for l in lines] | 
					
						
							|  |  |  |         arc = Arc( | 
					
						
							|  |  |  |             start_angle = angle1, | 
					
						
							|  |  |  |             angle = angle2-angle1, | 
					
						
							|  |  |  |             radius = 0.4, | 
					
						
							|  |  |  |             color = YELLOW | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         arc.shift(intersection_point) | 
					
						
							|  |  |  |         arc.insert_n_anchor_points(10) | 
					
						
							|  |  |  |         arc.generate_target() | 
					
						
							|  |  |  |         input_z = complex(*arc.get_center()[:2]) | 
					
						
							|  |  |  |         scale_factor = abs(2*input_z) | 
					
						
							|  |  |  |         arc.target.scale_about_point(1./scale_factor, intersection_point) | 
					
						
							|  |  |  |         arc.target.apply_complex_function(lambda z : z**2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         angle_tex = TexMobject( | 
					
						
							|  |  |  |             "%d^\\circ"%abs(int((angle2-angle1)*180/np.pi)) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         angle_tex.set_color(arc.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         angle_tex.add_background_rectangle() | 
					
						
							|  |  |  |         self.put_angle_tex_next_to_arc(angle_tex, arc) | 
					
						
							|  |  |  |         angle_arrow = Arrow( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             angle_tex, arc, | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             color = arc.get_color(), | 
					
						
							|  |  |  |             buff = 0.1, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         angle_group = VGroup(angle_tex, angle_arrow) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(ShowCreation, lines))) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             Write(angle_tex), | 
					
						
							|  |  |  |             ShowCreation(angle_arrow), | 
					
						
							|  |  |  |             ShowCreation(arc) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play(FadeOut(angle_group)) | 
					
						
							|  |  |  |         self.plane.add(*lines) | 
					
						
							|  |  |  |         self.apply_function(added_anims = [ | 
					
						
							|  |  |  |             MoveToTarget(arc, run_time = 5) | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         self.put_angle_tex_next_to_arc(angle_tex, arc) | 
					
						
							|  |  |  |         arrow = Arrow(angle_tex, arc, buff = 0.1) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         arrow.set_color(arc.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             Write(angle_tex), | 
					
						
							|  |  |  |             ShowCreation(arrow) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(FadeOut, [arc, angle_tex, arrow]))) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |     def put_angle_tex_next_to_arc(self, angle_tex, arc): | 
					
						
							|  |  |  |         vect = arc.point_from_proportion(0.5)-interpolate( | 
					
						
							|  |  |  |             arc.points[0], arc.points[-1], 0.5 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-08-15 17:30:24 -07:00
										 |  |  |         unit_vect = vect/get_norm(vect) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         angle_tex.move_to(arc.get_center() + 1.7*unit_vect) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |     def get_right_angle_bracket(self, input_z): | 
					
						
							|  |  |  |         output_z = input_z**2 | 
					
						
							|  |  |  |         derivative = 2*input_z | 
					
						
							|  |  |  |         rotation = np.log(derivative).imag | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         brackets = VGroup( | 
					
						
							|  |  |  |             Line(RIGHT, RIGHT+UP), | 
					
						
							|  |  |  |             Line(RIGHT+UP, UP) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         brackets.scale(0.15) | 
					
						
							|  |  |  |         brackets.set_stroke(width = 2) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         brackets.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         brackets.shift(0.02*UP) ##Why??? | 
					
						
							|  |  |  |         brackets.rotate(rotation, about_point = ORIGIN) | 
					
						
							|  |  |  |         brackets.shift(self.z_to_point(output_z)) | 
					
						
							|  |  |  |         return brackets | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  | class AngleAtZeroDerivativePoints(IntroduceAnglePreservation): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "use_homotopy" : True | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_title() | 
					
						
							|  |  |  |         self.is_before_transformation = True | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.plane.fade() | 
					
						
							|  |  |  |         line = Line(3*LEFT+0.5*UP, 3*RIGHT+0.5*DOWN) | 
					
						
							|  |  |  |         self.show_angle_preservation_between_lines( | 
					
						
							|  |  |  |             line, line.copy().rotate(np.pi/5) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |     def add_title(self): | 
					
						
							|  |  |  |         title = TexMobject("f(", "s", ")=", "s", "^2") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         title.set_color_by_tex("s", YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         title.scale(1.5) | 
					
						
							|  |  |  |         title.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         title.add_background_rectangle() | 
					
						
							|  |  |  |         derivative = TexMobject("f'(0) = 0") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         derivative.set_color(RED) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         derivative.scale(1.2) | 
					
						
							|  |  |  |         derivative.add_background_rectangle() | 
					
						
							|  |  |  |         derivative.next_to(title, DOWN) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add_foreground_mobjects(title, derivative) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def put_angle_tex_next_to_arc(self, angle_tex, arc): | 
					
						
							|  |  |  |         IntroduceAnglePreservation.put_angle_tex_next_to_arc( | 
					
						
							|  |  |  |             self, angle_tex, arc | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         if not self.is_before_transformation: | 
					
						
							|  |  |  |             two_dot = TexMobject("2 \\times ") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             two_dot.set_color(angle_tex.get_color()) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             two_dot.next_to(angle_tex, LEFT, buff = SMALL_BUFF) | 
					
						
							|  |  |  |             two_dot.add_background_rectangle() | 
					
						
							|  |  |  |             center = angle_tex.get_center() | 
					
						
							|  |  |  |             angle_tex.add_to_back(two_dot) | 
					
						
							|  |  |  |             angle_tex.move_to(center) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.is_before_transformation = False | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  | class AnglePreservationAtAnyPairOfPoints(IntroduceAnglePreservation): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.plane.fade() | 
					
						
							|  |  |  |         line_pairs = self.get_line_pairs() | 
					
						
							|  |  |  |         line_pair = line_pairs[0] | 
					
						
							|  |  |  |         for target_pair in line_pairs[1:]: | 
					
						
							|  |  |  |             self.play(Transform( | 
					
						
							|  |  |  |                 line_pair, target_pair, | 
					
						
							|  |  |  |                 run_time = 2, | 
					
						
							|  |  |  |                 path_arc = np.pi | 
					
						
							|  |  |  |             )) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.show_angle_preservation_between_lines(*line_pair) | 
					
						
							|  |  |  |         self.show_example_analytic_functions() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_line_pairs(self): | 
					
						
							|  |  |  |         return list(it.starmap(VGroup, [ | 
					
						
							|  |  |  |             ( | 
					
						
							|  |  |  |                 Line(3*DOWN, 3*LEFT+2*UP), | 
					
						
							|  |  |  |                 Line(2*LEFT+DOWN, 3*UP+RIGHT) | 
					
						
							|  |  |  |             ), | 
					
						
							|  |  |  |             ( | 
					
						
							|  |  |  |                 Line(2*RIGHT+DOWN, 3*LEFT+2*UP), | 
					
						
							|  |  |  |                 Line(LEFT+3*DOWN, 4*RIGHT+3*UP), | 
					
						
							|  |  |  |             ), | 
					
						
							|  |  |  |             ( | 
					
						
							|  |  |  |                 Line(LEFT+3*DOWN, LEFT+3*UP), | 
					
						
							|  |  |  |                 Line(5*LEFT+UP, 3*RIGHT+UP) | 
					
						
							|  |  |  |             ), | 
					
						
							|  |  |  |             ( | 
					
						
							|  |  |  |                 Line(4*RIGHT+3*DOWN, RIGHT+2*UP), | 
					
						
							|  |  |  |                 Line(3*DOWN+RIGHT, 2*UP+2*RIGHT) | 
					
						
							|  |  |  |             ), | 
					
						
							|  |  |  |         ])) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |     def show_example_analytic_functions(self): | 
					
						
							|  |  |  |         words = TextMobject("Examples of analytic functions:") | 
					
						
							|  |  |  |         words.shift(2*UP) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         words.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         words.add_background_rectangle() | 
					
						
							|  |  |  |         words.next_to(UP, UP).to_edge(LEFT) | 
					
						
							|  |  |  |         functions = TextMobject( | 
					
						
							|  |  |  |             "$e^x$, ", | 
					
						
							|  |  |  |             "$\\sin(x)$, ", | 
					
						
							|  |  |  |             "any polynomial, " | 
					
						
							|  |  |  |             "$\\log(x)$, ", | 
					
						
							|  |  |  |             "\\dots", | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         functions.next_to(ORIGIN, UP).to_edge(LEFT) | 
					
						
							|  |  |  |         for function in functions: | 
					
						
							|  |  |  |             function.add_to_back(BackgroundRectangle(function)) | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play(Write(words)) | 
					
						
							|  |  |  |         for function in functions: | 
					
						
							|  |  |  |             self.play(FadeIn(function)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  | class NoteZetaFunctionAnalyticOnRightHalf(ZetaTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "anchor_density" : 35, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_title() | 
					
						
							|  |  |  |         self.add_transformable_plane(animate = False) | 
					
						
							|  |  |  |         self.add_extra_plane_lines_for_zeta(animate = True) | 
					
						
							|  |  |  |         self.apply_zeta_function() | 
					
						
							|  |  |  |         self.note_right_angles() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |     def add_title(self): | 
					
						
							|  |  |  |         title = TexMobject( | 
					
						
							|  |  |  |             "\\zeta(s) = \\sum_{n=1}^\\infty \\frac{1}{n^s}" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         title[2].set_color(YELLOW) | 
					
						
							|  |  |  |         title[-1].set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         title.add_background_rectangle() | 
					
						
							|  |  |  |         title.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         self.add_foreground_mobjects(title) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def note_right_angles(self): | 
					
						
							|  |  |  |         intersection_inputs = [ | 
					
						
							|  |  |  |             complex(x, y) | 
					
						
							|  |  |  |             for x in np.arange(1+2./16, 1.4, 1./16) | 
					
						
							|  |  |  |             for y in np.arange(-0.5, 0.5, 1./16) | 
					
						
							|  |  |  |             if abs(y) > 1./16 | 
					
						
							|  |  |  |         ] | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         brackets = VGroup(*list(map( | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |             self.get_right_angle_bracket, | 
					
						
							|  |  |  |             intersection_inputs | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         ))) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         self.play(ShowCreation(brackets, run_time = 3)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |     def get_right_angle_bracket(self, input_z): | 
					
						
							|  |  |  |         output_z = zeta(input_z) | 
					
						
							|  |  |  |         derivative = d_zeta(input_z) | 
					
						
							|  |  |  |         rotation = np.log(derivative).imag | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         brackets = VGroup( | 
					
						
							|  |  |  |             Line(RIGHT, RIGHT+UP), | 
					
						
							|  |  |  |             Line(RIGHT+UP, UP) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         brackets.scale(0.1) | 
					
						
							|  |  |  |         brackets.set_stroke(width = 2) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         brackets.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-06 13:29:21 -08:00
										 |  |  |         brackets.rotate(rotation, about_point = ORIGIN) | 
					
						
							|  |  |  |         brackets.shift(self.z_to_point(output_z)) | 
					
						
							|  |  |  |         return brackets | 
					
						
							| 
									
										
										
										
											2016-12-03 15:56:33 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | class InfiniteContinuousJigsawPuzzle(ZetaTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "anchor_density" : 35, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.set_stage() | 
					
						
							|  |  |  |         self.add_title() | 
					
						
							|  |  |  |         self.show_jigsaw() | 
					
						
							|  |  |  |         self.name_analytic_continuation() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_stage(self): | 
					
						
							|  |  |  |         self.plane = self.get_dense_grid() | 
					
						
							|  |  |  |         left_plane = self.get_reflected_plane() | 
					
						
							|  |  |  |         self.plane.add(left_plane) | 
					
						
							|  |  |  |         self.apply_zeta_function(run_time = 0) | 
					
						
							|  |  |  |         self.remove(left_plane) | 
					
						
							|  |  |  |         lines_per_piece = 5 | 
					
						
							|  |  |  |         pieces = [ | 
					
						
							|  |  |  |             VGroup(*left_plane[lines_per_piece*i:lines_per_piece*(i+1)]) | 
					
						
							|  |  |  |             for i in range(len(list(left_plane))/lines_per_piece) | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         random.shuffle(pieces) | 
					
						
							|  |  |  |         self.pieces = pieces | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_title(self): | 
					
						
							|  |  |  |         title = TextMobject("Infinite ", "continuous ", "jigsaw puzzle") | 
					
						
							|  |  |  |         title.scale(1.5) | 
					
						
							|  |  |  |         title.to_edge(UP) | 
					
						
							|  |  |  |         for word in title: | 
					
						
							|  |  |  |             word.add_to_back(BackgroundRectangle(word)) | 
					
						
							|  |  |  |             self.play(FadeIn(word)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.add_foreground_mobjects(title) | 
					
						
							|  |  |  |         self.title = title | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def show_jigsaw(self): | 
					
						
							|  |  |  |         for piece in self.pieces: | 
					
						
							|  |  |  |             self.play(FadeIn(piece, run_time = 0.5)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def name_analytic_continuation(self): | 
					
						
							|  |  |  |         words = TextMobject("``Analytic continuation''") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         words.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         words.scale(1.5) | 
					
						
							|  |  |  |         words.next_to(self.title, DOWN, buff = LARGE_BUFF) | 
					
						
							|  |  |  |         words.add_background_rectangle() | 
					
						
							|  |  |  |         self.play(Write(words)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class ThatsHowZetaIsDefined(TeacherStudentsScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_zeta_definition() | 
					
						
							|  |  |  |         self.teacher_says("""
 | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             So that's how | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             $\\zeta(s)$ is defined | 
					
						
							|  |  |  |         """)
 | 
					
						
							|  |  |  |         self.change_student_modes(*["hooray"]*3) | 
					
						
							|  |  |  |         self.random_blink(2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def add_zeta_definition(self): | 
					
						
							|  |  |  |         zeta = TexMobject( | 
					
						
							|  |  |  |             "\\zeta(s) = \\sum_{n=1}^\\infty \\frac{1}{n^s}" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(zeta[2], zeta[-1]).set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         zeta.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         self.add(zeta) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ManyIntersectingLinesPreZeta(ZetaTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         "apply_zeta" : False, | 
					
						
							|  |  |  |         "lines_center" : RIGHT, | 
					
						
							|  |  |  |         "nudge_size" : 0.9, | 
					
						
							|  |  |  |         "function" : zeta, | 
					
						
							|  |  |  |         "angle" : np.pi/5, | 
					
						
							|  |  |  |         "arc_scale_factor" : 0.3, | 
					
						
							|  |  |  |         "shift_directions" : [LEFT, RIGHT], | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         self.establish_plane() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.add_title() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         line = Line(DOWN+2*LEFT, UP+2*RIGHT) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         lines = VGroup(line, line.copy().rotate(self.angle)) | 
					
						
							|  |  |  |         arc = Arc(start_angle = line.get_angle(), angle = self.angle) | 
					
						
							|  |  |  |         arc.scale(self.arc_scale_factor) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         arc.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         lines.add(arc) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         # lines.set_stroke(WHITE, width = 5) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         lines.shift(self.lines_center + self.nudge_size*RIGHT) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if self.apply_zeta: | 
					
						
							|  |  |  |             self.apply_zeta_function(run_time = 0) | 
					
						
							|  |  |  |             lines.set_stroke(width = 0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         added_anims = self.get_modified_line_anims(lines) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         for vect in self.shift_directions: | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             self.play( | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |                 ApplyMethod(lines.shift, 2*self.nudge_size*vect, path_arc = np.pi), | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |                 *added_anims, | 
					
						
							|  |  |  |                 run_time = 3 | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |     def establish_plane(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.add_extra_plane_lines_for_zeta() | 
					
						
							|  |  |  |         self.add_reflected_plane() | 
					
						
							|  |  |  |         self.plane.fade() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |     def add_title(self): | 
					
						
							|  |  |  |         if self.apply_zeta: | 
					
						
							|  |  |  |             title = TextMobject("After \\\\ transformation") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             title = TextMobject("Before \\\\ transformation") | 
					
						
							|  |  |  |         title.add_background_rectangle() | 
					
						
							|  |  |  |         title.to_edge(UP) | 
					
						
							|  |  |  |         self.add_foreground_mobjects(title) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_modified_line_anims(self, lines): | 
					
						
							|  |  |  |         return [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ManyIntersectingLinesPostZeta(ManyIntersectingLinesPreZeta): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |         "apply_zeta" : True, | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         # "anchor_density" : 5 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def get_modified_line_anims(self, lines): | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |         n_inserted_points = 30 | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         new_lines = lines.copy() | 
					
						
							|  |  |  |         new_lines.set_stroke(width = 5) | 
					
						
							|  |  |  |         def update_new_lines(lines_to_update): | 
					
						
							|  |  |  |             transformed = lines.copy() | 
					
						
							|  |  |  |             self.prepare_for_transformation(transformed) | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |             transformed.apply_complex_function(self.function) | 
					
						
							|  |  |  |             transformed.make_smooth() | 
					
						
							|  |  |  |             transformed.set_stroke(width = 5) | 
					
						
							|  |  |  |             for start, end in zip(lines_to_update, transformed): | 
					
						
							|  |  |  |                 if start.get_num_points() > 0: | 
					
						
							|  |  |  |                     start.points = np.array(end.points) | 
					
						
							|  |  |  |         return [UpdateFromFunc(new_lines, update_new_lines)] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ManyIntersectingLinesPreSSquared(ManyIntersectingLinesPreZeta): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |         "x_min" : -int(FRAME_X_RADIUS), | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         "apply_zeta" : False, | 
					
						
							|  |  |  |         "lines_center" : ORIGIN, | 
					
						
							|  |  |  |         "nudge_size" : 0.9, | 
					
						
							|  |  |  |         "function" : lambda z : z**2, | 
					
						
							|  |  |  |         "shift_directions" : [LEFT, RIGHT, UP, DOWN, DOWN+LEFT, UP+RIGHT], | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def establish_plane(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.plane.fade() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def apply_zeta_function(self, **kwargs): | 
					
						
							|  |  |  |         self.apply_complex_function(self.function, **kwargs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ManyIntersectingLinesPostSSquared(ManyIntersectingLinesPreSSquared): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "apply_zeta" : True, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def get_modified_line_anims(self, lines): | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |         n_inserted_points = 30 | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         new_lines = lines.copy() | 
					
						
							|  |  |  |         new_lines.set_stroke(width = 5) | 
					
						
							|  |  |  |         def update_new_lines(lines_to_update): | 
					
						
							|  |  |  |             transformed = lines.copy() | 
					
						
							|  |  |  |             self.prepare_for_transformation(transformed) | 
					
						
							|  |  |  |             transformed.apply_complex_function(self.function) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             transformed.make_smooth() | 
					
						
							|  |  |  |             transformed.set_stroke(width = 5) | 
					
						
							|  |  |  |             for start, end in zip(lines_to_update, transformed): | 
					
						
							|  |  |  |                 if start.get_num_points() > 0: | 
					
						
							|  |  |  |                     start.points = np.array(end.points) | 
					
						
							|  |  |  |         return [UpdateFromFunc(new_lines, update_new_lines)] | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | class ButWhatIsTheExensions(TeacherStudentsScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.student_says( | 
					
						
							|  |  |  |             """
 | 
					
						
							|  |  |  |             But what exactly \\emph{is} | 
					
						
							|  |  |  |             that continuation? | 
					
						
							|  |  |  |             """,
 | 
					
						
							|  |  |  |             target_mode = "sassy" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.change_student_modes("confused", "sassy", "confused") | 
					
						
							|  |  |  |         self.random_blink(2) | 
					
						
							|  |  |  |         self.teacher_says("""
 | 
					
						
							|  |  |  |             You're $\\$1{,}000{,}000$ richer | 
					
						
							|  |  |  |             if you can answer | 
					
						
							|  |  |  |             that fully | 
					
						
							| 
									
										
										
										
											2016-12-08 15:42:32 -08:00
										 |  |  |         """, target_mode = "shruggie")
 | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.change_student_modes(*["pondering"]*3) | 
					
						
							|  |  |  |         self.random_blink(3) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  | class MathematiciansLookingAtFunctionEquation(Scene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         equation = TexMobject( | 
					
						
							|  |  |  |             "\\zeta(s)", | 
					
						
							|  |  |  |             "= 2^s \\pi ^{s-1}", | 
					
						
							|  |  |  |             "\\sin\\left(\\frac{\\pi s}{2}\\right)", | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             "\\Gamma(1-s)", | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |             "\\zeta(1-s)", | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         equation.shift(UP) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         mathy = Mathematician().to_corner(DOWN+LEFT) | 
					
						
							|  |  |  |         mathys = VGroup(mathy) | 
					
						
							|  |  |  |         for x in range(2): | 
					
						
							|  |  |  |             mathys.add(Mathematician().next_to(mathys)) | 
					
						
							|  |  |  |         for mathy in mathys: | 
					
						
							|  |  |  |             mathy.change_mode("pondering") | 
					
						
							|  |  |  |             mathy.look_at(equation) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add(mathys) | 
					
						
							|  |  |  |         self.play(Write(VGroup(*equation[:-1]))) | 
					
						
							|  |  |  |         self.play(Transform( | 
					
						
							|  |  |  |             equation[0].copy(), | 
					
						
							|  |  |  |             equation[-1], | 
					
						
							|  |  |  |             path_arc = -np.pi/3, | 
					
						
							|  |  |  |             run_time = 2 | 
					
						
							|  |  |  |         )) | 
					
						
							|  |  |  |         for mathy in mathys: | 
					
						
							|  |  |  |             self.play(Blink(mathy)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | class DiscussZeros(ZetaTransformationScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.establish_plane() | 
					
						
							|  |  |  |         self.ask_about_zeros() | 
					
						
							|  |  |  |         self.show_trivial_zeros() | 
					
						
							|  |  |  |         self.show_critical_strip() | 
					
						
							|  |  |  |         self.transform_bit_of_critical_line() | 
					
						
							|  |  |  |         self.extend_transformed_critical_line() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def establish_plane(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.add_extra_plane_lines_for_zeta() | 
					
						
							|  |  |  |         self.add_reflected_plane() | 
					
						
							|  |  |  |         self.plane.fade() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def ask_about_zeros(self): | 
					
						
							|  |  |  |         dots = VGroup(*[ | 
					
						
							|  |  |  |             Dot( | 
					
						
							|  |  |  |                 (2+np.sin(12*alpha))*\ | 
					
						
							|  |  |  |                 rotate_vector(RIGHT, alpha+nudge) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             for alpha in np.arange(3*np.pi/20, 2*np.pi, 2*np.pi/5) | 
					
						
							|  |  |  |             for nudge in [random.random()*np.pi/6] | 
					
						
							|  |  |  |         ]) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         dots.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         q_marks = VGroup(*[ | 
					
						
							|  |  |  |             TexMobject("?").next_to(dot, UP) | 
					
						
							|  |  |  |             for dot in dots | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         arrows = VGroup(*[ | 
					
						
							|  |  |  |             Arrow(dot, ORIGIN, buff = 0.2, tip_length = 0.1) | 
					
						
							|  |  |  |             for dot in dots | 
					
						
							|  |  |  |         ]) | 
					
						
							|  |  |  |         question = TextMobject("Which numbers go to $0$?") | 
					
						
							|  |  |  |         question.add_background_rectangle() | 
					
						
							|  |  |  |         question.to_edge(UP) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for mob in dots, arrows, q_marks: | 
					
						
							|  |  |  |             self.play(ShowCreation(mob)) | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |         self.play(Write(question)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         dots.generate_target() | 
					
						
							|  |  |  |         for i, dot in enumerate(dots.target): | 
					
						
							|  |  |  |             dot.move_to(2*(i+1)*LEFT) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             FadeOut(arrows), | 
					
						
							|  |  |  |             FadeOut(q_marks), | 
					
						
							|  |  |  |             FadeOut(question), | 
					
						
							|  |  |  |             MoveToTarget(dots), | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.dots = dots | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def show_trivial_zeros(self): | 
					
						
							|  |  |  |         trivial_zero_words = TextMobject("``Trivial'' zeros") | 
					
						
							|  |  |  |         trivial_zero_words.next_to(ORIGIN, UP) | 
					
						
							|  |  |  |         trivial_zero_words.to_edge(LEFT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         randy = Randolph().flip() | 
					
						
							|  |  |  |         randy.to_corner(DOWN+RIGHT) | 
					
						
							|  |  |  |         bubble = randy.get_bubble() | 
					
						
							|  |  |  |         bubble.set_fill(BLACK, opacity = 0.8) | 
					
						
							|  |  |  |         bubble.write("$1^1 + 2^2 + 3^2 + \\cdots = 0$") | 
					
						
							|  |  |  |         bubble.resize_to_content() | 
					
						
							|  |  |  |         bubble.pin_to(randy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.plane.save_state() | 
					
						
							|  |  |  |         self.dots.save_state() | 
					
						
							|  |  |  |         for dot in self.dots.target: | 
					
						
							|  |  |  |             dot.move_to(ORIGIN) | 
					
						
							|  |  |  |         self.apply_zeta_function( | 
					
						
							|  |  |  |             added_anims = [MoveToTarget(self.dots, run_time = 3)], | 
					
						
							|  |  |  |             run_time = 3 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(3) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             self.plane.restore, | 
					
						
							|  |  |  |             self.plane.make_smooth, | 
					
						
							|  |  |  |             self.dots.restore, | 
					
						
							|  |  |  |             run_time = 2 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.remove(*self.get_mobjects_from_last_animation()) | 
					
						
							|  |  |  |         self.plane.restore() | 
					
						
							|  |  |  |         self.dots.restore() | 
					
						
							|  |  |  |         self.add(self.plane, self.dots) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.play(Write(trivial_zero_words)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.play(FadeIn(randy)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             randy.change_mode, "confused", | 
					
						
							|  |  |  |             ShowCreation(bubble), | 
					
						
							|  |  |  |             Write(bubble.content) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.play(Blink(randy)) | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         self.play(*list(map(FadeOut, [ | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             randy, bubble, bubble.content, trivial_zero_words | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         ]))) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def show_critical_strip(self): | 
					
						
							|  |  |  |         strip = Rectangle( | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             height = FRAME_HEIGHT, | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             width = 1 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         strip.next_to(ORIGIN, RIGHT, buff = 0) | 
					
						
							|  |  |  |         strip.set_stroke(width = 0) | 
					
						
							|  |  |  |         strip.set_fill(YELLOW, opacity = 0.3) | 
					
						
							|  |  |  |         name = TextMobject("Critical strip") | 
					
						
							|  |  |  |         name.add_background_rectangle() | 
					
						
							|  |  |  |         name.next_to(ORIGIN, LEFT) | 
					
						
							|  |  |  |         name.to_edge(UP) | 
					
						
							|  |  |  |         arrow = Arrow(name.get_bottom(), 0.5*RIGHT+UP) | 
					
						
							|  |  |  |         primes = TexMobject("2, 3, 5, 7, 11, 13, 17, \\dots") | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |         primes.to_corner(UP+RIGHT) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         # photo = Square() | 
					
						
							|  |  |  |         photo = ImageMobject("Riemann", invert = False) | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         photo.set_width(5) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         photo.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         new_dots = VGroup(*[ | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             Dot(0.5*RIGHT + y*UP) | 
					
						
							|  |  |  |             for y in np.linspace(-2.5, 3.2, 5) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         ]) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         new_dots.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         critical_line = Line( | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             0.5*RIGHT+FRAME_Y_RADIUS*DOWN, | 
					
						
							|  |  |  |             0.5*RIGHT+FRAME_Y_RADIUS*UP, | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             color = YELLOW | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.give_dots_wandering_anims() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(FadeIn(strip), *self.get_dot_wandering_anims()) | 
					
						
							|  |  |  |         self.play( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             Write(name, run_time = 1), | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             ShowCreation(arrow), | 
					
						
							|  |  |  |             *self.get_dot_wandering_anims() | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(*self.get_dot_wandering_anims()) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             FadeIn(primes), | 
					
						
							|  |  |  |             *self.get_dot_wandering_anims() | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         for x in range(7): | 
					
						
							|  |  |  |             self.play(*self.get_dot_wandering_anims()) | 
					
						
							|  |  |  |         self.play( | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             GrowFromCenter(photo), | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             FadeOut(name), | 
					
						
							|  |  |  |             FadeOut(arrow), | 
					
						
							|  |  |  |             *self.get_dot_wandering_anims() | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Transform(self.dots, new_dots)) | 
					
						
							|  |  |  |         self.play(ShowCreation(critical_line)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(3) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             photo.shift, 7*LEFT, | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |             *list(map(FadeOut, [ | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             primes, self.dots, strip | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |             ])) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         ) | 
					
						
							|  |  |  |         self.remove(photo) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.critical_line = critical_line | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def give_dots_wandering_anims(self): | 
					
						
							|  |  |  |         def func(t): | 
					
						
							|  |  |  |             result = (np.sin(6*2*np.pi*t) + 1)*RIGHT/2 | 
					
						
							|  |  |  |             result += 3*np.cos(2*2*np.pi*t)*UP | 
					
						
							|  |  |  |             return result | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.wandering_path = ParametricFunction(func) | 
					
						
							|  |  |  |         for i, dot in enumerate(self.dots): | 
					
						
							|  |  |  |             dot.target = dot.copy() | 
					
						
							|  |  |  |             q_mark = TexMobject("?") | 
					
						
							|  |  |  |             q_mark.next_to(dot.target, UP) | 
					
						
							|  |  |  |             dot.target.add(q_mark) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             dot.target.move_to(self.wandering_path.point_from_proportion( | 
					
						
							|  |  |  |                 (float(2+2*i)/(4*len(list(self.dots))))%1 | 
					
						
							|  |  |  |             )) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.dot_anim_count = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_dot_wandering_anims(self): | 
					
						
							|  |  |  |         self.dot_anim_count += 1 | 
					
						
							|  |  |  |         if self.dot_anim_count == 1: | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |             return list(map(MoveToTarget, self.dots)) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         denom = 4*(len(list(self.dots))) | 
					
						
							|  |  |  |         def get_rate_func(index): | 
					
						
							|  |  |  |             return lambda t : (float(self.dot_anim_count + 2*index + t)/denom)%1 | 
					
						
							|  |  |  |         return [ | 
					
						
							|  |  |  |             MoveAlongPath( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |                 dot, self.wandering_path, | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |                 rate_func = get_rate_func(i) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             for i, dot in enumerate(self.dots) | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def transform_bit_of_critical_line(self): | 
					
						
							|  |  |  |         self.play( | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             self.plane.scale, 0.8, | 
					
						
							|  |  |  |             self.critical_line.scale, 0.8, | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             rate_func = there_and_back, | 
					
						
							|  |  |  |             run_time = 2 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             self.plane.set_stroke, GREY, 1, | 
					
						
							|  |  |  |             Animation(self.critical_line) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.plane.add(self.critical_line) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.apply_zeta_function() | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait(2) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             self.plane.fade, | 
					
						
							|  |  |  |             Animation(self.critical_line) | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def extend_transformed_critical_line(self): | 
					
						
							|  |  |  |         def func(t): | 
					
						
							|  |  |  |             z = zeta(complex(0.5, t)) | 
					
						
							|  |  |  |             return z.real*RIGHT + z.imag*UP | 
					
						
							|  |  |  |         full_line = VGroup(*[ | 
					
						
							|  |  |  |             ParametricFunction(func, t_min = t0, t_max = t0+1) | 
					
						
							|  |  |  |             for t0 in range(100) | 
					
						
							|  |  |  |         ]) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:59:39 -07:00
										 |  |  |         full_line.set_color_by_gradient( | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  |             YELLOW, BLUE, GREEN, RED, YELLOW, BLUE, GREEN, RED, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(ShowCreation(full_line, run_time = 20, rate_func = None)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  | class AskAboutRelationToPrimes(TeacherStudentsScene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.student_says("""
 | 
					
						
							|  |  |  |             Whoa!  Where the heck | 
					
						
							|  |  |  |             do primes come in here? | 
					
						
							|  |  |  |         """, target_mode = "confused")
 | 
					
						
							|  |  |  |         self.random_blink(3) | 
					
						
							|  |  |  |         self.teacher_says("""
 | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             Perhaps in a | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |             different video. | 
					
						
							|  |  |  |         """, target_mode = "hesitant")
 | 
					
						
							|  |  |  |         self.random_blink(3) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  | class HighlightCriticalLineAgain(DiscussZeros): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.establish_plane() | 
					
						
							|  |  |  |         title = TexMobject("\\zeta(", "s", ") = 0") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         title.set_color_by_tex("s", YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         title.add_background_rectangle() | 
					
						
							|  |  |  |         title.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         self.add(title) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         strip = Rectangle( | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             height = FRAME_HEIGHT, | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             width = 1 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         strip.next_to(ORIGIN, RIGHT, buff = 0) | 
					
						
							|  |  |  |         strip.set_stroke(width = 0) | 
					
						
							|  |  |  |         strip.set_fill(YELLOW, opacity = 0.3) | 
					
						
							|  |  |  |         line = Line( | 
					
						
							| 
									
										
										
										
											2018-03-30 11:25:37 -07:00
										 |  |  |             0.5*RIGHT+FRAME_Y_RADIUS*UP, | 
					
						
							|  |  |  |             0.5*RIGHT+FRAME_Y_RADIUS*DOWN, | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             color = YELLOW | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         randy = Randolph().to_corner(DOWN+LEFT) | 
					
						
							|  |  |  |         million = TexMobject("\\$1{,}000{,}000") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         million.set_color(GREEN_B) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         million.next_to(ORIGIN, UP+LEFT) | 
					
						
							|  |  |  |         million.shift(2*LEFT) | 
					
						
							|  |  |  |         arrow1 = Arrow(million.get_right(), line.get_top()) | 
					
						
							|  |  |  |         arrow2 = Arrow(million.get_right(), line.get_bottom()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add(randy, strip) | 
					
						
							|  |  |  |         self.play(Write(million)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             randy.change_mode, "pondering", | 
					
						
							|  |  |  |             randy.look_at, line.get_top(), | 
					
						
							|  |  |  |             ShowCreation(arrow1), | 
					
						
							|  |  |  |             run_time = 3 | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             randy.look_at, line.get_bottom(), | 
					
						
							|  |  |  |             ShowCreation(line), | 
					
						
							|  |  |  |             Transform(arrow1, arrow2) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(FadeOut(arrow1)) | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play(randy.look_at, line.get_center()) | 
					
						
							|  |  |  |         self.play(randy.change_mode, "confused") | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play(randy.change_mode, "pondering") | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class DiscussSumOfNaturals(Scene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         title = TexMobject( | 
					
						
							|  |  |  |             "\\zeta(s) = \\sum_{n=1}^\\infty \\frac{1}{n^s}" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         VGroup(title[2], title[-1]).set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         title.to_corner(UP+LEFT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         neg_twelfth, eq, zeta_neg_1, sum_naturals = equation = TexMobject( | 
					
						
							|  |  |  |             "-\\frac{1}{12}", | 
					
						
							|  |  |  |             "=", | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             "\\zeta(-1)", | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             "= 1 + 2 + 3 + 4 + \\cdots" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         neg_twelfth.set_color(GREEN_B) | 
					
						
							|  |  |  |         VGroup(*zeta_neg_1[2:4]).set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         q_mark = TexMobject("?").next_to(sum_naturals[0], UP) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         q_mark.set_color(RED) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         randy = Randolph() | 
					
						
							|  |  |  |         randy.to_corner(DOWN+LEFT) | 
					
						
							|  |  |  |         analytic_continuation = TextMobject("Analytic continuation") | 
					
						
							|  |  |  |         analytic_continuation.next_to(title, RIGHT, 3*LARGE_BUFF) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         sum_to_zeta = Arrow(title.get_corner(DOWN+RIGHT), zeta_neg_1) | 
					
						
							|  |  |  |         sum_to_ac = Arrow(title.get_right(), analytic_continuation) | 
					
						
							|  |  |  |         ac_to_zeta = Arrow(analytic_continuation.get_bottom(), zeta_neg_1.get_top()) | 
					
						
							|  |  |  |         cross = TexMobject("\\times") | 
					
						
							|  |  |  |         cross.scale(2) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         cross.set_color(RED) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         cross.rotate(np.pi/6) | 
					
						
							|  |  |  |         cross.move_to(sum_to_zeta.get_center()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         brace = Brace(VGroup(zeta_neg_1, sum_naturals)) | 
					
						
							|  |  |  |         words = TextMobject( | 
					
						
							|  |  |  |             "If not equal, at least connected", | 
					
						
							|  |  |  |             "\\\\(see links in description)" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         words.next_to(brace, DOWN) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add(neg_twelfth, eq, zeta_neg_1, randy, title) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             Write(sum_naturals), | 
					
						
							|  |  |  |             Write(q_mark), | 
					
						
							|  |  |  |             randy.change_mode, "confused" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play(randy.change_mode, "angry") | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(sum_to_zeta), | 
					
						
							|  |  |  |             Write(cross) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             Transform(sum_to_zeta, sum_to_ac), | 
					
						
							|  |  |  |             FadeOut(cross), | 
					
						
							|  |  |  |             Write(analytic_continuation), | 
					
						
							|  |  |  |             randy.change_mode, "pondering", | 
					
						
							|  |  |  |             randy.look_at, analytic_continuation, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(ShowCreation(ac_to_zeta)) | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             GrowFromCenter(brace), | 
					
						
							|  |  |  |             Write(words[0]), | 
					
						
							|  |  |  |             randy.look_at, words[0], | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play(FadeIn(words[1])) | 
					
						
							|  |  |  |         self.play(Blink(randy)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class InventingMathPreview(Scene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         rect = Rectangle(height = 9, width = 16) | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         rect.set_height(4) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         title = TextMobject("What does it feel like to invent math?") | 
					
						
							|  |  |  |         title.next_to(rect, UP) | 
					
						
							|  |  |  |         sum_tex = TexMobject("1+2+4+8+\\cdots = -1") | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         sum_tex.set_width(rect.get_width()-1) | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             ShowCreation(rect), | 
					
						
							|  |  |  |             Write(title) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Write(sum_tex)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-06 22:25:26 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  | class FinalAnimationTease(Scene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         morty = Mortimer().shift(2*(DOWN+RIGHT)) | 
					
						
							| 
									
										
										
										
											2017-02-14 14:00:50 -08:00
										 |  |  |         bubble = morty.get_bubble(SpeechBubble) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         bubble.write("""
 | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             Want to know what | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             $\\zeta'(s)$ looks like? | 
					
						
							|  |  |  |         """)
 | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.add(morty) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             morty.change_mode, "hooray", | 
					
						
							|  |  |  |             morty.look_at, bubble.content, | 
					
						
							|  |  |  |             ShowCreation(bubble), | 
					
						
							|  |  |  |             Write(bubble.content) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(morty)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  | class PatreonThanks(Scene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "specific_patrons" : [ | 
					
						
							|  |  |  |             "CrypticSwarm", | 
					
						
							|  |  |  |             "Ali Yahya", | 
					
						
							|  |  |  |             "Damion Kistler", | 
					
						
							|  |  |  |             "Juan Batiz-Benet", | 
					
						
							|  |  |  |             "Yu Jun", | 
					
						
							|  |  |  |             "Othman Alikhan", | 
					
						
							|  |  |  |             "Markus Persson", | 
					
						
							|  |  |  |             "Joseph John Cox", | 
					
						
							|  |  |  |             "Luc Ritchie", | 
					
						
							|  |  |  |             "Shimin Kuang", | 
					
						
							|  |  |  |             "Einar Johansen", | 
					
						
							|  |  |  |             "Rish Kundalia", | 
					
						
							|  |  |  |             "Achille Brighton", | 
					
						
							|  |  |  |             "Kirk Werklund", | 
					
						
							|  |  |  |             "Ripta Pasay", | 
					
						
							|  |  |  |             "Felipe Diniz", | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         morty = Mortimer() | 
					
						
							|  |  |  |         morty.next_to(ORIGIN, DOWN) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         n_patrons = len(self.specific_patrons) | 
					
						
							|  |  |  |         special_thanks = TextMobject("Special thanks to:") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         special_thanks.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         special_thanks.shift(3*UP) | 
					
						
							|  |  |  |         patreon_logo = ImageMobject("patreon", invert = False) | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         patreon_logo.set_height(1.5) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         patreon_logo.next_to(special_thanks, DOWN) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         left_patrons = VGroup(*list(map(TextMobject, | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             self.specific_patrons[:n_patrons/2] | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         ))) | 
					
						
							|  |  |  |         right_patrons = VGroup(*list(map(TextMobject, | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             self.specific_patrons[n_patrons/2:] | 
					
						
							| 
									
										
										
										
											2018-08-09 17:56:05 -07:00
										 |  |  |         ))) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         for patrons, vect in (left_patrons, LEFT), (right_patrons, RIGHT): | 
					
						
							|  |  |  |             patrons.arrange_submobjects(DOWN, aligned_edge = LEFT) | 
					
						
							|  |  |  |             patrons.next_to(special_thanks, DOWN) | 
					
						
							|  |  |  |             patrons.to_edge(vect, buff = LARGE_BUFF) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add(patreon_logo) | 
					
						
							|  |  |  |         self.play(morty.change_mode, "gracious") | 
					
						
							|  |  |  |         self.play(Write(special_thanks, run_time = 1)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             Write(left_patrons), | 
					
						
							|  |  |  |             morty.look_at, left_patrons | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play( | 
					
						
							|  |  |  |             Write(right_patrons), | 
					
						
							|  |  |  |             morty.look_at, right_patrons | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(Blink(morty)) | 
					
						
							|  |  |  |         for patrons in left_patrons, right_patrons: | 
					
						
							|  |  |  |             for index in 0, -1: | 
					
						
							|  |  |  |                 self.play(morty.look_at, patrons[index]) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |                 self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class CreditTwo(Scene): | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         morty = Mortimer() | 
					
						
							|  |  |  |         morty.next_to(ORIGIN, DOWN) | 
					
						
							|  |  |  |         morty.to_edge(RIGHT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         brother = PiCreature(color = GOLD_E) | 
					
						
							|  |  |  |         brother.next_to(morty, LEFT) | 
					
						
							|  |  |  |         brother.look_at(morty.eyes) | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         headphones = Headphones(height = 1) | 
					
						
							|  |  |  |         headphones.move_to(morty.eyes, aligned_edge = DOWN) | 
					
						
							|  |  |  |         headphones.shift(0.1*DOWN) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         url = TextMobject("www.audible.com/3blue1brown") | 
					
						
							|  |  |  |         url.to_corner(UP+RIGHT, buff = LARGE_BUFF) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.add(morty) | 
					
						
							|  |  |  |         self.play(Blink(morty)) | 
					
						
							|  |  |  |         self.play( | 
					
						
							| 
									
										
										
										
											2017-10-05 21:03:30 -05:00
										 |  |  |             FadeIn(headphones), | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |             Write(url), | 
					
						
							|  |  |  |             Animation(morty) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(morty.change_mode, "happy") | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |         for x in range(4): | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |             self.play(Blink(morty)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             FadeIn(brother), | 
					
						
							|  |  |  |             morty.look_at, brother.eyes | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.play(brother.change_mode, "surprised") | 
					
						
							|  |  |  |         self.play(Blink(brother)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.play( | 
					
						
							|  |  |  |             morty.look, LEFT, | 
					
						
							|  |  |  |             brother.change_mode, "happy", | 
					
						
							|  |  |  |             brother.look, LEFT | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |         for x in range(10): | 
					
						
							|  |  |  |             self.play(Blink(morty)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |             self.play(Blink(brother)) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class FinalAnimation(ZetaTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "min_added_anchors" : 100, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.add_extra_plane_lines_for_zeta() | 
					
						
							|  |  |  |         self.add_reflected_plane() | 
					
						
							|  |  |  |         title = TexMobject("s", "\\to \\frac{d\\zeta}{ds}(", "s", ")") | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         title.set_color_by_tex("s", YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         title.add_background_rectangle() | 
					
						
							|  |  |  |         title.scale(1.5) | 
					
						
							|  |  |  |         title.to_corner(UP+LEFT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.play(Write(title)) | 
					
						
							|  |  |  |         self.add_foreground_mobjects(title) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-07 18:37:56 -08:00
										 |  |  |         self.apply_complex_function(d_zeta, run_time = 8) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |         self.wait() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  | class Thumbnail(ZetaTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "anchor_density" : 35 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.add_extra_plane_lines_for_zeta() | 
					
						
							|  |  |  |         self.add_reflected_plane() | 
					
						
							|  |  |  |         self.apply_zeta_function() | 
					
						
							|  |  |  |         self.plane.set_stroke(width = 4) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 07:10:38 -08:00
										 |  |  |         div_sum = TexMobject("-\\frac{1}{12} = ", "1+2+3+4+\\cdots") | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         div_sum.set_width(FRAME_WIDTH-1) | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |         div_sum.to_edge(DOWN) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         div_sum.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |         for mob in div_sum.submobjects: | 
					
						
							|  |  |  |             mob.add_to_back(BackgroundRectangle(mob)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         zeta = TexMobject("\\zeta(s)") | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         zeta.set_height(FRAME_Y_RADIUS-1) | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |         zeta.to_corner(UP+LEFT) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         million = TexMobject("\\$1{,}000{,}000") | 
					
						
							| 
									
										
										
										
											2018-08-08 10:30:52 -07:00
										 |  |  |         million.set_width(FRAME_X_RADIUS+1) | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |         million.to_edge(UP+RIGHT) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         million.set_color(GREEN_B) | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |         million.add_background_rectangle() | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-08 21:18:44 -08:00
										 |  |  |         self.add(div_sum, million, zeta) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 07:10:38 -08:00
										 |  |  | class ZetaPartialSums(ZetaTransformationScene): | 
					
						
							|  |  |  |     CONFIG = { | 
					
						
							|  |  |  |         "anchor_density" : 35, | 
					
						
							|  |  |  |         "num_partial_sums" : 12, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     def construct(self): | 
					
						
							|  |  |  |         self.add_transformable_plane() | 
					
						
							|  |  |  |         self.add_extra_plane_lines_for_zeta() | 
					
						
							|  |  |  |         self.prepare_for_transformation(self.plane) | 
					
						
							| 
									
										
										
										
											2016-12-01 15:15:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-26 07:10:38 -08:00
										 |  |  |         N_list = [2**k for k in range(self.num_partial_sums)] | 
					
						
							|  |  |  |         sigma = TexMobject( | 
					
						
							|  |  |  |             "\\sum_{n = 1}^N \\frac{1}{n^s}" | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         sigmas = [] | 
					
						
							|  |  |  |         for N in N_list + ["\\infty"]: | 
					
						
							|  |  |  |             tex = TexMobject(str(N)) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |             tex.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-26 07:10:38 -08:00
										 |  |  |             new_sigma = sigma.copy() | 
					
						
							|  |  |  |             top = new_sigma[0] | 
					
						
							|  |  |  |             tex.move_to(top, DOWN) | 
					
						
							|  |  |  |             new_sigma.remove(top) | 
					
						
							|  |  |  |             new_sigma.add(tex) | 
					
						
							|  |  |  |             new_sigma.to_corner(UP+LEFT) | 
					
						
							|  |  |  |             sigmas.append(new_sigma) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def get_partial_sum_func(n_terms): | 
					
						
							|  |  |  |             return lambda s : sum([1./(n**s) for n in range(1, n_terms+1)]) | 
					
						
							|  |  |  |         interim_planes = [ | 
					
						
							|  |  |  |             self.plane.copy().apply_complex_function( | 
					
						
							|  |  |  |                 get_partial_sum_func(N) | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |             for N in N_list | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |         interim_planes.append(self.plane.copy().apply_complex_function(zeta)) | 
					
						
							|  |  |  |         symbol = VGroup(TexMobject("s")) | 
					
						
							|  |  |  |         symbol.scale(2) | 
					
						
							| 
									
										
										
										
											2018-03-30 11:51:31 -07:00
										 |  |  |         symbol.set_color(YELLOW) | 
					
						
							| 
									
										
										
										
											2016-12-26 07:10:38 -08:00
										 |  |  |         symbol.to_corner(UP+LEFT) | 
					
						
							|  |  |  |         for plane, sigma in zip(interim_planes, sigmas): | 
					
						
							|  |  |  |             self.play( | 
					
						
							|  |  |  |                 Transform(self.plane, plane), | 
					
						
							|  |  |  |                 Transform(symbol, sigma) | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2018-01-15 19:15:05 -08:00
										 |  |  |             self.wait() |