# -*- 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-2024 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/dictionary'
require 'hexapdf/error'
module HexaPDF
class Document
# This class provides methods for creating and managing the destinations of a PDF file.
#
# A destination describes a particular view of a PDF document, consisting of the page, the view
# location and a magnification factor. See Destination for details.
#
# Such destinations may be directly specified where needed, e.g. for link annotations, or they
# may be named and later referenced through the name. This class allows to create destinations
# with or without a name.
#
# See: PDF2.0 s12.3.2
class Destinations
# Wraps an explicit destination array to allow easy access to query its properties.
#
# A *destination array* has the form
#
# [page, type, *arguments]
#
# where +page+ is either a page object or a page number (in case of a destination to a page in
# a remote PDF document), +type+ is the destination type (see below) and +arguments+ are the
# required arguments for the specific type of destination.
#
# == Destination Types
#
# There are eight different types of destinations, each taking different arguments. The
# arguments are marked up in the list below and are in the correct order for use in the
# destination array. The first name in the list is the PDF internal name, the second one the
# explicit, more descriptive one used by HexaPDF (though the PDF internal name can also be
# used):
#
# :XYZ, :xyz::
# Display the page with the given (+left+, +top+) coordinate at the upper-left corner of
# the window and the specified magnification (+zoom+) factor. A +nil+ value for any
# argument means not changing it from the current value.
#
# :Fit, :fit_page::
# Display the page so that it fits horizontally and vertically within the window.
#
# :FitH, :fit_page_horizontal::
# Display the page so that it fits horizontally within the window, with the given +top+
# coordinate being at the top of the window. A +nil+ value for +top+ means not changing it
# from the current value.
#
# :FitV, :fit_page_vertical::
# Display the page so that it fits vertically within the window, with the given +left+
# coordinate being at the left of the window. A +nil+ value for +left+ means not changing
# it from the current value.
#
# :FitR, :fit_rectangle::
# Display the page so that the rectangle specified by (+left+, +bottom+)-(+right+, +top+)
# fits horizontally and vertically within the window.
#
# :FitB, :fit_bounding_box::
# Display the page so that its bounding box fits horizontally and vertically within the
# window.
#
# :FitBH, :fit_bounding_box_horizontal::
# Display the page so that its bounding box fits horizontally within the window, with the
# given +top+ coordinate being at the top of the window. A +nil+ value for +top+ means not
# changing it from the current value.
#
# :FitBV, :fit_bounding_box_vertical::
# Display the page so that its bounding box fits vertically within the window, with the
# given +left+ coordinate being at the left of the window. A +nil+ value for +left+ means
# not changing it from the current value.
class Destination
TYPE_MAPPING = { #:nodoc:
XYZ: :xyz,
Fit: :fit_page,
FitH: :fit_page_horizontal,
FitV: :fit_page_vertical,
FitR: :fit_rectangle,
FitB: :fit_bounding_box,
FitBH: :fit_bounding_box_horizontal,
FitBV: :fit_bounding_box_vertical,
}
REVERSE_TYPE_MAPPING = Hash[*TYPE_MAPPING.flatten.reverse] #:nodoc:
# Returns +true+ if the destination is valid.
def self.valid?(destination)
TYPE_MAPPING.key?(destination[1]) &&
(destination[0].kind_of?(Integer) || destination[0]&.type == :Page) &&
destination[2..-1].all? {|item| item.nil? || item.kind_of?(Numeric) }
end
# Creates a new Destination for the given +destination+ specification which may be an
# explicit destination array or a dictionary with a /D entry (as allowed for a named
# destination).
def initialize(destination)
@destination = if destination.kind_of?(HexaPDF::Dictionary) || destination.kind_of?(Hash)
destination[:D]
else
destination
end
end
# Returns +true+ if the destination references a destination in a remote document.
def remote?
@destination[0].kind_of?(Numeric)
end
# Returns the referenced page.
#
# The return value is either a page object or, in case of a destination to a remote
# document, a page number.
def page
@destination[0]
end
# Returns the type of destination.
def type
TYPE_MAPPING[@destination[1]]
end
# Returns the argument +left+ if used by the destination, raises an error otherwise.
def left
case type
when :xyz, :fit_page_vertical, :fit_rectangle, :fit_bounding_box_vertical
@destination[2]
else
raise HexaPDF::Error, "No such argument for destination type #{type}"
end
end
# Returns the argument +top+ if used by the destination, raises an error otherwise.
def top
case type
when :xyz
@destination[3]
when :fit_page_horizontal, :fit_bounding_box_horizontal
@destination[2]
when :fit_rectangle
@destination[5]
else
raise HexaPDF::Error, "No such argument for destination type #{type}"
end
end
# Returns the argument +right+ if used by the destination, raises an error otherwise.
def right
case type
when :fit_rectangle
@destination[4]
else
raise HexaPDF::Error, "No such argument for destination type #{type}"
end
end
# Returns the argument +bottom+ if used by the destination, raises an error otherwise.
def bottom
case type
when :fit_rectangle
@destination[3]
else
raise HexaPDF::Error, "No such argument for destination type #{type}"
end
end
# Returns the argument +zoom+ if used by the destination, raises an error otherwise.
def zoom
case type
when :xyz
@destination[4]
else
raise HexaPDF::Error, "No such argument for destination type #{type}"
end
end
# Returns +true+ if the destination is valid.
def valid?
self.class.valid?(@destination)
end
# Returns the wrapped destination array.
def value
@destination
end
end
include Enumerable
# Creates a new Destinations object for the given PDF document.
def initialize(document)
@document = document
end
# :call-seq:
# destinations.use_or_create(name) -> name
# destinations.use_or_create(destination) -> destination
# destinations.use_or_create(page) -> destination
# destinations.use_or_create(type:, page, **options) -> destination
#
# Uses the given destination name/array or creates a destination array based on the given
# arguments.
#
# This is the main utility method for other parts of HexaPDF for getting a valid destination
# array based on various different types of the given arguments:
#
# String::
#
# If a string is provided, it is assumed to be a named destination. If the named
# destination exists, the destination itself is returned. Otherwise an error is raised.
#
# Array::
#
# If a valid destination array is provided, it is returned. Otherwise an error is raised.
#
# Page dictionary::
#
# If the value is a valid page dictionary object, a fit to page (#create_fit_page)
# destination array is created and returned.
#
# Integer::
#
# If the value is an integer, it is interpreted as a zero-based page index and a fit to
# page (#create_fit_page) destination array is created and returned.
#
# Hash containing at least :type and :page::
#
# If the value is a hash, the :type key specifies the type of the destination that should
# be created and the :page key the target page. Which other keys are allowed depends on
# the destination type, so see the various create_XXX methods. Uses #create to do the job.
def use_or_create(value)
case value
when String
if self[value]
value
else
raise HexaPDF::Error, "Named destination '#{value}' doesn't exist"
end
when Array
raise HexaPDF::Error, "Invalid destination array" unless Destination.new(value).valid?
value
when HexaPDF::Dictionary
if value.type != :Page
raise HexaPDF::Error, "Invalid dictionary type '#{value.type}' given, needs to be a page"
end
create_fit_page(value)
when Integer
if value < 0 || value >= @document.pages.count
raise ArgumentError, "Page index #{value} out of bounds"
end
create_fit_page(@document.pages[value])
when Hash
type = value.delete(:type) { raise ArgumentError, "Missing keyword argument :type" }
page = value.delete(:page) { raise ArgumentError, "Missing keyword argument :page" }
create(type, page, **value)
else
raise ArgumentError, "Invalid argument type '#{value.class}'"
end
end
# :call-seq:
# destinations.create(type, page, **options) -> dest or name
#
# Creates a new destination array with the given +type+ (see Destination for all available
# type names; PDF internal type names are also allowed) and +page+ by calling the respective
# +create_type+ method.
def create(type, page, **options)
send("create_#{Destination::TYPE_MAPPING.fetch(type, type)}", page, **options)
end
# :call-seq:
# destinations.create_xyz(page, left: nil, top: nil, zoom: nil) -> dest
# destinations.create_xyz(page, name: nil, left: nil, top: nil, zoom: nil) -> name
#
# Creates a new xyz destination array for the given arguments and returns it or, in case
# a name is given, the name.
#
# The arguments +page+, +left+, +top+ and +zoom+ are described in detail in the Destination
# class description.
#
# If the argument +name+ is given, the created destination array is added to the destinations
# name tree under that name for reuse later, overwriting an existing entry if there is one.
def create_xyz(page, name: nil, left: nil, top: nil, zoom: nil)
destination = [page, Destination::REVERSE_TYPE_MAPPING.fetch(:xyz), left, top, zoom]
name ? (add(name, destination); name) : destination
end
# :call-seq:
# destinations.create_fit_page(page) -> dest
# destinations.create_fit_page(page, name: nil) -> name
#
# Creates a new fit to page destination array for the given arguments and returns it or, in
# case a name is given, the name.
#
# The argument +page+ is described in detail in the Destination class description.
#
# If the argument +name+ is given, the created destination array is added to the destinations
# name tree under that name for reuse later, overwriting an existing entry if there is one.
def create_fit_page(page, name: nil)
destination = [page, Destination::REVERSE_TYPE_MAPPING.fetch(:fit_page)]
name ? (add(name, destination); name) : destination
end
# :call-seq:
# destinations.create_fit_page_horizontal(page, top: nil) -> dest
# destinations.create_fit_page_horizontal(page, name: nil, top: nil) -> name
#
# Creates a new fit page horizontal destination array for the given arguments and returns it
# or, in case a name is given, the name.
#
# The arguments +page and +top+ are described in detail in the Destination class description.
#
# If the argument +name+ is given, the created destination array is added to the destinations
# name tree under that name for reuse later, overwriting an existing entry if there is one.
def create_fit_page_horizontal(page, name: nil, top: nil)
destination = [page, Destination::REVERSE_TYPE_MAPPING.fetch(:fit_page_horizontal), top]
name ? (add(name, destination); name) : destination
end
# :call-seq:
# destinations.create_fit_page_vertical(page, left: nil) -> dest
# destinations.create_fit_page_vertical(page, name: nil, left: nil) -> name
#
# Creates a new fit page vertical destination array for the given arguments and returns it or,
# in case a name is given, the name.
#
# The arguments +page and +left+ are described in detail in the Destination class description.
#
# If the argument +name+ is given, the created destination array is added to the destinations
# name tree under that name for reuse later, overwriting an existing entry if there is one.
def create_fit_page_vertical(page, name: nil, left: nil)
destination = [page, Destination::REVERSE_TYPE_MAPPING.fetch(:fit_page_vertical), left]
name ? (add(name, destination); name) : destination
end
# :call-seq:
# destinations.create_fit_rectangle(page, left:, bottom:, right:, top:) -> dest
# destinations.create_fit_rectangle(page, name: nil, left:, bottom:, right:, top:) -> name
#
# Creates a new fit to rectangle destination array for the given arguments and returns it or,
# in case a name is given, the name.
#
# The arguments +page+, +left+, +bottom+, +right+ and +top+ are described in detail in the
# Destination class description.
#
# If the argument +name+ is given, the created destination array is added to the destinations
# name tree under that name for reuse later, overwriting an existing entry if there is one.
def create_fit_rectangle(page, left:, bottom:, right:, top:, name: nil)
destination = [page, Destination::REVERSE_TYPE_MAPPING.fetch(:fit_rectangle),
left, bottom, right, top]
name ? (add(name, destination); name) : destination
end
# :call-seq:
# destinations.create_fit_bounding_box(page) -> dest
# destinations.create_fit_bounding_box(page, name: nil) -> name
#
# Creates a new fit to bounding box destination array for the given arguments and returns it
# or, in case a name is given, the name.
#
# The argument +page+ is described in detail in the Destination class description.
#
# If the argument +name+ is given, the created destination array is added to the destinations
# name tree under that name for reuse later, overwriting an existing entry if there is one.
def create_fit_bounding_box(page, name: nil)
destination = [page, Destination::REVERSE_TYPE_MAPPING.fetch(:fit_bounding_box)]
name ? (add(name, destination); name) : destination
end
# :call-seq:
# destinations.create_fit_bounding_box_horizontal(page, top: nil) -> dest
# destinations.create_fit_bounding_box_horizontal(page, name: nil, top: nil) -> name
#
# Creates a new fit bounding box horizontal destination array for the given arguments and
# returns it or, in case a name is given, the name.
#
# The arguments +page and +top+ are described in detail in the Destination class description.
#
# If the argument +name+ is given, the created destination array is added to the destinations
# name tree under that name for reuse later, overwriting an existing entry if there is one.
def create_fit_bounding_box_horizontal(page, name: nil, top: nil)
destination = [page, Destination::REVERSE_TYPE_MAPPING.fetch(:fit_bounding_box_horizontal), top]
name ? (add(name, destination); name) : destination
end
# :call-seq:
# destinations.create_fit_bounding_box_vertical(page, left: nil) -> dest
# destinations.create_fit_bounding_box_vertical(page, name: nil, left: nil) -> name
#
# Creates a new fit bounding box vertical destination array for the given arguments and
# returns it or, in case a name is given, the name.
#
# The arguments +page and +left+ are described in detail in the Destination class description.
#
# If the argument +name+ is given, the created destination array is added to the destinations
# name tree under that name for reuse later, overwriting an existing entry if there is one.
def create_fit_bounding_box_vertical(page, name: nil, left: nil)
destination = [page, Destination::REVERSE_TYPE_MAPPING.fetch(:fit_bounding_box_vertical), left]
name ? (add(name, destination); name) : destination
end
# :call-seq:
# destinations.add(name, destination)
#
# Adds the given +destination+ under +name+ (a String) to the destinations name tree.
#
# If the name does already exist, an error is raised.
def add(name, destination)
destinations.add_entry(name, destination)
end
# :call-seq:
# destinations.delete(name) -> destination
#
# Deletes the destination specified via +name+ (a String) from the destinations name tree and
# returns it or +nil+ if no destination was registered under that name.
def delete(name)
destinations.delete_entry(name)
end
# :call-seq:
# destinations.resolve(string_name) -> destination or nil
# destinations.resolve(symbol_name) -> destination or nil
# destinations.resolve(dest_array) -> destination or nil
#
# Resolves the given value to a valid destination object, if possible, or otherwise returns
# +nil+.
#
# * If the given value is a string, it is treated as a destination name and looked up in the
# destination name tree.
#
# * If the given value is a symbol, it is treated as an old-style destination name and looked
# up in the destination dictionary.
#
# * If the given value is an array, it is treated as a destination array itself.
def resolve(value)
result = case value
when String
destinations.find_entry(value)
when PDFArray
value.value
when Array
value
when Symbol
@document.catalog[:Dests]&.[](value)
end
result = Destination.new(result) if result
result&.valid? ? result : nil
end
# :call-seq:
# destinations[name] -> destination
#
# Returns the destination registered under the given +name+ (a String) or +nil+ if no
# destination was registered under that name.
def [](name)
destinations.find_entry(name)
end
# :call-seq:
# destinations.each {|name, dest| block } -> destinations
# destinations.each -> Enumerator
#
# Iterates over all named destinations of the PDF, yielding the name and the destination
# wrapped into a Destination object.
def each
return to_enum(__method__) unless block_given?
destinations.each_entry do |name, dest|
yield(name, Destination.new(dest))
end
self
end
private
# Returns the root of the destinations name tree.
def destinations
@document.catalog.names.destinations
end
end
end
end