From 629670a6775dc5320e1031a557ad4d6ebe4e0246 Mon Sep 17 00:00:00 2001 From: Dan Allen Date: Fri, 14 Oct 2016 20:45:33 -0600 Subject: [PATCH 1/3] only fit image within bounds of running content if contain option is set --- lib/asciidoctor-pdf/converter.rb | 18 ++++++++++-------- 1 file changed, 10 insertions(+), 8 deletions(-) diff --git a/lib/asciidoctor-pdf/converter.rb b/lib/asciidoctor-pdf/converter.rb index 10df12915..c1f7759f3 100644 --- a/lib/asciidoctor-pdf/converter.rb +++ b/lib/asciidoctor-pdf/converter.rb @@ -2130,12 +2130,11 @@ def layout_running_content periphery, doc, opts = {} if (val.include? ':') && val =~ ImageAttributeValueRx && ::File.readable?(path = (ThemeLoader.resolve_theme_asset $1, (doc.attr 'pdf-stylesdir'))) attrs = (AttributeList.new $2).parse - width = resolve_explicit_width attrs, bounds.width - # QUESTION should we lookup and scale intrinsic width if explicit width is not given? - unless width - width = [bounds.width, (intrinsic_image_dimensions path)[:width] * 0.75].min + unless (width = resolve_explicit_width attrs, bounds.width) + # QUESTION should we lookup and scale intrinsic width if explicit width is not given? + width = (intrinsic_image_dimensions path)[:width] * 0.75 end - side_content[position] = { path: path, width: width } + side_content[position] = { path: path, width: width, fit: (attrs.key? 'contain-option') } else side_content[position] = val end @@ -2307,9 +2306,12 @@ def layout_running_content periphery, doc, opts = {} float do # NOTE bounding_box is redundant if trim_v_padding is 0 bounding_box [colspec[:x], cursor - trim_padding[0]], width: colspec[:width], height: (bounds.height - trim_v_padding) do - #image content[:path], vposition: trim_img_valign, position: colspec[:align], width: content[:width] - # NOTE use :fit to prevent image from overflowing page (at the cost of scaling it) - image content[:path], vposition: trim_img_valign, position: colspec[:align], fit: [content[:width], bounds.height] + if content[:fit] + # NOTE use :fit to prevent image from overflowing page (at the cost of scaling it) + image content[:path], vposition: trim_img_valign, position: colspec[:align], fit: [content[:width], bounds.height] + else + image content[:path], vposition: trim_img_valign, position: colspec[:align], width: content[:width] + end end end when ::String From 60017b76c223e2cbc1992d98045d114ea018d737 Mon Sep 17 00:00:00 2001 From: Dan Allen Date: Fri, 14 Oct 2016 20:56:59 -0600 Subject: [PATCH 2/3] fix alignment for SVG image in running content --- lib/asciidoctor-pdf/prawn_ext/images.rb | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/lib/asciidoctor-pdf/prawn_ext/images.rb b/lib/asciidoctor-pdf/prawn_ext/images.rb index 12cfd1a2b..380a05187 100644 --- a/lib/asciidoctor-pdf/prawn_ext/images.rb +++ b/lib/asciidoctor-pdf/prawn_ext/images.rb @@ -11,7 +11,8 @@ def extended base def image file, opts = {} # FIXME handle case when SVG is a File or IO object if ::String === file && (file.downcase.end_with? '.svg') - opts[:at] ||= bounds.top_left + opts[:position] ||= :left + opts[:vposition] ||= cursor opts[:fallback_font_name] ||= default_svg_font if respond_to? :default_svg_font svg (::IO.read file), opts else From 4c7d1bbb1c28daa90274cda4271f7b62c1d6bcba Mon Sep 17 00:00:00 2001 From: Dan Allen Date: Fri, 14 Oct 2016 23:45:43 -0600 Subject: [PATCH 3/3] resolves #625 honor pdfwidth attribute for image in running content - honor pdfwidth attribute for image in running content - position SVG correctly, including vertical alignment - only fit image to bounds if fit=contain or fit=scale-down attribute is used - calculate image width from percentage value based on column width, not content width --- lib/asciidoctor-pdf/converter.rb | 98 ++++++++++++++------- lib/asciidoctor-pdf/prawn_ext/extensions.rb | 7 ++ lib/asciidoctor-pdf/prawn_ext/images.rb | 2 - 3 files changed, 72 insertions(+), 35 deletions(-) diff --git a/lib/asciidoctor-pdf/converter.rb b/lib/asciidoctor-pdf/converter.rb index c1f7759f3..2475e9a71 100644 --- a/lib/asciidoctor-pdf/converter.rb +++ b/lib/asciidoctor-pdf/converter.rb @@ -952,7 +952,7 @@ def convert_image node, opts = {} if !width && (svg_obj.document.root.attributes.key? 'width') # NOTE scale native width & height by 75% to convert px to pt; restrict width to bounds.width if (adjusted_w = [bounds.width, rendered_w * 0.75].min) != rendered_w - # FIXME would be nice to have a resize method (available as of prawn-svg 0.25.2); for now, just reconstruct + # FIXME use new resize method (available as of prawn-svg 0.25.2); reconstruct manually for now svg_obj = ::Prawn::Svg::Interface.new svg_data, self, position: alignment, width: (rendered_w = adjusted_w), @@ -1019,12 +1019,15 @@ def convert_image node, opts = {} img_x, img_y = image_position rendered_w, rendered_h, position: alignment link_box = [img_x, (img_y - rendered_h), (img_x + rendered_w), img_y] end + image_top = cursor embed_image image_obj, image_info, width: rendered_w, position: alignment if link link_annotation link_box, Border: [0, 0, 0], A: { Type: :Action, S: :URI, URI: link.as_pdf } end + # NOTE Asciidoctor disables automatic advancement of cursor for images, so handle it manually + move_down rendered_h if cursor == image_top rescue => e warn %(asciidoctor: WARNING: could not embed image: #{image_path}; #{e.message}) end @@ -2120,34 +2123,8 @@ def layout_running_content periphery, doc, opts = {} doc.set_attr 'document-title', doctitle.main doc.set_attr 'document-subtitle', doctitle.subtitle doc.set_attr 'page-count', num_pages - - # TODO move this to a method so it can be reused; cache results - content_dict = PageSides.inject({}) do |acc, side| - side_content = {} - ColumnPositions.each do |position| - if (val = @theme[%(#{periphery}_#{side}_#{position}_content)]) - # TODO support image URL (using resolve_image_path) - if (val.include? ':') && val =~ ImageAttributeValueRx && - ::File.readable?(path = (ThemeLoader.resolve_theme_asset $1, (doc.attr 'pdf-stylesdir'))) - attrs = (AttributeList.new $2).parse - unless (width = resolve_explicit_width attrs, bounds.width) - # QUESTION should we lookup and scale intrinsic width if explicit width is not given? - width = (intrinsic_image_dimensions path)[:width] * 0.75 - end - side_content[position] = { path: path, width: width, fit: (attrs.key? 'contain-option') } - else - side_content[position] = val - end - end - end - # NOTE set fallbacks if not explicitly disabled - if side_content.empty? && periphery == :footer && @theme[%(footer_#{side}_content)] != 'none' - side_content = { side == :recto ? :right : :left => '{page-number}' } - end - - acc[side] = side_content - acc - end + allow_uri_read = doc.attr? 'allow-uri-read' + svg_fallback_font = default_svg_font if periphery == :header trim_line_metrics = calc_line_metrics(@theme.header_line_height || @theme.base_line_height) @@ -2243,6 +2220,40 @@ def layout_running_content periphery, doc, opts = {} acc end + # TODO move this to a method so it can be reused; cache results + content_dict = PageSides.inject({}) do |acc, side| + side_content = {} + ColumnPositions.each do |position| + if (val = @theme[%(#{periphery}_#{side}_#{position}_content)]) + # TODO support image URL (using resolve_image_path) + if (val.include? ':') && val =~ ImageAttributeValueRx && + ::File.readable?(path = (ThemeLoader.resolve_theme_asset $1, (doc.attr 'pdf-stylesdir'))) + attrs = (AttributeList.new $2).parse + col_width = colspec_dict[side][position][:width] + if (fit = attrs['fit']) == 'contain' + width = col_width + else + unless (width = resolve_explicit_width attrs, col_width) + # QUESTION should we lookup and scale intrinsic width if explicit width is not given? + width = (intrinsic_image_dimensions path)[:width] * 0.75 + end + width = col_width if fit == 'scale-down' && width > col_width + end + side_content[position] = { path: path, width: width, fit: !!fit } + else + side_content[position] = val + end + end + end + # NOTE set fallbacks if not explicitly disabled + if side_content.empty? && periphery == :footer && @theme[%(footer_#{side}_content)] != 'none' + side_content = { side == :recto ? :right : :left => '{page-number}' } + end + + acc[side] = side_content + acc + end + stamps = {} if trim_bg_color || trim_border_color PageSides.each do |side| @@ -2306,11 +2317,32 @@ def layout_running_content periphery, doc, opts = {} float do # NOTE bounding_box is redundant if trim_v_padding is 0 bounding_box [colspec[:x], cursor - trim_padding[0]], width: colspec[:width], height: (bounds.height - trim_v_padding) do - if content[:fit] - # NOTE use :fit to prevent image from overflowing page (at the cost of scaling it) - image content[:path], vposition: trim_img_valign, position: colspec[:align], fit: [content[:width], bounds.height] + if (img_path = content[:path]).downcase.end_with? '.svg' + svg_data = ::IO.read img_path + svg_opts = { + position: colspec[:align], + vposition: trim_img_valign, + width: content[:width], + # TODO enforce jail in safe mode + enable_file_requests_with_root: (::File.dirname img_path), + enable_web_requests: allow_uri_read, + fallback_font_name: svg_fallback_font + } + svg_obj = ::Prawn::Svg::Interface.new svg_data, self, svg_opts + # FIXME use new resize method (available as of prawn-svg 0.25.2); reconstruct manually for now + if content[:fit] && (rendered_h = svg_obj.document.sizing.output_height) > bounds.height + svg_opts[:width] *= bounds.height / rendered_h + svg_obj = ::Prawn::Svg::Interface.new svg_data, self, svg_opts + end + svg_obj.draw else - image content[:path], vposition: trim_img_valign, position: colspec[:align], width: content[:width] + img_opts = { position: colspec[:align], vposition: trim_img_valign } + if content[:fit] + img_opts[:fit] = [content[:width], bounds.height] + else + img_opts[:width] = content[:width] + end + image img_path, img_opts end end end diff --git a/lib/asciidoctor-pdf/prawn_ext/extensions.rb b/lib/asciidoctor-pdf/prawn_ext/extensions.rb index 1cd85562d..e6f194bb1 100644 --- a/lib/asciidoctor-pdf/prawn_ext/extensions.rb +++ b/lib/asciidoctor-pdf/prawn_ext/extensions.rb @@ -435,6 +435,13 @@ def move_up n super unless n == 0 end + # Override built-in move_text_position method to prevent Prawn from advancing + # to next page if image doesn't fit before rendering image. + #-- + # NOTE could use :at option when calling image/embed_image instead + def move_text_position h + end + # Short-circuits the call to the built-in move_down operation # when n is 0. # diff --git a/lib/asciidoctor-pdf/prawn_ext/images.rb b/lib/asciidoctor-pdf/prawn_ext/images.rb index 380a05187..84717c0aa 100644 --- a/lib/asciidoctor-pdf/prawn_ext/images.rb +++ b/lib/asciidoctor-pdf/prawn_ext/images.rb @@ -11,8 +11,6 @@ def extended base def image file, opts = {} # FIXME handle case when SVG is a File or IO object if ::String === file && (file.downcase.end_with? '.svg') - opts[:position] ||= :left - opts[:vposition] ||= cursor opts[:fallback_font_name] ||= default_svg_font if respond_to? :default_svg_font svg (::IO.read file), opts else