# -*- encoding: utf-8; frozen_string_literal: true -*- # #-- # This file is part of HexaPDF. # # HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby # Copyright (C) 2014-2021 Thomas Leitner # # HexaPDF is free software: you can redistribute it and/or modify it # under the terms of the GNU Affero General Public License version 3 as # published by the Free Software Foundation with the addition of the # following permission added to Section 15 as permitted in Section 7(a): # FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY # THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON # INFRINGEMENT OF THIRD PARTY RIGHTS. # # HexaPDF is distributed in the hope that it will be useful, but WITHOUT # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or # FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public # License for more details. # # You should have received a copy of the GNU Affero General Public License # along with HexaPDF. If not, see . # # The interactive user interfaces in modified source and object code # versions of HexaPDF must display Appropriate Legal Notices, as required # under Section 5 of the GNU Affero General Public License version 3. # # In accordance with Section 7(b) of the GNU Affero General Public # License, a covered work must retain the producer line in every PDF that # is created or manipulated using HexaPDF. # # If the GNU Affero General Public License doesn't fit your need, # commercial licenses are available at . #++ require 'hexapdf/document' require 'hexapdf/layout' module HexaPDF # The composer class can be used to create PDF documents from scratch. It uses Frame and Box # objects underneath. # # == Usage # # First, a new Composer objects needs to be created, either using ::new or the utility method # ::create. # # On creation a HexaPDF::Document object is created as well the first page and an accompanying # HexaPDF::Layout::Frame object. The frame is used by the various methods for general document # layout tasks, like positioning of text, images, and so on. By default, it covers the whole page # except the margin area. How the frame gets created can be customized by overriding the # #create_frame method. # # Once the Composer object is created, its methods can be used to draw text, images, ... on the # page. Behind the scenes HexaPDF::Layout::Box (and subclass) objects are created and drawn on the # page via the frame. # # The base style that is used by all these boxes can be defined using the #base_style method which # returns a HexaPDF::Layout::Style object. The only style property that is set by default is the # font (Times) because otherwise there would be problems with text drawing operations (font is the # only style property that has no valid default value). # # If the frame of a page is full and a box doesn't fit anymore, a new page is automatically # created. The box is either split into two boxes where one fits on the first page and the other # on the new page, or it is drawn completely on the new page. A new page can also be created by # calling the #new_page method. # # The #x and #y methods provide the point where the next box would be drawn if it fits the # available space. This information can be used, for example, for custom drawing operations # through #canvas which provides direct access to the HexaPDF::Content::Canvas object of the # current page. # # When using #canvas and modifying the graphics state, care has to be taken to avoid problems with # later box drawing operations since the graphics state cannot completely be reset (e.g. # transformations of the canvas cannot always be undone). So it is best to save the graphics state # before and restore it afterwards. # # == Example # # HexaPDF::Composer.create('output.pdf', margin: 36) do |pdf| # pdf.base_style.font_size(20).align(:center) # pdf.text("Hello World", valign: :center) # end class Composer # Creates a new PDF document and writes it to +output+. The +options+ are passed to ::new. # # Example: # # HexaPDF::Composer.create('output.pdf', margin: 36) do |pdf| # ... # end def self.create(output, **options, &block) new(**options, &block).write(output) end # The PDF document that is created. attr_reader :document # The current page (a HexaPDF::Type::Page object). attr_reader :page # The Content::Canvas of the current page. Can be used to perform arbitrary drawing operations. attr_reader :canvas # The Layout::Frame for automatic box placement. attr_reader :frame # The base style which is used when no explicit style is provided to methods (e.g. to #text). attr_reader :base_style # Creates a new Composer object and optionally yields it to the given block. # # page_size:: # Can be any valid predefined page size (see Type::Page::PAPER_SIZE) or an array [llx, lly, # urx, ury] specifying a custom page size. # # page_orientation:: # Specifies the orientation of the page, either +:portrait+ or +:landscape+. Only used if # +page_size+ is one of the predefined page sizes. # # margin:: # The margin to use. See Layout::Style::Quad#set for possible values. def initialize(page_size: :A4, page_orientation: :portrait, margin: 36) #:yields: composer @document = HexaPDF::Document.new @page_size = page_size @page_orientation = page_orientation @margin = Layout::Style::Quad.new(margin) new_page @base_style = Layout::Style.new(font: 'Times') yield(self) if block_given? end # Creates a new page, making it the current one. # # If any of +page_size+, +page_orientation+ or +margin+ are set, they will be used instead of # the default values and will become the default values. # # Examples: # # composer.new_page # uses the default values # composer.new_page(page_size: :A5, margin: [72, 36]) def new_page(page_size: nil, page_orientation: nil, margin: nil) @page_size = page_size if page_size @page_orientation = page_orientation if page_orientation @margin = Layout::Style::Quad.new(margin) if margin @page = @document.pages.add(@page_size, orientation: @page_orientation) @canvas = @page.canvas create_frame end # The x-position of the cursor inside the current frame. def x @frame.x end # The y-position of the cursor inside the current frame. def y @frame.y end # Writes the PDF document to the given output. # # See Document#write for details. def write(output, optimize: true, **options) @document.write(output, optimize: optimize, **options) end # Draws the given text at the current position into the current frame. # # This method is the main method for displaying text on a PDF page. It uses a Layout::TextBox # behind the scenes to do the actual work. # # The text will be positioned at the current position if possible. Otherwise the next best # position is used. If the text doesn't fit onto the current page or only partially, new pages # are created automatically. # # The arguments +width+ and +height+ are used as constraints and are respected when fitting the # box. # # The text is styled using the given +style+ object (see Layout::Style) or, if no style object # is specified, the base style (see #base_style). If any additional style +options+ are # specified, the used style is copied and the additional styles are applied. # # See HexaPDF::Layout::TextBox for details. def text(str, width: 0, height: 0, style: nil, **options) style = update_style(style, options) draw_box(Layout::TextBox.new([Layout::TextFragment.create(str, style)], width: width, height: height, style: style)) end # Draws text like #text but where parts of it can be formatted differently. # # The argument +data+ needs to be an array of String or Hash objects: # # * A String object is treated like {text: data}. # # * Hashes can contain any style properties and the following special keys: # # text:: The text to be formatted. # # link:: A URL that should be linked to. If no text is provided but a link, the link is used # as text. # # style:: A Layout::Style object to use as basis instead of the style created from the +style+ # and +options+ arguments. # # If any style properties are set, the used style is copied and the additional properties # applied. # # Examples: # # composer.formatted_text(["Some string"]) # The same as #text # composer.formatted_text(["Some ", {text: "string", fill_color: 128}] # composer.formatted_text(["Some ", {link: "https://example.com", text: "Example"}]) # composer.formatted_text(["Some ", {text: "string", style: my_style}]) def formatted_text(data, width: 0, height: 0, style: nil, **options) style = update_style(style, options) data.map! do |hash| if hash.kind_of?(String) Layout::TextFragment.create(hash, style) else link = hash.delete(:link) text = hash.delete(:text) || link || "" used_style = update_style(hash.delete(:style), options) || style if link || !hash.empty? used_style = used_style.dup hash.each {|key, value| used_style.send(key, value) } used_style.overlays.add(:link, uri: link) if link end Layout::TextFragment.create(text, used_style) end end draw_box(Layout::TextBox.new(data, width: width, height: height, style: style)) end # Draws the given image file at the current position. # # See #text for details on +width+, +height+, +style+ and +options+. def image(file, width: 0, height: 0, style: nil, **options) style = update_style(style, options) image = document.images.add(file) draw_box(Layout::ImageBox.new(image, width: width, height: height, style: style)) end # Draws the given Layout::Box. # # The box is drawn into the current frame if possible. If it doesn't fit, the box is split. If # it still doesn't fit, a new region of the frame is determined and then the process starts # again. # # If none or only some parts of the box fit into the current frame, one or more new pages are # created for the rest of the box. def draw_box(box) drawn_on_page = true while true if @frame.fit(box) @frame.draw(@canvas, box) break elsif @frame.full? new_page drawn_on_page = false else draw_box, box = @frame.split(box) if draw_box @frame.draw(@canvas, draw_box) drawn_on_page = true elsif !@frame.find_next_region unless drawn_on_page raise HexaPDF::Error, "Box doesn't fit on empty page" end new_page drawn_on_page = false end end end end private # Creates the frame into which boxes are layed out when a new page is created. def create_frame media_box = @page.box @frame = Layout::Frame.new(media_box.left + @margin.left, media_box.bottom + @margin.bottom, media_box.width - @margin.left - @margin.right, media_box.height - @margin.bottom - @margin.top) end # Updates the Layout::Style object +style+ if one is provided, or the base style, with the style # options to make it work in all cases. def update_style(style, options = {}) style ||= base_style style = style.dup.update(**options) unless options.empty? style.font(base_style.font) unless style.font? style.font(@document.fonts.add(style.font)) unless style.font.respond_to?(:pdf_object) style end end end