gramps/src/docgen/LPRDoc.py
2007-06-28 06:02:03 +00:00

1246 lines
43 KiB
Python

#
# Gramps - a GTK+/GNOME based genealogy program
#
# Copyright (C) 2000-2006 Donald N. Allingham
# Copyright (C) 2007 Brian G. Matherly
#
# Modifications and feature additions:
# 2002 Donald A. Peterson
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program 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 General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#
# Written by Billy C. Earney, 2003-2004
# Modified by Alex Roitman, 2004-2005
# $Id$
"""LPR document generator"""
#------------------------------------------------------------------------
#
# python modules
#
#------------------------------------------------------------------------
from string import punctuation
from gettext import gettext as _
#------------------------------------------------------------------------
#
# GNOME/GTK Modules
#
#------------------------------------------------------------------------
import gtk.gdk
import Errors
try:
import gnomeprint, gnomeprint.ui
except ImportError:
raise Errors.UnavailableError(
_("Cannot be loaded because python bindings "
"for GNOME print are not installed"))
### FIXME ###
if gnomeprint.Context.__dict__.has_key('grestore'):
support_photos = 1
else:
support_photos = 0
print "LPRDoc: Photos and rotated text (used in FanChart)"
print " are not supported for direct printing."
print " Get gnome-python from CVS"
print " or wait for the next gnome-python release."
### end FIXME ###
#------------------------------------------------------------------------
#
# gramps modules
#
#------------------------------------------------------------------------
import BaseDoc
from PluginUtils import register_text_doc, register_draw_doc, register_book_doc
from ReportBase import ReportUtils
rgb_color = ReportUtils.rgb_color
#------------------------------------------------------------------------
#
# Constants
#
#------------------------------------------------------------------------
# Spacing in points (distance between the bottoms of two adjacent lines)
_LINE_SPACING = 20
# Elevation of superscripts: a fraction of it's size
_SUPER_ELEVATION_FRACTION = 0.3
# Number of points to subtract to get the superscrip size
_SUPER_SIZE_REDUCTION = 2
# Factor which multiplies the font size to get line spacing for the font
_EXTRA_SPACING_FACTOR = 1.2
# Grey color to use for box shadows
_SHADOW_COLOR = (192,192,192)
# Font constants -- specific for gnome-print
_TTF_FREEFONT = (
('FreeSerif Medium','FreeSerif Bold','FreeSerif Italic','FreeSerif BoldItalic' ),
('FreeSans Medium','FreeSans Bold','FreeSans Oblique','FreeSans BoldOblique'),
('FreeMono Medium','FreeMono Bold','FreeMono Oblique','FreeMono BoldOblique')
)
_MS_TTFONT = (
('Times New Roman Regular','Times New Roman Bold','Times New Roman Italic','Times New Roman Bold Italic' ),
('Arial Regular','Arial Bold','Arial Italic','Arial Bold Italic'),
('Courier New Regular','Courier New Bold','Courier New Italic','Courier New Bold Italic')
)
_GNOME_FONT = (
('Serif Regular','Serif Bold','Serif Italic','Serif Bold Italic' ),
('Sans Regular','Sans Bold','Sans Italic','Sans Bold Italic'),
('Monospace Regular','Monospace New Bold','Monospace New Italic','Monospace New Bold Italic')
)
# Search for ttf-freefont first
ttf_not_found = 0
for family in _TTF_FREEFONT:
for font in family:
if font not in gnomeprint.font_list():
ttf_not_found = 1
break
if ttf_not_found:
print "LPRDoc: Free true type fonts not found."
# Search for MS ttfs
ms_not_found = 0
for family in _MS_TTFONT:
for font in family:
if font not in gnomeprint.font_list():
ms_not_found = 1
break
if ms_not_found:
print " Microsoft true type fonts not found."
print " Using Gnome standard fonts."
print " Non-ascii characters will appear garbled in the output."
print " INSTALL Free true type fonts"
print " from http://www.nongnu.org/freefont/"
_FONT_SET = _GNOME_FONT
else:
print " Found Microsoft true type fonts. Will use them for now."
print " These fonts are not free. "
print " You are advised to switch to Free true type fonts"
print " INSTALL Free true type fonts"
print " from http://www.nongnu.org/freefont/"
_FONT_SET = _MS_TTFONT
else:
_FONT_SET = _TTF_FREEFONT
# Formatting directive constants
_LINE_BREAK = "Break"
_BOLD = "Bold"
_SUPER = "Super"
_MONO = "Mono"
_POSTLEADER = "Postleader"
#------------------------------------------------------------------------
#
# Units conversion
#
#------------------------------------------------------------------------
def cm2u(cm):
"""
Convert cm to gnome-print units.
"""
return cm * 72.0 / 2.54
#------------------------------------------------------------------------
#
# font lookup function
#
#------------------------------------------------------------------------
def find_font_from_fontstyle(fontstyle):
"""
This function returns the gnomeprint.Font() object instance
corresponding to the parameters of BaseDoc.FontStyle() object.
fontstyle - a BaseDoc.FontStyle() instance
"""
if fontstyle.get_type_face() == BaseDoc.FONT_SERIF:
family = _FONT_SET[0]
elif fontstyle.get_type_face() == BaseDoc.FONT_SANS_SERIF:
family = _FONT_SET[1]
elif fontstyle.get_type_face() == BaseDoc.FONT_MONOSPACE:
family = _FONT_SET[2]
if fontstyle.get_bold():
if fontstyle.get_italic():
font = family[3]
else:
font = family[1]
elif fontstyle.get_italic():
font = family[2]
else:
font = family[0]
size = fontstyle.get_size()
return gnomeprint.font_find_closest(font,size)
#------------------------------------------------------------------------
#
# basic font-specific text formatting functions
#
#------------------------------------------------------------------------
def get_text_width(text,fontstyle):
"""
This function returns the width of text using given fontstyle
when not formatted.
text - a text whose width to find
fontstyle - a BaseDoc.FontStyle() instance
"""
font = find_font_from_fontstyle(fontstyle)
return font.get_width_utf8(text)
#------------------------------------------------------------------------
#
# add to paragraph taking care of the newline characters
#
#------------------------------------------------------------------------
def append_to_paragraph(paragraph,directive,text):
"""
Add a piece to the paragraph while
taking care of the newline characters.
paragraph - a GnomePrintParagraph() instance
directive - what to do with this piece
text - the text of the corresponding piece
"""
if not directive and not text:
return
text_list = text.split('\n')
for the_text in text_list[:-1]:
paragraph.add_piece(directive,the_text)
paragraph.add_piece(_LINE_BREAK,"")
paragraph.add_piece(directive,text_list[-1:][0])
#------------------------------------------------------------------------
#
# Paragraph class
#
#------------------------------------------------------------------------
class GnomePrintParagraph:
"""
A paragraph abstraction which provides the means for in-paragraph
formatting.
"""
def __init__(self,paragraph_style):
"""
Creates a GnomePrintParapgrah instance.
paragraph_style - an instance of BaseDoc paragraph style object
"""
self.style = paragraph_style
self.fontstyle = self.style.get_font()
self.piece_list = []
self.lines = []
self.height = None
def add_piece(self,directive,text):
"""
Add a piece to the paragraph.
directive - what to do with this piece
text - the text of the corresponding piece
"""
self.piece_list.append((directive,text))
def get_piece_list(self):
"""
Return a list of pieces for the paragraph.
"""
return self.piece_list
def get_fontstyle(self):
"""
Return fontstyle for the paragraph.
"""
return self.fontstyle
def get_alignment(self):
"""
Return requested alignment of the paragraph.
"""
return self.style.get_alignment()
def get_min_width(self):
"""
Determine the minimal width of the paragraph (longest word).
"""
max_word_size = 0
for (directive,text) in self.piece_list:
fontstyle = BaseDoc.FontStyle(self.fontstyle)
if directive == _BOLD:
fontstyle.set_bold(1)
elif directive == _SUPER:
size = fontstyle.get_size()
fontstyle.set_size(size-_SUPER_SIZE_REDUCTION)
elif directive == _MONO:
fontstyle.set_type_face(BaseDoc.FONT_MONOSPACE)
for word in text.split():
length = get_text_width(word,fontstyle)
if length > max_word_size:
max_word_size = length
return max_word_size
def get_height(self,width):
"""
Determine the height the paragraph would have
if formatted for a given width.
width - required formatting width
"""
if not self.lines:
self.format(width)
return self.height
def format(self,width):
"""
Format the paragraph for a given width.
This is a complex procedure. It assembles lines from the paragraph's
pieces. It also sets the height of the whole paragraph and
the widths available after the lines are assembled.
width - required formatting width
"""
if self.lines:
return
width = width - cm2u(self.style.get_right_margin()) \
- cm2u(self.style.get_left_margin())
nlines = 1
avail_width = width
start_piece = end_piece = start_word = end_word = 0
first = 1
for piece_num in range(len(self.piece_list)):
end_piece = piece_num
(directive,text) = self.piece_list[piece_num]
fontstyle = BaseDoc.FontStyle(self.fontstyle)
if directive == _BOLD:
fontstyle.set_bold(1)
elif directive == _SUPER:
size = fontstyle.get_size()
fontstyle.set_size(size-_SUPER_SIZE_REDUCTION)
elif directive == _MONO:
fontstyle.set_type_face(BaseDoc.FONT_MONOSPACE)
if first:
first = 0
avail_width = avail_width - cm2u(self.style.get_first_indent())
if text and avail_width > get_text_width(text,fontstyle):
avail_width -= get_text_width(text,fontstyle)
end_word = len(text.split())
elif directive == _LINE_BREAK:
nlines += 1
end_word = 0
self.lines.append((start_piece,start_word,end_piece,end_word,avail_width))
avail_width = width
start_piece = end_piece
start_word = 0
elif text and avail_width <= get_text_width(text,fontstyle):
# divide up text
textlist = text.split()
the_text = ""
for word_num in range(len(textlist)):
word = textlist[word_num]
if get_text_width(the_text + word + " ",fontstyle) <= avail_width:
the_text = the_text + word + " "
else:
# the_text contains as much as avail_width allows
nlines += 1
end_word = word_num
avail_width -= get_text_width(the_text,fontstyle)
self.lines.append((start_piece,start_word,end_piece,end_word,avail_width))
avail_width = width
the_text = word + " "
start_piece = end_piece
start_word = word_num
# if the_text still contains data, we will want to print it out
if the_text:
avail_width = width - get_text_width(the_text,fontstyle)
end_word = len(textlist)
self.lines.append((start_piece,start_word,end_piece,end_word,avail_width))
self.height = nlines * self.fontstyle.get_size() \
* _EXTRA_SPACING_FACTOR \
+ cm2u(self.style.get_top_margin()
+self.style.get_bottom_margin())
def get_lines(self):
"""
Return a list of assemlbed lines for the paragraph.
Each element is a tuple corresponding to the line's contents:
(start_piece,start_word,end_piece,end_word,avail_width)
"""
return self.lines
#------------------------------------------------------------------------
#
# Photo class
#
#------------------------------------------------------------------------
class GnomePrintPhoto:
"""
A photo abstraction which provides the means for correct photo placement.
Way less complex that paragraph, but still useful.
"""
def __init__(self,name,pos,x_size,y_size):
"""
Creates a GnomePrintPhoto instance.
"""
self.name = name
self.alignment = pos
self.pixbuf = gtk.gdk.pixbuf_new_from_file(name)
self.height = self.pixbuf.get_height()
self.width = self.pixbuf.get_width()
max_size = cm2u(max(x_size,y_size))
self.scale_x = int(max_size * float(self.width)/max(self.height,
self.width))
self.scale_y = int(max_size * float(self.height)/max(self.height,
self.width))
def get_image(self):
"""
Return the raw image of the photo.
"""
return self.pixbuf.get_pixels()
def get_has_alpha(self):
"""
Return has_alpha of the photo.
"""
return self.pixbuf.get_has_alpha()
def get_rowstride(self):
"""
Return the rowstride of the photo.
"""
return self.pixbuf.get_rowstride()
def get_height(self,width=None):
"""
Return the real height of the photo as it should appear on the page.
"""
return self.scale_y
def get_width(self):
"""
Return the real width of the photo as it should appear on the page.
"""
return self.scale_x
def get_min_width(self):
"""
Return the minimum width of the photo as it should appear on the page.
"""
return self.scale_x
def get_image_height(self):
"""
Return the height of the photo in terms of image's pixels.
"""
return self.height
def get_image_width(self):
"""
Return the width of the photo in terms of image's pixels.
"""
return self.width
def get_alignment(self):
"""
Return the alignment of the photo.
"""
return self.alignment
#------------------------------------------------------------------------
#
# LPRDoc class
#
#------------------------------------------------------------------------
class LPRDoc(BaseDoc.BaseDoc,BaseDoc.TextDoc,BaseDoc.DrawDoc):
"""Gnome-print document interface class. Derived from BaseDoc."""
#------------------------------------------------------------------------
#
# General methods
#
#------------------------------------------------------------------------
def open(self,filename):
"""Sets up initialization."""
#set up variables needed to keep track of which state we are in
self.in_table = 0
self.in_cell = 0
self.page_count = 0
self.page_open = 0
self.brand_new_page = 0
self.paragraph = None
self.cell_data = []
self.table_data = []
#create main variables for this print job
self.job = gnomeprint.Job(gnomeprint.config_default())
self.gpc = self.job.get_context()
#find out what the width and height of the page is
width, height = gnomeprint.job_get_page_size_from_config(
self.job.get_config())
self.left_margin = cm2u(self.paper.get_left_margin())
self.right_margin = width - cm2u(self.paper.get_right_margin())
self.top_margin = height - cm2u(self.paper.get_top_margin())
self.bottom_margin = cm2u(self.paper.get_bottom_margin())
self.start_page(self)
self.filename = ""
def close(self):
"""Clean up and close the document."""
#gracefully end page before we close the doc if a page is open
if self.page_open:
self.end_page()
self.job.close()
self.show_print_dialog()
def start_page(self,orientation=None):
"""Create a new page."""
# Don't start new page if it is just started
if self.brand_new_page:
return
#reset variables dealing with opening a page
if (self.page_open):
self.end_page()
self.page_open = 1
self.page_count += 1
self.x = self.left_margin
self.y = self.top_margin
self.gpc.beginpage(str(self.page_count))
self.gpc.moveto(self.x, self.y)
self.brand_new_page = 1
def end_page(self):
"""Close the current page."""
if (self.page_open):
self.page_open = 0
self.gpc.showpage()
self.brand_new_page = 0
def page_break(self):
"Forces a page break, creating a new page."
# If we're already at the very top, relax and do nothing
if not self.brand_new_page:
self.end_page()
self.start_page()
#------------------------------------------------------------------------
#
# Text methods
#
#------------------------------------------------------------------------
def string_width(self,fontstyle,text):
"Override generic Fontscale-based width."
return get_text_width(text,fontstyle)
def start_paragraph(self,style_name,leader=None):
"""Paragraphs handling - A Gramps paragraph is any
single body of text, from a single word, to several sentences.
We assume a linebreak at the end of each paragraph."""
# Instantiate paragraph object and initialize buffers
style_sheet = self.get_style_sheet()
style = style_sheet.get_paragraph_style(style_name)
self.paragraph = GnomePrintParagraph(style)
if leader:
append_to_paragraph(self.paragraph,"",leader)
self.paragraph_directive = _POSTLEADER
else:
self.paragraph_directive = ""
self.paragraph_text = ""
self.brand_new_page = 0
def end_paragraph(self):
"""End the current paragraph."""
# Add current text/directive to paragraoh,
# then either add paragrah to the list of cell's paragraphs
# or print it right away if not in cell
append_to_paragraph(self.paragraph,self.paragraph_directive,
self.paragraph_text)
if self.in_cell:
# We're inside cell. Add paragrah to celldata
self.cell_data.append(self.paragraph)
else:
# paragraph not in table: write it right away
self.x, self.y = self.write_paragraph(self.paragraph,
self.x, self.y,
self.right_margin - self.left_margin)
self.paragraph = None
self.brand_new_page = 0
def start_bold(self):
"""Bold face."""
append_to_paragraph(self.paragraph,self.paragraph_directive,
self.paragraph_text)
self.paragraph_directive = _BOLD
self.paragraph_text = ""
self.brand_new_page = 0
def end_bold(self):
"""End bold face."""
append_to_paragraph(self.paragraph,self.paragraph_directive,
self.paragraph_text)
self.paragraph_directive = ""
self.paragraph_text = ""
self.brand_new_page = 0
def start_superscript(self):
append_to_paragraph(self.paragraph,self.paragraph_directive,
self.paragraph_text)
self.paragraph_directive = _SUPER
self.paragraph_text = ""
self.brand_new_page = 0
def end_superscript(self):
append_to_paragraph(self.paragraph,self.paragraph_directive,
self.paragraph_text)
self.paragraph_directive = ""
self.paragraph_text = ""
self.brand_new_page = 0
def start_table(self,name,style_name):
"""Begin new table."""
# initialize table, compute its width, find number of columns
self.table_data = []
self.in_table = 1
styles = self.get_style_sheet()
self.tbl_style = styles.get_table_style(style_name)
self.ncols = self.tbl_style.get_columns()
self.rownum = -1
self.table_width = (self.right_margin - self.left_margin) * \
self.tbl_style.get_width() / 100.0
self.cell_widths = []
self.gp_cell_styles = []
self.brand_new_page = 0
def end_table(self):
"""Close the table environment."""
# output table contents
self.output_table()
self.in_table = 0
self.y = self.advance_line(self.y)
self.brand_new_page = 0
def start_row(self):
"""Begin a new row."""
# Initialize row, compute cell widths
self.row_data = []
self.rownum = self.rownum + 1
self.cellnum = -1
self.span = 1
self.cell_widths.append([0] * self.ncols)
self.gp_cell_styles.append([None] * self.ncols)
for cell in range(self.ncols):
self.cell_widths[self.rownum][cell] = self.table_width * \
self.tbl_style.get_column_width(cell) / 100.0
self.brand_new_page = 0
def end_row(self):
"""End the row (new line)."""
# add row data to the data we have for the current table
self.table_data.append(self.row_data)
self.brand_new_page = 0
def start_cell(self,style_name,span=1):
"""Add an entry to the table."""
# Initialize a cell, take care of span>1 cases
self.brand_new_page = 0
self.in_cell = 1
self.cell_data = []
self.cellnum = self.cellnum + self.span
self.span = span
styles = self.get_style_sheet()
cstyle = styles.get_cell_style(style_name)
self.gp_cell_styles[self.rownum][self.cellnum] = cstyle
for extra_cell in range(1,span):
self.cell_widths[self.rownum][self.cellnum] += \
self.cell_widths[self.rownum][self.cellnum + extra_cell]
self.cell_widths[self.rownum][self.cellnum + extra_cell] = 0
def end_cell(self):
"""Prepares for next cell."""
# append the cell text to the row data
self.in_cell = 0
self.row_data.append(self.cell_data)
self.brand_new_page = 0
def add_media_object(self,name,pos,x,y):
"""Add photo to report."""
photo = GnomePrintPhoto(name,pos,x,y)
if self.in_cell:
# We're inside cell. Add photo to celldata
self.cell_data.append(photo)
else:
# photo not in table: write it right away
self.x, self.y = self.write_photo(photo,
self.x, self.y,
self.right_margin - self.left_margin)
self.brand_new_page = 0
def write_photo(self,photo,x,y,alloc_width):
"""
Write the photo.
photo - GnomePrintPhoto instance
x,y - coordinates to start at
alloc_width - allocated width
"""
self.brand_new_page = 0
# FIXME -- remove when gnome-python is released and hits every distro
if not support_photos:
return (x,y)
# end FIXME
width = photo.get_width()
height = photo.get_height()
if y - height < self.bottom_margin:
self.end_page()
self.start_page()
y = self.y
if photo.get_alignment() == 'center':
add_x = 0.5* (alloc_width - width)
elif photo.get_alignment() == 'right':
add_x = alloc_width - width
else:
add_x = 0
self.gpc.gsave()
self.gpc.translate(x+add_x,y-height)
self.gpc.scale(width,height)
if photo.get_has_alpha():
self.gpc.rgbaimage(photo.get_image(),
photo.get_image_width(),
photo.get_image_height(),
photo.get_rowstride())
else:
self.gpc.rgbimage(photo.get_image(),
photo.get_image_width(),
photo.get_image_height(),
photo.get_rowstride())
self.gpc.grestore()
y = y - height
return (x,y)
def write_text(self,text,mark=None):
"""Add the text to the paragraph"""
self.brand_new_page = 0
# Take care of superscript tags
super_count = text.count('<super>')
for num in range(super_count):
start = text.find('<super>')
self.paragraph_text = self.paragraph_text + text[:start]
append_to_paragraph(self.paragraph,self.paragraph_directive,self.paragraph_text)
self.paragraph_text = ""
text = text[start+7:]
start = text.find('</super>')
self.paragraph_text = self.paragraph_text + text[:start]
append_to_paragraph(self.paragraph,_SUPER,self.paragraph_text)
self.paragraph_text = ""
text = text[start+8:]
self.paragraph_text = self.paragraph_text + text
def write_note(self,text,format,style_name):
self.brand_new_page = 0
if format == 1:
for line in text.split('\n'):
self.start_paragraph(style_name)
self.paragraph_directive = _MONO
self.write_text(line)
self.end_paragraph()
elif format == 0:
for line in text.split('\n\n'):
self.start_paragraph(style_name)
line = line.replace('\n',' ')
line = ' '.join(line.split())
self.write_text(line)
self.end_paragraph()
#function to help us advance a line
def advance_line(self,y,paragraph=None):
self.brand_new_page = 0
if paragraph:
spacing = paragraph.fontstyle.get_size() * _EXTRA_SPACING_FACTOR
else:
spacing = _LINE_SPACING
new_y = y - spacing
if y < self.bottom_margin:
x = self.x
self.end_page()
self.start_page()
new_y = self.y
self.x = x
return new_y
def write_paragraph(self,paragraph,x,y,width):
"""
Write the contents of the paragraph, observing per-piece info.
paragraph - GnomePrintParagraph instance
x,y - coordinates to start at
width - allocated width
"""
self.brand_new_page = 0
if not paragraph.get_piece_list():
return (x,y)
paragraph.format(width)
x = x + cm2u(paragraph.style.get_left_margin())
width = width - cm2u(paragraph.style.get_right_margin()) \
- cm2u(paragraph.style.get_left_margin())
left_margin = x
no_space = 0
next_no_space = 0
first = 1
if y - paragraph.fontstyle.get_size() * _EXTRA_SPACING_FACTOR \
< self.bottom_margin:
self.end_page()
self.start_page()
x = left_margin
y = self.y
if y != self.top_margin:
y = y - cm2u(paragraph.style.get_top_margin())
line_number = 0
total_lines = len(paragraph.get_lines())
# Loop over lines which were assembled by paragraph.format()
for (start_piece,start_word,end_piece,end_word,avail_width) \
in paragraph.get_lines():
line_number += 1
if paragraph.get_alignment() == BaseDoc.PARA_ALIGN_CENTER:
x = x + 0.5 * avail_width
elif paragraph.get_alignment() == BaseDoc.PARA_ALIGN_RIGHT:
x = x + avail_width
elif paragraph.get_alignment() == BaseDoc.PARA_ALIGN_LEFT:
pass
elif paragraph.get_alignment() == BaseDoc.PARA_ALIGN_JUSTIFY:
print "LPRDoc: Paragraph justification not supported."
print " Falling back to left-justified mode."
if first:
first = 0
x = x + cm2u(paragraph.style.get_first_indent())
y = y - paragraph.fontstyle.get_size() * _EXTRA_SPACING_FACTOR
# Loop over pieces that constitute the line
for piece_num in range(start_piece,end_piece+1):
(directive,text) = paragraph.get_piece_list()[piece_num]
fontstyle = BaseDoc.FontStyle(paragraph.get_fontstyle())
if directive == _BOLD:
fontstyle.set_bold(1)
elif directive == _SUPER:
size = fontstyle.get_size()
fontstyle.set_size(size-_SUPER_SIZE_REDUCTION)
y = y + _SUPER_ELEVATION_FRACTION * fontstyle.get_size()
elif directive == _MONO:
fontstyle.set_type_face(BaseDoc.FONT_MONOSPACE)
elif directive == _POSTLEADER:
x = left_margin
if text == '':
next_no_space = 1
textlist = text.split()
if start_piece == end_piece:
the_textlist = textlist[start_word:end_word]
elif piece_num > start_piece and piece_num < end_piece:
the_textlist = textlist[:]
elif piece_num == start_piece:
the_textlist = textlist[start_word:]
elif piece_num == end_piece:
the_textlist = textlist[:end_word]
the_text = ' '.join(the_textlist)
if piece_num == start_piece \
or directive == _SUPER \
or directive == _POSTLEADER \
or next_no_space \
or no_space \
or (the_text and the_text[0] in punctuation):
spacer = ""
else:
spacer = " "
the_text = spacer + the_text
self.gpc.setfont(find_font_from_fontstyle(fontstyle))
self.gpc.moveto(x, y)
self.gpc.show(the_text)
x = x + get_text_width(the_text,fontstyle)
if directive == _SUPER:
y = y - _SUPER_ELEVATION_FRACTION * fontstyle.get_size()
if directive != _POSTLEADER and next_no_space:
next_no_space = 0
# If this was the linebreak, no space on the next line's start
if end_word:
no_space = 0
else:
no_space = 1
if line_number < total_lines:
y = self.advance_line(y,paragraph)
x = left_margin
x = x - cm2u(paragraph.style.get_left_margin())
y = y - cm2u(paragraph.style.get_bottom_margin())
return (x,y)
def output_table(self):
"""Do calcs on data in table and output data in a formatted way."""
self.brand_new_page = 0
min_col_size = [0] * self.ncols
max_vspace = [0] * len(self.table_data)
for row_num in range(len(self.table_data)):
row = self.table_data[row_num][:]
#do calcs on each row and keep track on max length of each column
for col in range(self.ncols):
if not self.cell_widths[row_num][col]:
continue
padding = cm2u(self.gp_cell_styles[row_num][col].get_padding())
the_max = 0
for paragraph in row[col]:
the_min = paragraph.get_min_width()
if the_min > min_col_size[col]:
min_col_size[col] = the_min
the_max += paragraph.get_height(
self.cell_widths[row_num][col])
the_max += 2 * padding
if the_max > max_vspace[row_num]:
max_vspace[row_num] = the_max
#is table width larger than the width of the paper?
min_table_width = 0
for size in min_col_size:
min_table_width = min_table_width + size
if min_table_width > (self.right_margin - self.left_margin):
print "LPRDoc: Table does not fit onto the page."
#for now we will assume left justification of tables
#output data in table
for row_num in range(len(self.table_data)):
row = self.table_data[row_num]
# If this row puts us below the bottom, start new page here
if self.y - max_vspace[row_num] < self.bottom_margin:
self.end_page()
self.start_page()
x = self.left_margin #reset so that x is at margin
col_y = self.y # all columns start at the same height
for col in range(self.ncols):
if not self.cell_widths[row_num][col]:
continue
self.y = col_y
padding = cm2u(self.gp_cell_styles [row_num][col].get_padding())
for paragraph in row[col]:
if paragraph.__class__.__name__ == 'GnomePrintPhoto':
write_item = self.write_photo
else:
write_item = self.write_paragraph
junk, self.y = write_item(paragraph,
x + padding, self.y - padding,
self.cell_widths[row_num][col] \
- 2 * padding)
x = x + self.cell_widths[row_num][col] # set up margin for this row
self.y = col_y - max_vspace[row_num]
#------------------------------------------------------------------------
#
# Graphic methods
#
#------------------------------------------------------------------------
def draw_path(self,style,path):
self.brand_new_page = 0
style_sheet = self.get_style_sheet()
stype = style_sheet.get_draw_style(style)
self.gpc.setlinewidth(stype.get_line_width())
fill_color = rgb_color(stype.get_fill_color())
color = rgb_color(stype.get_color())
point = path[0]
x = cm2u(point[0]) + self.left_margin
y = self.top_margin - cm2u(point[1])
self.gpc.moveto(x,y)
for point in path[1:]:
x = cm2u(point[0]) + self.left_margin
y = self.top_margin - cm2u(point[1])
self.gpc.lineto(x,y)
self.gpc.closepath()
self.gpc.setrgbcolor(fill_color[0],fill_color[1],fill_color[2])
self.gpc.fill()
point = path[0]
x = cm2u(point[0]) + self.left_margin
y = self.top_margin - cm2u(point[1])
self.gpc.moveto(x,y)
for point in path[1:]:
x = cm2u(point[0]) + self.left_margin
y = self.top_margin - cm2u(point[1])
self.gpc.lineto(x,y)
self.gpc.closepath()
self.gpc.setrgbcolor(color[0],color[1],color[2])
self.gpc.stroke()
self.gpc.setrgbcolor(0,0,0)
def draw_box(self,style,text,x,y, w, h):
#assuming that we start drawing box from current position
style_sheet = self.get_style_sheet()
self.brand_new_page = 0
x = self.left_margin + cm2u(x)
y = self.top_margin - cm2u(y)
box_style = style_sheet.get_draw_style(style)
bh = cm2u(h)
bw = cm2u(w)
if box_style.get_shadow():
ss = cm2u(box_style.get_shadow_space())
color = rgb_color(_SHADOW_COLOR)
path = (
(x+ss,y-bh), (x+ss,y-bh-ss), (x+bw+ss,y-bh-ss),
(x+bw+ss,y-ss), (x+bw,y-ss), (x+bw,y-bh), (x+ss,y-bh),
)
x_i,y_i = path[0]
self.gpc.moveto(x_i,y_i)
for (x_i,y_i) in path[1:]:
self.gpc.lineto(x_i,y_i)
self.gpc.closepath()
self.gpc.setrgbcolor(color[0],color[1],color[2])
self.gpc.fill()
self.gpc.setrgbcolor(0,0,0)
self.gpc.rect_stroked(x,y,bw,-bh)
if text:
para_name = box_style.get_paragraph_style()
para_style = style_sheet.get_paragraph_style(para_name)
fontstyle = para_style.get_font()
lines = text.split('\n')
start_x = x + 0.5 * fontstyle.get_size()
start_y = y - fontstyle.get_size() * _EXTRA_SPACING_FACTOR
for line in lines:
if not line.split():
continue
self.gpc.setfont(find_font_from_fontstyle(fontstyle))
self.gpc.moveto(start_x,start_y)
self.gpc.show(line)
start_y -= fontstyle.get_size() * _EXTRA_SPACING_FACTOR
def draw_text(self,style,text,x,y):
self.brand_new_page = 0
style_sheet = self.get_style_sheet()
box_style = style_sheet.get_draw_style(style)
para_name = box_style.get_paragraph_style()
para_style = style_sheet.get_paragraph_style(para_name)
fontstyle = para_style.get_font()
start_x = self.left_margin + cm2u(x)
start_y = self.top_margin - cm2u(y) - fontstyle.get_size() * _EXTRA_SPACING_FACTOR
self.gpc.setfont(find_font_from_fontstyle(fontstyle))
self.gpc.moveto(start_x,start_y)
self.gpc.show(text)
def center_text(self,style,text,x,y):
self.brand_new_page = 0
style_sheet = self.get_style_sheet()
box_style = style_sheet.get_draw_style(style)
para_name = box_style.get_paragraph_style()
para_style = style_sheet.get_paragraph_style(para_name)
fontstyle = para_style.get_font()
width = get_text_width(text,fontstyle)
start_x = self.left_margin + cm2u(x) - 0.5 * width
start_y = self.top_margin - cm2u(y) \
- fontstyle.get_size() * _EXTRA_SPACING_FACTOR
self.gpc.setfont(find_font_from_fontstyle(fontstyle))
self.gpc.moveto(start_x, start_y)
self.gpc.show(text)
def rotate_text(self,style,text,x,y,angle):
self.brand_new_page = 0
# FIXME - remove when new gnome-python is in all distros
if not support_photos:
return
# end FIXME
style_sheet = self.get_style_sheet()
box_style = style_sheet.get_draw_style(style)
para_name = box_style.get_paragraph_style()
para_style = style_sheet.get_paragraph_style(para_name)
fontstyle = para_style.get_font()
y_start = self.top_margin - cm2u(y)
x_start = self.left_margin + cm2u(x)
size = fontstyle.get_size()
self.gpc.gsave()
self.gpc.translate(x_start,y_start)
self.gpc.rotate(-angle)
this_y = 0
for line in text:
if not line.split():
continue
width = get_text_width(line,fontstyle)
this_x = -0.5 * width
self.gpc.setfont(find_font_from_fontstyle(fontstyle))
self.gpc.moveto(this_x,this_y)
self.gpc.show(line)
this_y -= size * _EXTRA_SPACING_FACTOR
self.gpc.grestore()
def draw_line(self,style,x1,y1,x2,y2):
self.brand_new_page = 0
x1 = cm2u(x1) + self.left_margin
x2 = cm2u(x2) + self.left_margin
y1 = self.top_margin - cm2u(y1)
y2 = self.top_margin - cm2u(y2)
self.gpc.line_stroked(x1,y1,x2,y2)
#------------------------------------------------------------------------
#
# Print job methods
#
#------------------------------------------------------------------------
#function to print text to a printer
def do_print(self,dialog,job):
self.gpc = gnomeprint.Context(dialog.get_config())
job.render(self.gpc)
self.gpc.close()
#I believe this is a print preview
def show_preview(self,dialog):
w = gnomeprint.ui.JobPreview(self.job, _("Print Preview"))
w.set_property('allow-grow', 1)
w.set_property('allow-shrink', 1)
w.set_transient_for(dialog)
w.show_all()
#function used to get users response and do a certain
#action depending on that response
def print_dialog_response(self, dialog, resp, job):
if resp == gnomeprint.ui.DIALOG_RESPONSE_PREVIEW:
self.show_preview(dialog)
elif resp == gnomeprint.ui.DIALOG_RESPONSE_CANCEL:
dialog.destroy()
elif resp == gnomeprint.ui.DIALOG_RESPONSE_PRINT:
self.do_print(dialog, self.job)
dialog.destroy()
#function displays a window that allows user to choose
#to print, show, etc
def show_print_dialog(self):
dialog = gnomeprint.ui.Dialog(self.job, _("Print..."),
gnomeprint.ui.DIALOG_RANGE|gnomeprint.ui.DIALOG_COPIES)
dialog.construct_range_page(
gnomeprint.ui.RANGE_ALL|gnomeprint.ui.RANGE_RANGE,
1, self.page_count, "A", "Pages: ")
dialog.connect('response', self.print_dialog_response, self.job)
dialog.show()
#------------------------------------------------------------------------
#
# Register the document generator with the system
#
#------------------------------------------------------------------------
register_text_doc(
name=_("Print..."),
classref=LPRDoc,
table=1,
paper=1,
style=1,
ext="",
print_report_label=None,
clname='print')
register_book_doc(
_("Print..."),
LPRDoc,
1,
1,
1,
"",
'print')
register_draw_doc(
_("Print..."),
LPRDoc,
1,
1,
"",
None,
'print')