2017-02-16 13:03:26 -08:00
|
|
|
from vectorized_mobject import VMobject, VGroup
|
2016-04-20 19:24:54 -07:00
|
|
|
from svg_mobject import SVGMobject, VMobjectFromSVGPathstring
|
2016-07-26 12:32:51 -07:00
|
|
|
from topics.geometry import BackgroundRectangle
|
2015-10-28 17:18:50 -07:00
|
|
|
from helpers import *
|
2016-07-21 15:16:49 -07:00
|
|
|
import collections
|
2015-10-28 17:18:50 -07:00
|
|
|
|
2016-08-02 15:50:32 -07:00
|
|
|
TEX_MOB_SCALE_FACTOR = 0.05
|
|
|
|
TEXT_MOB_SCALE_FACTOR = 0.05
|
2016-04-20 19:24:54 -07:00
|
|
|
|
|
|
|
|
|
|
|
class TexSymbol(VMobjectFromSVGPathstring):
|
2016-07-19 11:08:31 -07:00
|
|
|
def pointwise_become_partial(self, mobject, a, b):
|
2016-04-20 19:24:54 -07:00
|
|
|
#TODO, this assumes a = 0
|
|
|
|
if b < 0.5:
|
|
|
|
b = 2*b
|
|
|
|
width = 1
|
|
|
|
opacity = 0
|
|
|
|
else:
|
|
|
|
width = 2 - 2*b
|
|
|
|
opacity = 2*b - 1
|
|
|
|
b = 1
|
2016-07-19 11:08:31 -07:00
|
|
|
VMobjectFromSVGPathstring.pointwise_become_partial(
|
2016-04-20 19:24:54 -07:00
|
|
|
self, mobject, 0, b
|
|
|
|
)
|
|
|
|
self.set_stroke(width = width)
|
|
|
|
self.set_fill(opacity = opacity)
|
|
|
|
|
2016-04-17 19:29:27 -07:00
|
|
|
|
2016-04-17 00:31:38 -07:00
|
|
|
class TexMobject(SVGMobject):
|
2016-02-27 16:32:53 -08:00
|
|
|
CONFIG = {
|
2015-10-28 17:18:50 -07:00
|
|
|
"template_tex_file" : TEMPLATE_TEX_FILE,
|
2016-04-17 12:59:53 -07:00
|
|
|
"stroke_width" : 0,
|
2016-04-17 19:29:27 -07:00
|
|
|
"fill_opacity" : 1.0,
|
|
|
|
"fill_color" : WHITE,
|
2015-12-13 15:41:45 -08:00
|
|
|
"should_center" : True,
|
2016-08-22 21:22:21 -07:00
|
|
|
"arg_separator" : " ",
|
2016-07-21 15:16:49 -07:00
|
|
|
"enforce_new_line_structure" : False,
|
2016-08-02 15:50:32 -07:00
|
|
|
"initial_scale_factor" : TEX_MOB_SCALE_FACTOR,
|
2016-07-12 15:16:20 -07:00
|
|
|
"organize_left_to_right" : False,
|
2016-04-23 23:36:05 -07:00
|
|
|
"propogate_style_to_family" : True,
|
2016-09-10 17:35:15 -07:00
|
|
|
"alignment" : "",
|
2015-10-28 17:18:50 -07:00
|
|
|
}
|
2016-08-09 14:07:23 -07:00
|
|
|
def __init__(self, *args, **kwargs):
|
2016-04-17 00:31:38 -07:00
|
|
|
digest_config(self, kwargs, locals())
|
2016-08-09 14:07:23 -07:00
|
|
|
##TODO, Eventually remove this
|
|
|
|
if len(args) == 1 and isinstance(args[0], list):
|
2016-08-10 10:26:07 -07:00
|
|
|
self.args = args[0]
|
2016-08-09 14:07:23 -07:00
|
|
|
##
|
2016-08-10 10:26:07 -07:00
|
|
|
assert(all([isinstance(a, str) for a in self.args]))
|
2016-11-07 11:05:41 -08:00
|
|
|
self.tex_string = self.get_modified_expression()
|
|
|
|
file_name = tex_to_svg_file(
|
|
|
|
self.tex_string,
|
|
|
|
self.template_tex_file
|
|
|
|
)
|
|
|
|
SVGMobject.__init__(self, file_name = file_name, **kwargs)
|
2016-07-12 10:34:35 -07:00
|
|
|
if self.organize_left_to_right:
|
|
|
|
self.organize_submobjects_left_to_right()
|
2016-04-17 12:59:53 -07:00
|
|
|
|
2016-07-21 15:16:49 -07:00
|
|
|
|
2016-04-20 19:24:54 -07:00
|
|
|
def path_string_to_mobject(self, path_string):
|
|
|
|
#Overwrite superclass default to use
|
|
|
|
#specialized path_string mobject
|
|
|
|
return TexSymbol(path_string)
|
|
|
|
|
|
|
|
|
2016-07-21 11:00:45 -07:00
|
|
|
def generate_points(self):
|
|
|
|
SVGMobject.generate_points(self)
|
2016-08-09 14:07:23 -07:00
|
|
|
if len(self.args) > 1:
|
|
|
|
self.handle_multiple_args()
|
2016-07-21 11:00:45 -07:00
|
|
|
|
2016-07-21 15:16:49 -07:00
|
|
|
def get_modified_expression(self):
|
2016-08-22 21:22:21 -07:00
|
|
|
result = self.arg_separator.join(self.args)
|
2016-07-21 15:16:49 -07:00
|
|
|
if self.enforce_new_line_structure:
|
|
|
|
result = result.replace("\n", " \\\\ \n ")
|
2016-09-10 17:35:15 -07:00
|
|
|
result = " ".join([self.alignment, result])
|
|
|
|
result = result.strip()
|
2017-02-16 13:03:26 -08:00
|
|
|
result = self.remove_stray_braces(result)
|
2016-07-21 15:16:49 -07:00
|
|
|
return result
|
2016-07-21 11:00:45 -07:00
|
|
|
|
2017-02-16 13:03:26 -08:00
|
|
|
def remove_stray_braces(self, tex):
|
|
|
|
"""
|
|
|
|
Makes TexMobject resiliant to unmatched { at start
|
|
|
|
"""
|
|
|
|
num_lefts, num_rights = [
|
|
|
|
tex.count(char)
|
|
|
|
for char in "{}"
|
|
|
|
]
|
2017-02-16 20:21:35 -08:00
|
|
|
if num_rights > num_lefts:
|
|
|
|
backwards = tex[::-1].replace("}", "", num_rights-num_lefts)
|
|
|
|
tex = backwards[::-1]
|
|
|
|
elif num_lefts > num_rights:
|
|
|
|
tex = tex.replace("{", "", num_lefts-num_rights)
|
2017-02-16 13:03:26 -08:00
|
|
|
return tex
|
|
|
|
|
|
|
|
|
2016-08-23 13:38:33 -07:00
|
|
|
def get_tex_string(self):
|
|
|
|
return self.tex_string
|
|
|
|
|
2016-08-09 14:07:23 -07:00
|
|
|
def handle_multiple_args(self):
|
2016-07-21 11:00:45 -07:00
|
|
|
new_submobjects = []
|
|
|
|
curr_index = 0
|
2016-08-18 12:54:04 -07:00
|
|
|
self.expression_parts = list(self.args)
|
2016-08-09 14:07:23 -07:00
|
|
|
for expr in self.args:
|
2017-03-28 11:12:02 -07:00
|
|
|
sub_tex_mob = TexMobject(expr, **self.CONFIG)
|
|
|
|
new_index = curr_index + len(sub_tex_mob.submobjects)
|
|
|
|
sub_tex_mob.submobjects = self.submobjects[curr_index:new_index]
|
|
|
|
new_submobjects.append(sub_tex_mob)
|
2016-07-21 11:00:45 -07:00
|
|
|
curr_index = new_index
|
|
|
|
self.submobjects = new_submobjects
|
2016-04-17 12:59:53 -07:00
|
|
|
return self
|
|
|
|
|
2017-03-22 15:06:17 -07:00
|
|
|
def get_parts_by_tex(self, tex, substring = True):
|
2017-03-10 16:55:23 -08:00
|
|
|
def test(tex1, tex2):
|
|
|
|
return tex1 == tex2 or (substring and tex1 in tex2)
|
2017-03-22 15:06:17 -07:00
|
|
|
|
2017-03-28 11:12:02 -07:00
|
|
|
tex_submobjects = filter(
|
|
|
|
lambda m : isinstance(m, TexMobject),
|
|
|
|
self.submobject_family()
|
|
|
|
)
|
2017-03-25 12:18:33 -07:00
|
|
|
if hasattr(self, "expression_parts"):
|
2017-03-28 11:12:02 -07:00
|
|
|
tex_submobjects.remove(self)
|
|
|
|
return filter(
|
|
|
|
lambda m : test(tex, m.get_tex_string()),
|
|
|
|
tex_submobjects
|
|
|
|
)
|
2017-03-25 12:18:33 -07:00
|
|
|
|
2017-03-22 15:06:17 -07:00
|
|
|
def get_part_by_tex(self, tex, **kwargs):
|
|
|
|
all_parts = self.get_parts_by_tex(tex, **kwargs)
|
|
|
|
return all_parts[0] if all_parts else None
|
2017-03-10 16:55:23 -08:00
|
|
|
|
|
|
|
def highlight_by_tex(self, tex, color, **kwargs):
|
|
|
|
parts_to_color = self.get_parts_by_tex(tex, **kwargs)
|
|
|
|
for part in parts_to_color:
|
|
|
|
part.highlight(color)
|
2016-08-18 12:54:04 -07:00
|
|
|
return self
|
|
|
|
|
2016-07-12 10:34:35 -07:00
|
|
|
def organize_submobjects_left_to_right(self):
|
2016-04-17 19:29:27 -07:00
|
|
|
self.submobjects.sort(
|
|
|
|
lambda m1, m2 : int((m1.get_left()-m2.get_left())[0])
|
|
|
|
)
|
|
|
|
|
2016-07-22 11:22:31 -07:00
|
|
|
def add_background_rectangle(self, color = BLACK, opacity = 0.75):
|
2016-08-28 18:26:08 -07:00
|
|
|
self.background_rectangle = BackgroundRectangle(
|
2016-08-03 16:51:16 -07:00
|
|
|
self, color = color,
|
|
|
|
fill_opacity = opacity
|
|
|
|
)
|
2016-07-25 16:04:54 -07:00
|
|
|
letters = VMobject(*self.submobjects)
|
2016-08-28 18:26:08 -07:00
|
|
|
self.submobjects = [self.background_rectangle, letters]
|
2016-07-22 11:22:31 -07:00
|
|
|
return self
|
2016-04-20 19:24:54 -07:00
|
|
|
|
2015-10-28 17:18:50 -07:00
|
|
|
class TextMobject(TexMobject):
|
2016-02-27 16:32:53 -08:00
|
|
|
CONFIG = {
|
2015-10-28 17:18:50 -07:00
|
|
|
"template_tex_file" : TEMPLATE_TEXT_FILE,
|
2016-08-02 15:50:32 -07:00
|
|
|
"initial_scale_factor" : TEXT_MOB_SCALE_FACTOR,
|
2016-07-21 15:16:49 -07:00
|
|
|
"enforce_new_line_structure" : True,
|
2016-09-10 17:35:15 -07:00
|
|
|
"alignment" : "\\centering",
|
2015-10-28 17:18:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-12-16 19:49:50 -08:00
|
|
|
class Brace(TexMobject):
|
2016-02-27 16:32:53 -08:00
|
|
|
CONFIG = {
|
2015-10-28 17:18:50 -07:00
|
|
|
"buff" : 0.2,
|
2017-02-16 13:03:26 -08:00
|
|
|
"width_multiplier" : 2,
|
|
|
|
"max_num_quads" : 15,
|
|
|
|
"min_num_quads" : 0,
|
2015-10-28 17:18:50 -07:00
|
|
|
}
|
2015-12-16 19:49:50 -08:00
|
|
|
def __init__(self, mobject, direction = DOWN, **kwargs):
|
2016-08-03 16:51:16 -07:00
|
|
|
digest_config(self, kwargs, locals())
|
2015-12-16 19:49:50 -08:00
|
|
|
angle = -np.arctan2(*direction[:2]) + np.pi
|
|
|
|
mobject.rotate(-angle)
|
|
|
|
left = mobject.get_corner(DOWN+LEFT)
|
|
|
|
right = mobject.get_corner(DOWN+RIGHT)
|
2017-02-06 21:43:46 -08:00
|
|
|
target_width = right[0]-left[0]
|
|
|
|
|
|
|
|
## Adding int(target_width) qquads gives approximately the right width
|
2017-02-16 13:03:26 -08:00
|
|
|
num_quads = np.clip(
|
|
|
|
int(self.width_multiplier*target_width),
|
|
|
|
self.min_num_quads, self.max_num_quads
|
|
|
|
)
|
|
|
|
tex_string = "\\underbrace{%s}"%(num_quads*"\\qquad")
|
2017-02-06 21:43:46 -08:00
|
|
|
TexMobject.__init__(self, tex_string, **kwargs)
|
2017-03-10 10:11:32 -08:00
|
|
|
self.tip_point_index = np.argmin(self.get_all_points()[:,1])
|
2017-02-06 21:43:46 -08:00
|
|
|
self.stretch_to_fit_width(target_width)
|
2016-04-23 23:36:05 -07:00
|
|
|
self.shift(left - self.get_corner(UP+LEFT) + self.buff*DOWN)
|
2015-12-16 19:49:50 -08:00
|
|
|
for mob in mobject, self:
|
|
|
|
mob.rotate(angle)
|
2016-07-19 11:08:31 -07:00
|
|
|
|
2017-03-10 13:54:52 -08:00
|
|
|
def put_at_tip(self, mob, use_next_to = True, **kwargs):
|
|
|
|
if use_next_to:
|
|
|
|
mob.next_to(
|
|
|
|
self.get_tip(),
|
|
|
|
np.round(self.get_direction()),
|
|
|
|
**kwargs
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
mob.move_to(self.get_tip())
|
|
|
|
buff = kwargs.get("buff", DEFAULT_MOBJECT_TO_MOBJECT_BUFFER)
|
|
|
|
shift_distance = mob.get_width()/2.0+buff
|
|
|
|
mob.shift(self.get_direction()*shift_distance)
|
2016-08-03 16:51:16 -07:00
|
|
|
return self
|
|
|
|
|
2016-08-10 10:26:07 -07:00
|
|
|
def get_text(self, *text, **kwargs):
|
|
|
|
text_mob = TextMobject(*text)
|
2016-08-03 16:51:16 -07:00
|
|
|
self.put_at_tip(text_mob, **kwargs)
|
|
|
|
return text_mob
|
|
|
|
|
2017-03-09 15:50:40 -08:00
|
|
|
def get_tip(self):
|
|
|
|
# Very specific to the LaTeX representation
|
|
|
|
# of a brace, but it's the only way I can think
|
|
|
|
# of to get the tip regardless of orientation.
|
2017-03-10 10:11:32 -08:00
|
|
|
# return self.submobjects[2].get_anchors()[7]
|
|
|
|
return self.get_all_points()[self.tip_point_index]
|
2017-03-09 15:50:40 -08:00
|
|
|
|
|
|
|
def get_direction(self):
|
|
|
|
vect = self.get_tip() - self.get_center()
|
|
|
|
return vect/np.linalg.norm(vect)
|
|
|
|
|
|
|
|
|
2016-08-03 16:51:16 -07:00
|
|
|
|
2015-10-28 17:18:50 -07:00
|
|
|
|
2016-05-03 23:14:40 -07:00
|
|
|
def tex_hash(expression, template_tex_file):
|
|
|
|
return str(hash(expression + template_tex_file))
|
2015-10-28 17:18:50 -07:00
|
|
|
|
2016-04-17 12:59:53 -07:00
|
|
|
def tex_to_svg_file(expression, template_tex_file):
|
2016-05-03 23:14:40 -07:00
|
|
|
image_dir = os.path.join(
|
|
|
|
TEX_IMAGE_DIR,
|
|
|
|
tex_hash(expression, template_tex_file)
|
|
|
|
)
|
2015-10-28 17:18:50 -07:00
|
|
|
if os.path.exists(image_dir):
|
|
|
|
return get_sorted_image_list(image_dir)
|
2016-04-17 00:31:38 -07:00
|
|
|
tex_file = generate_tex_file(expression, template_tex_file)
|
2016-04-17 12:59:53 -07:00
|
|
|
dvi_file = tex_to_dvi(tex_file)
|
|
|
|
return dvi_to_svg(dvi_file)
|
2015-10-28 17:18:50 -07:00
|
|
|
|
|
|
|
|
2016-04-17 00:31:38 -07:00
|
|
|
def generate_tex_file(expression, template_tex_file):
|
2016-05-03 23:14:40 -07:00
|
|
|
result = os.path.join(
|
|
|
|
TEX_DIR,
|
|
|
|
tex_hash(expression, template_tex_file)
|
2016-08-18 12:54:04 -07:00
|
|
|
) + ".tex"
|
2015-10-28 17:18:50 -07:00
|
|
|
if not os.path.exists(result):
|
2016-04-17 00:31:38 -07:00
|
|
|
print "Writing \"%s\" to %s"%(
|
|
|
|
"".join(expression), result
|
2015-10-28 17:18:50 -07:00
|
|
|
)
|
|
|
|
with open(template_tex_file, "r") as infile:
|
|
|
|
body = infile.read()
|
|
|
|
body = body.replace(TEX_TEXT_TO_REPLACE, expression)
|
|
|
|
with open(result, "w") as outfile:
|
|
|
|
outfile.write(body)
|
|
|
|
return result
|
|
|
|
|
2016-04-17 12:59:53 -07:00
|
|
|
def tex_to_dvi(tex_file):
|
|
|
|
result = tex_file.replace(".tex", ".dvi")
|
|
|
|
if not os.path.exists(result):
|
2015-10-28 17:18:50 -07:00
|
|
|
commands = [
|
2016-04-17 12:59:53 -07:00
|
|
|
"latex",
|
2015-10-28 17:18:50 -07:00
|
|
|
"-interaction=batchmode",
|
|
|
|
"-output-directory=" + TEX_DIR,
|
|
|
|
tex_file,
|
|
|
|
"> /dev/null"
|
|
|
|
]
|
|
|
|
#TODO, Error check
|
|
|
|
os.system(" ".join(commands))
|
|
|
|
return result
|
|
|
|
|
2016-04-17 12:59:53 -07:00
|
|
|
def dvi_to_svg(dvi_file, regen_if_exists = False):
|
2015-10-28 17:18:50 -07:00
|
|
|
"""
|
|
|
|
Converts a dvi, which potentially has multiple slides, into a
|
|
|
|
directory full of enumerated pngs corresponding with these slides.
|
|
|
|
Returns a list of PIL Image objects for these images sorted as they
|
|
|
|
where in the dvi
|
|
|
|
"""
|
2016-04-17 12:59:53 -07:00
|
|
|
result = dvi_file.replace(".dvi", ".svg")
|
|
|
|
if not os.path.exists(result):
|
2015-10-28 17:18:50 -07:00
|
|
|
commands = [
|
2016-04-17 12:59:53 -07:00
|
|
|
"dvisvgm",
|
|
|
|
dvi_file,
|
|
|
|
"-n",
|
|
|
|
"-v",
|
|
|
|
"0",
|
|
|
|
"-o",
|
2016-04-17 00:31:38 -07:00
|
|
|
result,
|
2016-04-17 12:59:53 -07:00
|
|
|
"> /dev/null"
|
2015-10-28 17:18:50 -07:00
|
|
|
]
|
|
|
|
os.system(" ".join(commands))
|
2016-04-17 00:31:38 -07:00
|
|
|
return result
|
|
|
|
|
|
|
|
|
2015-10-28 17:18:50 -07:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|