15.4. Создание документов в формате PDF с помощью библиотеки PDF::Writer
Библиотека PDF::Writer
предназначена для создания PDF-документов из программы на языке Ruby. Ее можно установить из gem-пакета или скачать с сайта RubyForge. Последовательность создания документа проста:
require 'rubygems'
require 'pdf/writer'
pdf = PDF::Writer.new
15.4.1. Основные концепции и приемы
Одна из серьезных проблем, встающих перед любым дизайнером документов, - текстовые шрифты. Библиотека PDF::Writer
поддерживает пять основных шрифтов, причем первые три допускают полужирное и курсивное начертание:
• Times-Roman
• Helvetica
• Courier
• ZapfDingbats
• Symbol
Если шрифт не указан, по умолчанию предполагается Helvetica. При выборе шрифта можно создать таблицу замены символов, которая позволяет имитировать символы, не имеющие графического начертания или отсутствующие в кодовой странице. В шрифтах Times-Roman, Helvetica и Courier по 315 печатаемых символов (из них у 149 есть предопределенные байтовые коды); в шрифте Symbol — 190 символов (у 189 есть предопределенные коды), а в шрифте ZapfDingbats — 202 символа (всем соответствуют коды). Шрифты представлены в кодировке Adobe, но в момент выбора шрифта отдельные символы можно переопределить.
Текущая версия не позволяет напечатать все 315 символов, определенных в шрифтовом файле, поскольку после того как шрифт выбран, изменить таблицу замены символов уже невозможно. В последующих версиях PDF::Writer
эта проблема будет решена.
В следующем примере мы задали для PDF-документа шрифт Times-Roman. Программа чтения PDF- файлов будет считать, что текст представлен в кодировке WinAnsiEncoding
, но вместо символа с кодом 0x01
подставит глиф «lozenge» (ромб), еще увидим его ниже (листинг 15.11).
pdf.select_font 'Times-Roman',
{ :encoding => 'WinAnsiEncoding',
:differences => {0x01 => 'lozenge'}
}
Библиотека PDF::Writer
располагает средствами для форматирования текста и создания таблиц, которые хорошо документированы. Не так очевидно, что пока не срабатывает автоматическая разбивка на страницы, можно форматировать страницу вручную весьма любопытными способами. С помощью переноса осей и масштабирования мы можем нарисовать четыре страницы на одной.
В текущей версии PDF::Writer
(1.1.3) каждая такая «страница» должна полностью умещаться на одной физической странице. Если в дело вмешивается механизм автоматического разбиения на страницы, то будет создана новая физическая страница. В следующих версиях усовершенствованный вариант этой техники будет работать и для многоколонных страниц.
Для демонстрации создадим метод quadrant
(листинг 15.10). Он войдет также составной частью в длинный пример из следующего раздела, который преследует две цели: показать, как создается документ из четырех страниц и как можно разместить четыре страницы PDF-документа на одной странице книги, сэкономив тем самым место.
def quadrant(pdf, quad)
raise unless block_given?
mx = pdf.absolute_x_middle
my = pdf.absolute_y_middle
pdf.save_state
case quad
when :ul
pdf.translate_axis(0, my)
when :ur
pdf.translate_axis(mx, my)
when :ll
nil # pdf.translate_axis(0, 0)
when :lr
pdf.translate_axis(mx, 0)
end
pdf.scale_axis(0.5, 0.5)
pdf.у = pdf.page_height
yield
pdf.restore_state
end
Здесь каждая страница целиком строится в отдельном блоке. Таким образом, мы можем изменять масштаб и положение осей, никак не затрагивая код построения страницы. Первым делом мы, конечно, сохраняем текущее состояние. Это позволит нам не восстанавливать вручную масштаб и начало системы координат по завершении работы. Перед тем как приступать к конструированию, мы помещаем начало координат квадранта в нужное место страницы (pdf.translate_axis x, y)
.
Предположим, что начало координат находится не в точке (0, 0)
, а в точке (50, 50)
. Тогда отрезок из точки (15, 20)
в точку (35, 40)
на самом деле будет соединять точки с координатами (65, 70)
и (85, 90)
. Но код рисования отрезка об этом ничего не знает.
После переноса оси (то есть сдвига начала координат) мы можем изменить масштаб вдоль оси. Чтобы получить четыре квадранта, следует уменьшить вдвое масштаб по осям X и Y (pdf.scale_axis 0.5, 0.5
). Иными словами, если бы сейчас я провел отрезок между точками (0, 0)
и (90, 90)
, то без переноса осей он соединял бы точки с физическими координатами (0, 0)
и (45, 45)
, а с переносом — точки с координатами (90, 90)
и (135, 135)
. В любом случае будет проведена линия вдоль диагонали длиной 90 единиц измерения. Просто из-за масштабирования сами единицы стали в два раза меньше.
Затем мы отдаем управление блоку, а когда он закончит работу, восстанавливаем состояние, вызывая предоставленный библиотекой метод restore_state
. Иначе пришлось бы вручную