diff --git a/.gitignore b/.gitignore index 3c7c12ac..baebf8dd 100644 --- a/.gitignore +++ b/.gitignore @@ -5,3 +5,4 @@ /.tox /MANIFEST /build +*.iml diff --git a/mammoth/documents.py b/mammoth/documents.py index 7559d406..17936bfd 100644 --- a/mammoth/documents.py +++ b/mammoth/documents.py @@ -82,6 +82,11 @@ class TableCell(HasChildren): class Break(Element): break_type = cobble.field() +@cobble.data +class Size(object): + width = cobble.field() + height = cobble.field() + line_break = Break("line") page_break = Break("page") column_break = Break("column") @@ -97,6 +102,7 @@ class Image(Element): alt_text = cobble.field() content_type = cobble.field() open = cobble.field() + size = cobble.field(default=None) def document(children, notes=None, comments=None): diff --git a/mammoth/docx/body_xml.py b/mammoth/docx/body_xml.py index 8329bcfc..7b4e05e7 100644 --- a/mammoth/docx/body_xml.py +++ b/mammoth/docx/body_xml.py @@ -11,6 +11,8 @@ from .styles_xml import Styles from .uris import replace_fragment, uri_to_zip_entry_name +EMU_PER_PIXEL = 9525 + if sys.version_info >= (3, ): unichr = chr @@ -423,23 +425,32 @@ def inline(element): alt_text = properties.get("descr") else: alt_text = properties.get("title") + dimensions = element.find_child_or_null("wp:extent").attributes + size = documents.Size( + width=str(_emu_to_pixel(dimensions.get("cx"))), + height=str(_emu_to_pixel(dimensions.get("cy"))) + ) + blips = element.find_children("a:graphic") \ .find_children("a:graphicData") \ .find_children("pic:pic") \ .find_children("pic:blipFill") \ .find_children("a:blip") - return _read_blips(blips, alt_text) + return _read_blips(blips, alt_text, size) - def _read_blips(blips, alt_text): - return _ReadResult.concat(lists.map(lambda blip: _read_blip(blip, alt_text), blips)) + def _emu_to_pixel(emu): + return int(round(float(emu) / EMU_PER_PIXEL)) - def _read_blip(element, alt_text): - return _read_image(lambda: _find_blip_image(element), alt_text) + def _read_blips(blips, alt_text, size): + return _ReadResult.concat(lists.map(lambda blip: _read_blip(blip, alt_text, size), blips)) - def _read_image(find_image, alt_text): + def _read_blip(element, alt_text, size): + return _read_image(lambda: _find_blip_image(element), alt_text, size) + + def _read_image(find_image, alt_text, size=None): image_path, open_image = find_image() content_type = content_types.find_content_type(image_path) - image = documents.image(alt_text=alt_text, content_type=content_type, open=open_image) + image = documents.image(alt_text=alt_text, content_type=content_type, size=size, open=open_image) if content_type in ["image/png", "image/gif", "image/jpeg", "image/svg+xml", "image/tiff"]: messages = [] @@ -478,14 +489,37 @@ def open_image(): return image_path, open_image - def read_imagedata(element): + def shape(element): + if len(element.children) == 1: + imagedata = element.find_child("v:imagedata") + if imagedata: + size = _read_shape_size(element) + return read_imagedata(imagedata, size) + return read_child_elements(element) + + def _read_shape_size(element): + style_attribute = element.attributes.get("style") + if not style_attribute: + return None + style = style_attribute.split(";") + width = _extract_size_from_style("width", style) + height = _extract_size_from_style("height", style) + size = documents.Size(width=width, height=height) + return size + + def _extract_size_from_style(style_name, style): + with_column = "{}:".format(style_name) + raw_size = next(iter(filter(lambda s: s.startswith(with_column), style))) + return raw_size.replace(with_column, "") + + def read_imagedata(element, style=None): relationship_id = element.attributes.get("r:id") if relationship_id is None: warning = results.warning("A v:imagedata element without a relationship ID was ignored") return _empty_result_with_message(warning) else: title = element.attributes.get("o:title") - return _read_image(lambda: _find_embedded_image(relationship_id), title) + return _read_image(lambda: _find_embedded_image(relationship_id), title, style) def note_reference_reader(note_type): def note_reference(element): @@ -522,7 +556,7 @@ def read_sdt(element): "v:group": read_child_elements, "v:rect": read_child_elements, "v:roundrect": read_child_elements, - "v:shape": read_child_elements, + "v:shape": shape, "v:textbox": read_child_elements, "w:txbxContent": read_child_elements, "w:pict": pict, diff --git a/mammoth/images.py b/mammoth/images.py index 9dba353f..89770803 100644 --- a/mammoth/images.py +++ b/mammoth/images.py @@ -8,6 +8,9 @@ def convert_image(image): attributes = func(image).copy() if image.alt_text: attributes["alt"] = image.alt_text + if image.size: + attributes["width"] = image.size.width + attributes["height"] = image.size.height return [html.element("img", attributes)] diff --git a/setup.py b/setup.py index 7da0d14d..751686f2 100644 --- a/setup.py +++ b/setup.py @@ -12,7 +12,7 @@ def read(fname): name='mammoth', version='1.4.15', description='Convert Word documents from docx to simple and clean HTML and Markdown', - long_description=read("README"), + long_description=read("README.md"), author='Michael Williamson', author_email='mike@zwobble.org', url='http://github.com/mwilliamson/python-mammoth', diff --git a/tests/cli_tests.py b/tests/cli_tests.py index 4728c6b0..82c2ef01 100644 --- a/tests/cli_tests.py +++ b/tests/cli_tests.py @@ -36,7 +36,7 @@ def html_is_written_to_file_if_output_file_is_set(): def inline_images_are_included_in_output_if_writing_to_single_file(): docx_path = test_path("tiny-picture.docx") result = _local.run(["mammoth", docx_path]) - assert_equal(b"""

""", result.output) + assert_equal(b"""

""", result.output) @istest @@ -50,7 +50,7 @@ def images_are_written_to_separate_files_if_output_dir_is_set(): assert_equal(b"", result.stderr_output) assert_equal(b"", result.output) with open(output_path) as output_file: - assert_equal("""

""", output_file.read()) + assert_equal("""

""", output_file.read()) with open(image_path, "rb") as image_file: assert_equal(_image_base_64, base64.b64encode(image_file.read())) diff --git a/tests/conversion_tests.py b/tests/conversion_tests.py index c4e667f3..49752380 100644 --- a/tests/conversion_tests.py +++ b/tests/conversion_tests.py @@ -529,6 +529,19 @@ def images_have_alt_tags_if_available(): image_html = parse_xml(io.StringIO(result.value)) assert_equal('It\'s a hat', image_html.attributes["alt"]) +@istest +def images_have_width_and_height_tags_if_available(): + image = documents.image( + alt_text=None, + content_type="image/png", + size=documents.Size(width="42", height="51"), + open=lambda: io.BytesIO(b"abc") + ) + result = convert_document_element_to_html(image) + image_html = parse_xml(io.StringIO(result.value)) + assert_equal('42', image_html.attributes["width"]) + assert_equal('51', image_html.attributes["height"]) + @istest def can_define_custom_conversion_for_images(): diff --git a/tests/docx/body_xml_tests.py b/tests/docx/body_xml_tests.py index 94061fda..37e0d425 100644 --- a/tests/docx/body_xml_tests.py +++ b/tests/docx/body_xml_tests.py @@ -4,7 +4,7 @@ import sys from precisely import assert_that, is_sequence -from nose.tools import istest, assert_equal +from nose.tools import istest, assert_equal, assert_is_none from nose_parameterized import parameterized, param import funk @@ -961,18 +961,18 @@ class ImageTests(object): IMAGE_RELATIONSHIP_ID = "rId5" def _read_embedded_image(self, element): + return self._read_embedded_images(element)[0] + + def _read_embedded_images(self, element): relationships = Relationships([ _image_relationship(self.IMAGE_RELATIONSHIP_ID, "media/hat.png"), ]) - mocks = funk.Mocks() docx_file = mocks.mock() funk.allows(docx_file).open("word/media/hat.png").returns(io.BytesIO(self.IMAGE_BYTES)) - content_types = mocks.mock() funk.allows(content_types).find_content_type("word/media/hat.png").returns("image/png") - - return _read_and_get_document_xml_element( + return _read_and_get_document_xml_elements( element, content_types=content_types, relationships=relationships, @@ -980,20 +980,73 @@ def _read_embedded_image(self, element): ) @istest - def can_read_imagedata_elements_with_rid_attribute(self): - imagedata_element = xml_element("v:imagedata", { - "r:id": self.IMAGE_RELATIONSHIP_ID, - "o:title": "It's a hat" - }) + def can_read_shape_elements_with_rid_and_size_attributes(self): + shape_element = xml_element("v:shape", {"style": "width:31.5pt;height:38.25pt"}, [ + xml_element("v:imagedata", { + "r:id": self.IMAGE_RELATIONSHIP_ID, + "o:title": "It's a hat" + }) + ]) - image = self._read_embedded_image(imagedata_element) + image = self._read_embedded_image(shape_element) assert_equal(documents.Image, type(image)) assert_equal("It's a hat", image.alt_text) assert_equal("image/png", image.content_type) + assert_equal(documents.Size(width="31.5pt", height="38.25pt"), image.size) with image.open() as image_file: assert_equal(self.IMAGE_BYTES, image_file.read()) + @istest + def cannot_resize_shape_with_multiple_nodes(self): + shape_element = xml_element("v:shape", {"style": "width:31.5pt;height:38.25pt"}, [ + xml_element("v:imagedata", { + "r:id": self.IMAGE_RELATIONSHIP_ID, + "o:title": "It's a hat" + }), + xml_element("v:textbox", {}, [ + xml_element("w:txbxContent", {}, [ + _paragraph_with_style_id("textbox-content") + ]) + ]) + ]) + + nodes = self._read_embedded_images(shape_element) + + assert_equal(2, len(nodes)) + image_node = nodes[0] + assert_equal(documents.Image, type(image_node)) + assert_equal("It's a hat", image_node.alt_text) + assert_is_none(image_node.size) + + @istest + def can_read_shape_elements_with_unused_style_elements(self): + shape_element = xml_element("v:shape", {"style": "width:31.5pt;position:absolute;height:38.25pt"}, [ + xml_element("v:imagedata", { + "r:id": self.IMAGE_RELATIONSHIP_ID, + "o:title": "It's a hat" + }) + ]) + + image = self._read_embedded_image(shape_element) + + assert_equal(documents.Image, type(image)) + assert_equal(documents.Size(width="31.5pt", height="38.25pt"), image.size) + + @istest + def can_read_shape_elements_with_inch_size_attributes(self): + shape_element = xml_element("v:shape", {"style": "width:0.58in;height:0.708in"}, [ + xml_element("v:imagedata", { + "r:id": self.IMAGE_RELATIONSHIP_ID, + "o:title": "It's a hat" + }) + ]) + + image = self._read_embedded_image(shape_element) + + assert_equal(documents.Image, type(image)) + assert_equal(documents.Size(width="0.58in", height="0.708in"), image.size) + @istest def when_imagedata_element_has_no_relationship_id_then_it_is_ignored_with_warning(self): imagedata_element = xml_element("v:imagedata") @@ -1009,6 +1062,7 @@ def can_read_inline_pictures(self): drawing_element = _create_inline_image( blip=_embedded_blip(self.IMAGE_RELATIONSHIP_ID), description="It's a hat", + extent=(9525, 19000) ) image = self._read_embedded_image(drawing_element) @@ -1016,6 +1070,7 @@ def can_read_inline_pictures(self): assert_equal(documents.Image, type(image)) assert_equal("It's a hat", image.alt_text) assert_equal("image/png", image.content_type) + assert_equal(documents.Size(width="1", height="2"), image.size) with image.open() as image_file: assert_equal(self.IMAGE_BYTES, image_file.read()) @@ -1307,9 +1362,9 @@ def _text_element(value): return xml_element("w:t", {}, [xml_text(value)]) -def _create_inline_image(blip, description=None, title=None): +def _create_inline_image(blip, description=None, title=None, extent=None): return xml_element("w:drawing", {}, [ - xml_element("wp:inline", {}, _create_image_elements(blip, description=description, title=title)) + xml_element("wp:inline", {}, _create_image_elements(blip, description=description, title=title, extent=extent)) ]) @@ -1319,15 +1374,19 @@ def _create_anchored_image(description, blip): ]) -def _create_image_elements(blip, description=None, title=None): +def _create_image_elements(blip, description=None, title=None, extent=None): properties = {} if description is not None: properties["descr"] = description if title is not None: properties["title"] = title - + extent = { + "cx": extent[0] if extent else "0", + "cy": extent[1] if extent else "0" + } return [ xml_element("wp:docPr", properties), + xml_element("wp:extent", extent), xml_element("a:graphic", {}, [ xml_element("a:graphicData", {}, [ xml_element("pic:pic", {}, [ diff --git a/tests/images_tests.py b/tests/images_tests.py index cf2e2844..9962d252 100644 --- a/tests/images_tests.py +++ b/tests/images_tests.py @@ -17,11 +17,16 @@ def data_uri_encodes_images_in_base64(): image = mammoth.documents.Image( alt_text=None, content_type="image/jpeg", + size=mammoth.documents.Size(width="800", height="600"), open=lambda: io.BytesIO(image_bytes), ) result = mammoth.images.data_uri(image) assert_that(result, contains( - has_properties(attributes={"src": ""}), + has_properties(attributes={ + "src": "", + "width": "800", + "height": "600", + }), )) diff --git a/tests/mammoth_tests.py b/tests/mammoth_tests.py index 1dcbaa9c..175c6cc8 100644 --- a/tests/mammoth_tests.py +++ b/tests/mammoth_tests.py @@ -112,7 +112,7 @@ def warning_if_style_mapping_is_not_understood(): def inline_images_referenced_by_path_relative_to_part_are_included_in_output(): with open(test_path("tiny-picture.docx"), "rb") as fileobj: result = mammoth.convert_to_html(fileobj=fileobj) - assert_equal("""

""", result.value) + assert_equal("""

""", result.value) assert_equal([], result.messages) @@ -120,7 +120,7 @@ def inline_images_referenced_by_path_relative_to_part_are_included_in_output(): def inline_images_referenced_by_path_relative_to_base_are_included_in_output(): with open(test_path("tiny-picture-target-base-relative.docx"), "rb") as fileobj: result = mammoth.convert_to_html(fileobj=fileobj) - assert_equal("""

""", result.value) + assert_equal("""

""", result.value) assert_equal([], result.messages) @@ -128,7 +128,7 @@ def inline_images_referenced_by_path_relative_to_base_are_included_in_output(): def images_stored_outside_of_document_are_included_in_output(): with open(test_path("external-picture.docx"), "rb") as fileobj: result = mammoth.convert_to_html(fileobj=fileobj) - assert_equal("""

""", result.value) + assert_equal("""

""", result.value) assert_equal([], result.messages) @@ -173,7 +173,7 @@ def convert_image(image): with open(test_path("tiny-picture.docx"), "rb") as fileobj: result = mammoth.convert_to_html(fileobj=fileobj, convert_image=convert_image) - assert_equal("""

""", result.value) + assert_equal("""

""", result.value) assert_equal([], result.messages)