# encoding: utf-8 # text.rb : Implements PDF text primitives # # Copyright May 2008, Gregory Brown. All Rights Reserved. # # This is free software. Please see the LICENSE and COPYING files for details. require "zlib" require "pdf/core/text" require_relative "text/formatted" require_relative "text/box" module Prawn module Text include PDF::Core::Text include Prawn::Text::Formatted # No-Break Space Prawn::Text::NBSP = " " # Zero Width Space (indicate word boundaries without a space) Prawn::Text::ZWSP = [8203].pack("U") # Soft Hyphen (invisible, except when causing a line break) Prawn::Text::SHY = "­" # @group Stable API # If you want text to flow onto a new page or between columns, this is the # method to use. If, instead, if you want to place bounded text outside of # the flow of a document (for captions, labels, charts, etc.), use Text::Box # or its convenience method text_box. # # Draws text on the page. Prawn attempts to wrap the text to fit within your # current bounding box (or margin_box if no bounding box is being used). # Text will flow onto the next page when it reaches the bottom of the # bounding box. Text wrap in Prawn does not re-flow linebreaks, so if you # want fully automated text wrapping, be sure to remove newlines before # attempting to draw your string. # # == Examples # # pdf.text "Will be wrapped when it hits the edge of your bounding box" # pdf.text "This will be centered", :align => :center # pdf.text "This will be right aligned", :align => :right # pdf.text "This includes inline " + # "formatting", :inline_format => true # # If your font contains kerning pair data that Prawn can parse, the # text will be kerned by default. You can disable kerning by including # a false :kerning option. If you want to disable kerning on an # entire document, set default_kerning = false for that document # # === Text Positioning Details # # The text is positioned at font.ascender below the baseline, # making it easy to use this method within bounding boxes and spans. # # == Encoding # # Note that strings passed to this function should be encoded as UTF-8. # If you get unexpected characters appearing in your rendered document, # check this. # # If the current font is a built-in one, although the string must be # encoded as UTF-8, only characters that are available in WinAnsi # are allowed. # # If an empty box is rendered to your PDF instead of the character you # wanted it usually means the current font doesn't include that character. # # == Options (default values marked in []) # # :inline_format:: # boolean. If true, then the string parameter is interpreted # as a HTML-esque string that recognizes the following tags # (assuming the default text formatter is used): # \:: bold # \:: italic # \:: underline # \:: strikethrough # \:: subscript # \:: superscript # \:: # with the following attributes (using double or single quotes) # size="24":: # attribute for setting size # character_spacing="2.5":: # attribute for setting character spacing # name="Helvetica":: # attribute for setting the font. The font name must be an # AFM font with the desired faces or must be a font that is # already registered using Prawn::Document#font_families # \:: # with the following attributes # rgb="ffffff" or rgb="#ffffff":: # c="100" m="100" y="100" k="100":: # \:: # with the following attributes # href="http://example.com":: an external link # Note that you must explicitly underline and color using the # appropriate tags if you which to draw attention to the link # # :kerning:: boolean. Whether or not to use kerning (if it # is available with the current font) # [value of document.default_kerning?] # :size:: number. The font size to use. [current font # size] # :color:: an RGB color ("ff0000") or CMYK array [10, 20, 30, 40]. # :character_spacing:: number. The amount of space to add # to or remove from the default character # spacing. [0] # :style:: The style to use. The requested style must be part of # the current font familly. [current style] # :indent_paragraphs:: number. The amount to indent the # first line of each paragraph. Omit this # option if you do not want indenting # :direction:: # :ltr, :rtl, Direction of the text (left-to-right # or right-to-left) [value of document.text_direction] # :fallback_fonts:: # An array of font names. Each name must be the name of an AFM font or # the name that was used to register a family of TTF fonts (see # Prawn::Document#font_families). If present, then each glyph will be # rendered using the first font that includes the glyph, starting with # the current font and then moving through :fallback_fonts from # left to right. # :align:: # :left, :center, :right, or # :justify Alignment within the bounding box # [:left if direction is :ltr, :right if direction is :rtl] # :valign:: :top, :center, or :bottom. # Vertical alignment within the bounding box [:top] # :leading:: # number. Additional space between lines [value of # document.default_leading] # :final_gap:: boolean. If true, then the space between # each line is included below the last line; # otherwise, document.y is placed just below the # descender of the last line printed [true] # :mode:: The text rendering mode to use. Use this to specify if the # text should render with the fill color, stroke color or # both. See the comments to text_rendering_mode() to see # a list of valid options. [0] # # == Exceptions # # Raises ArgumentError if :at option included # # Raises Prawn::Errrors::CannotFit if not wide enough to print # any text # def text(string, options={}) return false if string.nil? # we modify the options. don't change the user's hash options = options.dup if p = options[:inline_format] p = [] unless p.is_a?(Array) options.delete(:inline_format) array = self.text_formatter.format(string, *p) else array = [{ :text => string }] end formatted_text(array, options) end # Draws formatted text to the page. # Formatted text is comprised of an array of hashes, where each hash defines # text and format information. See Text::Formatted#formatted_text_box for # more information on the structure of this array # # == Example # # text([{ :text => "hello" }, # { :text => "world", # :size => 24, # :styles => [:bold, :italic] }]) # # == Options # # Accepts the same options as #text # # == Exceptions # # Same as for #text # def formatted_text(array, options={}) options = inspect_options_for_text(options.dup) if color = options.delete(:color) array = array.map do |fragment| fragment[:color] ? fragment : fragment.merge(:color => color) end end if @indent_paragraphs self.text_formatter.array_paragraphs(array).each do |paragraph| options[:skip_encoding] = false remaining_text = draw_indented_formatted_line(paragraph, options) options[:skip_encoding] = true if @no_text_printed # unless this paragraph was an empty line unless @all_text_printed @bounding_box.move_past_bottom options[:skip_encoding] = false remaining_text = draw_indented_formatted_line(paragraph, options) options[:skip_encoding] = true end end remaining_text = fill_formatted_text_box(remaining_text, options) draw_remaining_formatted_text_on_new_pages(remaining_text, options) end else remaining_text = fill_formatted_text_box(array, options) options[:skip_encoding] = true draw_remaining_formatted_text_on_new_pages(remaining_text, options) end end # Draws text on the page, beginning at the point specified by the :at option # the string is assumed to be pre-formatted to properly fit the page. # # pdf.draw_text "Hello World", :at => [100,100] # pdf.draw_text "Goodbye World", :at => [50,50], :size => 16 # # If your font contains kerning pair data that Prawn can parse, the # text will be kerned by default. You can disable kerning by including # a false :kerning option. If you want to disable kerning on an # entire document, set default_kerning = false for that document # # === Text Positioning Details: # # Prawn will position your text by the left-most edge of its baseline, and # flow along a single line. (This means that :align will not work) # # == Rotation # # Text can be rotated before it is placed on the canvas by specifying the # :rotate option with a given angle. Rotation occurs counter-clockwise. # # == Encoding # # Note that strings passed to this function should be encoded as UTF-8. # If you get unexpected characters appearing in your rendered document, # check this. # # If the current font is a built-in one, although the string must be # encoded as UTF-8, only characters that are available in WinAnsi # are allowed. # # If an empty box is rendered to your PDF instead of the character you # wanted it usually means the current font doesn't include that character. # # == Options (default values marked in []) # # :at:: [x, y](required). The position at which to start the text # :kerning:: boolean. Whether or not to use kerning (if it # is available with the current font) # [value of default_kerning?] # :size:: number. The font size to use. [current font # size] # :style:: The style to use. The requested style must be part of # the current font familly. [current style] # # :rotate:: number. The angle to which to rotate text # # == Exceptions # # Raises ArgumentError if :at option omitted # # Raises ArgumentError if :align option included # def draw_text(text, options) options = inspect_options_for_draw_text(options.dup) # dup because normalize_encoding changes the string text = text.to_s.dup save_font do process_text_options(options) font.normalize_encoding!(text) font_size(options[:size]) { draw_text!(text, options) } end end # Low level text placement method. All font and size alterations # should already be set # def draw_text!(text, options) x,y = map_to_absolute(options[:at]) add_text_content(text,x,y,options) end # Gets height of text in PDF points. # Same options as #text, except as noted. # Not compatible with :indent_paragraphs option # # ==Example # # height_of("hello\nworld") # # == Exceptions # # Raises NotImplementedError if :indent_paragraphs # option included # # Raises Prawn::Errrors::CannotFit if not wide enough to print # any text # def height_of(string, options={}) height_of_formatted([{ :text => string }], options) end # Gets height of formatted text in PDF points. # See documentation for #height_of. # # ==Example # # height_of_formatted([{ :text => "hello" }, # { :text => "world", # :size => 24, # :styles => [:bold, :italic] }]) # def height_of_formatted(array, options={}) if options[:indent_paragraphs] raise NotImplementedError, ":indent_paragraphs option not available" + "with height_of" end process_final_gap_option(options) box = Text::Formatted::Box.new(array, options.merge(:height => 100000000, :document => self)) box.render(:dry_run => true) height = box.height height += box.line_gap + box.leading if @final_gap height end private def draw_remaining_formatted_text_on_new_pages(remaining_text, options) while remaining_text.length > 0 @bounding_box.move_past_bottom previous_remaining_text = remaining_text remaining_text = fill_formatted_text_box(remaining_text, options) break if remaining_text == previous_remaining_text end end def draw_indented_formatted_line(string, options) indent(@indent_paragraphs) do fill_formatted_text_box(string, options.dup.merge(:single_line => true)) end end def fill_formatted_text_box(text, options) merge_text_box_positioning_options(options) box = Text::Formatted::Box.new(text, options) remaining_text = box.render @no_text_printed = box.nothing_printed? @all_text_printed = box.everything_printed? self.y -= box.height self.y -= box.line_gap + box.leading if @final_gap remaining_text end def merge_text_box_positioning_options(options) bottom = @bounding_box.stretchy? ? @margin_box.absolute_bottom : @bounding_box.absolute_bottom options[:height] = y - bottom options[:width] = bounds.width options[:at] = [@bounding_box.left_side - @bounding_box.absolute_left, y - @bounding_box.absolute_bottom] end def inspect_options_for_draw_text(options) if options[:at].nil? raise ArgumentError, "The :at option is required for draw_text" elsif options[:align] raise ArgumentError, "The :align option does not work with draw_text" end if options[:kerning].nil? then options[:kerning] = default_kerning? end valid_options = PDF::Core::Text::VALID_OPTIONS + [:at, :rotate] Prawn.verify_options(valid_options, options) options end def inspect_options_for_text(options) if options[:at] raise ArgumentError, ":at is no longer a valid option with text." + "use draw_text or text_box instead" end process_final_gap_option(options) process_indent_paragraphs_option(options) options[:document] = self options end def process_final_gap_option(options) @final_gap = options[:final_gap].nil? || options[:final_gap] options.delete(:final_gap) end def process_indent_paragraphs_option(options) @indent_paragraphs = options[:indent_paragraphs] options.delete(:indent_paragraphs) end def move_text_position(dy) bottom = @bounding_box.stretchy? ? @margin_box.absolute_bottom : @bounding_box.absolute_bottom @bounding_box.move_past_bottom if (y - dy) < bottom self.y -= dy end end end