# # Gramps - a GTK+/GNOME based genealogy program # # Copyright (C) 2000-2006 Donald N. Allingham # Copyright (C) 2005-2006 Serge Noiraud # # 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 # # $Id$ #------------------------------------------------------------------------- # # Standard Python Modules # #------------------------------------------------------------------------- import os import zipfile import time import locale from cStringIO import StringIO from math import pi, cos, sin #------------------------------------------------------------------------- # # Gramps modules # #------------------------------------------------------------------------- import BaseDoc import const from PluginUtils import register_text_doc, register_draw_doc, register_book_doc from ReportBase import ReportUtils import ImgManip import FontScale import Mime import Utils import Errors #------------------------------------------------------------------------- # # internationalization # #------------------------------------------------------------------------- from gettext import gettext as _ from xml.sax.saxutils import escape _apptype = 'application/vnd.oasis.opendocument.text' _esc_map = { '\x1a' : '', '\x0c' : '', '\n' : '', '<super>' : '', '</super>' : '', } #------------------------------------------------------------------------- # # ODFDoc # #------------------------------------------------------------------------- class ODFDoc(BaseDoc.BaseDoc): def __init__(self,styles,type,template,orientation=BaseDoc.PAPER_PORTRAIT): BaseDoc.BaseDoc.__init__(self,styles,type,template,orientation) self.cntnt = None self.filename = None self.level = 0 self.time = "0000-00-00T00:00:00" self.new_page = 0 self.new_cell = 0 self.page = 0 self.first_page = 1 def set_mode(self, mode): self.mode = mode def open(self,filename): t = time.localtime(time.time()) self.time = "%04d-%02d-%02dT%02d:%02d:%02d" % \ (t[0],t[1],t[2],t[3],t[4],t[5]) if filename[-4:] != ".odt": self.filename = filename + ".odt" else: self.filename = filename self.filename = os.path.normpath(os.path.abspath(self.filename)) self.cntnt = StringIO() def init(self): assert (not self.init_called) self.init_called = True current_locale = locale.getlocale() self.lang = current_locale[0] if self.lang: self.lang = self.lang.replace('_','-') else: self.lang = "en-US" self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('') self.cntnt.write('') self.cntnt.write('\n') self.cntnt.write('') self.cntnt.write('') self.cntnt.write('\n') for style_name in self.draw_styles.keys(): style = self.draw_styles[style_name] self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') for style_name in self.style_list.keys(): style = self.style_list[style_name] self.cntnt.write('\n' % style_name) self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write(' 0: self.cntnt.write('fo:keep-with-next="true" ') align = style.get_alignment() if align == BaseDoc.PARA_ALIGN_LEFT: self.cntnt.write('fo:text-align="start" ') elif align == BaseDoc.PARA_ALIGN_RIGHT: self.cntnt.write('fo:text-align="end" ') elif align == BaseDoc.PARA_ALIGN_CENTER: self.cntnt.write('fo:text-align="center" ') self.cntnt.write('style:justify-single-word="false" ') else: self.cntnt.write('fo:text-align="justify" ') self.cntnt.write('style:justify-single-word="false" ') font = style.get_font() if font.get_type_face() == BaseDoc.FONT_SANS_SERIF: self.cntnt.write('style:font-name="Arial" ') else: self.cntnt.write('style:font-name="Times New Roman" ') self.cntnt.write('fo:font-size="%.2fpt" ' % font.get_size()) self.cntnt.write('style:font-size-asian="%.2fpt" ' % font.get_size()) color = font.get_color() self.cntnt.write('fo:color="#%02x%02x%02x" ' % color) if font.get_bold(): self.cntnt.write('fo:font-weight="bold" ') if font.get_italic(): self.cntnt.write('fo:font-style="italic" ') if font.get_underline(): self.cntnt.write('style:text-underline="single" ') self.cntnt.write('style:text-underline-color="font-color" ') self.cntnt.write('fo:text-indent="%.2fcm"\n' % style.get_first_indent()) self.cntnt.write('fo:margin-right="%.2fcm"\n' % style.get_right_margin()) self.cntnt.write('fo:margin-left="%.2fcm"\n' % style.get_left_margin()) self.cntnt.write('fo:margin-top="0.00cm"\n') self.cntnt.write('fo:margin-bottom="0.212cm" ') self.cntnt.write('/>\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write(' ' % font.get_size()) self.cntnt.write('\n') for style_name in self.table_styles.keys(): style = self.table_styles[style_name] self.cntnt.write('\n') table_width = float(self.get_usable_width()) table_width_str = "%.2f" % table_width self.cntnt.write('\n') self.cntnt.write('\n') for col in range(0,style.get_columns()): self.cntnt.write('') width = table_width * float(style.get_column_width(col)/100.0) width_str = "%.4f" % width self.cntnt.write('' % width_str) self.cntnt.write('\n') for cell in self.cell_styles.keys(): cell_style = self.cell_styles[cell] self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') #Begin photo style self.cntnt.write('') self.cntnt.write('') self.cntnt.write('\n') self.cntnt.write('') self.cntnt.write('') self.cntnt.write('\n') self.cntnt.write(' ') self.cntnt.write(' ') self.cntnt.write('\n') self.cntnt.write('') self.cntnt.write('') self.cntnt.write('\n') #end of Photo style edits self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write(' \n') self.cntnt.write(' \n') def close(self): self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('\n') self._write_styles_file() self._write_manifest() self._write_meta_file() self._write_mimetype_file() self._write_zip() if self.print_req: app = Mime.get_application(_apptype) Utils.launch(app[0],self.filename) def add_media_object(self,name,pos,x_cm,y_cm): # try to open the image. If the open fails, it probably wasn't # a valid image (could be a PDF, or a non-image) try: image = ImgManip.ImgManip(name) (x,y) = image.size() ratio = float(x_cm)*float(y)/(float(y_cm)*float(x)) except: return if ratio < 1: act_width = x_cm act_height = y_cm*ratio else: act_height = y_cm act_width = x_cm/ratio media_list_item = (name,act_width,act_height) if not media_list_item in self.media_list: self.media_list.append(media_list_item) base = os.path.basename(name) tag = base.replace('.','_') if self.new_cell: self.cntnt.write('') if pos == "left": self.cntnt.write('') self.cntnt.write('\n') self.cntnt.write('\n') if self.new_cell: self.cntnt.write('\n') def start_table(self,name,style_name): self.cntnt.write('\n' % style_name) table = self.table_styles[style_name] for col in range(0,table.get_columns()): self.cntnt.write('\n') def end_table(self): self.cntnt.write('\n') def start_row(self): self.cntnt.write('\n') def end_row(self): self.cntnt.write('\n') def start_cell(self,style_name,span=1): self.span = span self.cntnt.write(' 1: self.cntnt.write(' table:number-columns-spanned="%s">\n' % span) else: self.cntnt.write('>\n') self.new_cell = 1 def end_cell(self): self.cntnt.write('\n') #for col in range(1,self.span): # self.cntnt.write('\n') self.new_cell = 0 def start_bold(self): self.cntnt.write('') def end_bold(self): self.cntnt.write('') def _add_zip(self,zfile,name,data,t): zipinfo = zipfile.ZipInfo(name.encode('latin-1')) zipinfo.date_time = t zipinfo.compress_type = zipfile.ZIP_DEFLATED zfile.writestr(zipinfo,data) def _write_zip(self): try: zfile = zipfile.ZipFile(self.filename,"w",zipfile.ZIP_DEFLATED) except IOError,msg: errmsg = "%s\n%s" % (_("Could not create %s") % self.filename, msg) raise Errors.ReportError(errmsg) except: raise Errors.ReportError(_("Could not create %s") % self.filename) t = time.localtime(time.time())[:6] self._add_zip(zfile,"META-INF/manifest.xml",self.mfile.getvalue(),t) self._add_zip(zfile,"content.xml",self.cntnt.getvalue(),t) self._add_zip(zfile,"meta.xml",self.meta.getvalue(),t) self._add_zip(zfile,"styles.xml",self.sfile.getvalue(),t) self._add_zip(zfile,"mimetype",self.mimetype.getvalue(),t) self.mfile.close() self.cntnt.close() self.meta.close() self.sfile.close() self.mimetype.close() for image in self.media_list: try: ifile = open(image[0],mode='rb') base = os.path.basename(image[0]) self._add_zip(zfile,"Pictures/%s" % base, ifile.read(),t) ifile.close() except: print "Could not open %s" % image[0] zfile.close() def _write_styles_file(self): self.sfile = StringIO() self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('') self.sfile.write('') self.sfile.write(' ') self.sfile.write(' ') self.sfile.write('') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write(' \n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write(' ') self.sfile.write(' ') self.sfile.write('\n') self.sfile.write(' ') self.sfile.write(' ') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') for key in self.style_list.keys(): style = self.style_list[key] self.sfile.write('\n') self.sfile.write(' 0: self.sfile.write('fo:keep-with-next="always" ') align = style.get_alignment() if align == BaseDoc.PARA_ALIGN_LEFT: self.sfile.write('fo:text-align="start" ') self.sfile.write('style:justify-single-word="false" ') elif align == BaseDoc.PARA_ALIGN_RIGHT: self.sfile.write('fo:text-align="end" ') elif align == BaseDoc.PARA_ALIGN_CENTER: self.sfile.write('fo:text-align="center" ') self.sfile.write('style:justify-single-word="false" ') else: self.sfile.write('fo:text-align="justify" ') self.sfile.write('style:justify-single-word="false" ') self.sfile.write('fo:text-indent="%.2fcm" ' % style.get_first_indent()) self.sfile.write('style:auto-text-indent="false"/> ') self.sfile.write('\n') self.sfile.write('\n') # Current no leading number format for headers #self.sfile.write('\n') #self.sfile.write('\n') #self.sfile.write('\n') #self.sfile.write('\n') #self.sfile.write('\n') #self.sfile.write('\n') #self.sfile.write('\n') #self.sfile.write('\n') #self.sfile.write('\n') #self.sfile.write('\n') #self.sfile.write('\n') #self.sfile.write('\n') self.sfile.write(' ') self.sfile.write(' ') self.sfile.write(' ') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('') self.sfile.write('') self.sfile.write('\n') self.sfile.write('') self.sfile.write('') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') # header self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') # footer self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') # self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') self.sfile.write('\n') # header #self.sfile.write('') #self.sfile.write('') #self.sfile.write(' TITRE : %s' % self.title) # How to get the document title here ? #self.sfile.write('') #self.sfile.write('') # footer #self.sfile.write('') #self.sfile.write('') #self.sfile.write('1') #self.sfile.write('/') #self.sfile.write('1') #self.sfile.write('') #self.sfile.write('') #self.sfile.write('') # self.sfile.write('') self.sfile.write('\n') self.sfile.write('\n') def page_break(self): self.new_page = 1 def start_page(self): self.cntnt.write('\n') def end_page(self): self.cntnt.write('\n') def start_paragraph(self,style_name,leader=None): style = self.style_list[style_name] self.level = style.get_header_level() if self.new_page == 1: self.new_page = 0 name = "NL%s" % style_name else: name = style_name if self.level == 0: self.cntnt.write('' % name) else: self.cntnt.write('') if leader != None: self.cntnt.write(leader) self.cntnt.write('') self.new_cell = 0 def end_paragraph(self): if self.level == 0: self.cntnt.write('\n') else: self.cntnt.write('\n') self.new_cell = 1 def write_note(self,text,format,style_name): if format == 1: text = text.replace('&','&') # Must be first text = text.replace('<','<') text = text.replace('>','>') # Replace multiple spaces: have to go from the largest number down for n in range(text.count(' '),1,-1): text = text.replace(' '*n, ' ' % (n-1) ) text = text.replace('\n','') text = text.replace('\t','') text = text.replace('<super>', '') text = text.replace('</super>','') self.start_paragraph(style_name) self.cntnt.write('') self.cntnt.write(text) self.cntnt.write('') 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() def write_text(self,text,mark=None): """ Uses the xml.sax.saxutils.escape function to convert XML entities. The _esc_map dictionary allows us to add our own mappings. """ if mark: key = escape(mark.key,_esc_map) key = key.replace('"','"') if mark.type == BaseDoc.INDEX_TYPE_ALP: self.cntnt.write('' % key) elif mark.type == BaseDoc.INDEX_TYPE_TOC: self.cntnt.write('' % mark.level) self.cntnt.write(escape(text,_esc_map)) def _write_manifest(self): self.mfile = StringIO() self.mfile.write('\n') self.mfile.write('') self.mfile.write('') for image in self.media_list: i = image[0] base = os.path.basename(i) self.mfile.write('') self.mfile.write('') self.mfile.write('') self.mfile.write('') self.mfile.write('') self.mfile.write('\n') def _write_mimetype_file(self): self.mimetype = StringIO() self.mimetype.write('application/vnd.oasis.opendocument.text') def _write_meta_file(self): self.meta = StringIO() self.meta.write('\n') self.meta.write('\n'); self.meta.write('\n') self.meta.write('') self.meta.write(const.program_name + ' ' + const.version) self.meta.write('\n') self.meta.write('') # It should be reasonable to have a true document title. but how ? # self.title ? #self.meta.write(_("Summary of %s") % self.name) self.meta.write('\n') self.meta.write('') #self.meta.write(_("Summary of %s") % name) self.meta.write('\n') self.meta.write('') self.meta.write('\n') self.meta.write('') self.meta.write(self.name) self.meta.write('\n') self.meta.write('') self.meta.write(self.time) self.meta.write('\n') self.meta.write('') self.meta.write(self.name) self.meta.write('\n') self.meta.write('') self.meta.write(self.time) self.meta.write('\n') self.meta.write('') self.meta.write('\n') self.meta.write('0-00-00T00:00:00\n') self.meta.write('%s\n' % self.lang) self.meta.write('1\n') self.meta.write('PT0S\n') self.meta.write('http://gramps-project.org') self.meta.write('\n') self.meta.write('\n') self.meta.write('\n') self.meta.write('\n') self.meta.write('\n') self.meta.write('\n') def rotate_text(self,style,text,x,y,angle): stype = self.draw_styles[style] pname = stype.get_paragraph_style() p = self.style_list[pname] font = p.get_font() size = font.get_size() height = size*(len(text)) oneline = (size/72.0)*2.54 width = 0 for line in text: width = max(width,FontScale.string_width(font,line)) wcm = (width/72.0)*2.54*1.2 hcm = (height/72.0)*2.54*1.2 rangle = -((pi/180.0) * angle) self.cntnt.write(' ') self.cntnt.write('') self.cntnt.write(' ' % pname) self.cntnt.write('\n' % pname) self.write_text('\n'.join(text)) self.cntnt.write('\n\n\n') self.cntnt.write('\n') def draw_path(self,style,path): minx = 9e12 miny = 9e12 maxx = 0 maxy = 0 for point in path: minx = min(point[0],minx) miny = min(point[1],miny) maxx = max(point[0],maxx) maxy = max(point[1],maxy) self.cntnt.write('\n') def draw_line(self,style,x1,y1,x2,y2): self.cntnt.write('' % y2) self.cntnt.write('\n') self.cntnt.write('\n') def draw_text(self,style,text,x,y): box_style = self.draw_styles[style] para_name = box_style.get_paragraph_style() pstyle = self.style_list[para_name] font = pstyle.get_font() if box_style.get_width(): sw = box_style.get_width()*1.2 else: sw = ReportUtils.pt2cm(FontScale.string_width(font,text))*1.3 self.cntnt.write('' % float(y)) self.cntnt.write(' ' ) self.cntnt.write('' % para_name) self.cntnt.write('' % font.get_size() ) self.cntnt.write(text) self.cntnt.write('') self.cntnt.write('\n') self.cntnt.write('\n') def draw_bar(self,style,x,y,x2,y2): box_style = self.draw_styles[style] self.cntnt.write('' % float(y)) self.cntnt.write('\n') def draw_box(self,style,text,x,y): box_style = self.draw_styles[style] para_name = box_style.get_paragraph_style() shadow_width = box_style.get_shadow_space() if box_style.get_shadow(): self.cntnt.write('\n' % (float(y)+shadow_width)) self.cntnt.write('\n') self.cntnt.write('\n' % float(y)) if text != "": self.cntnt.write('' % para_name) self.cntnt.write('' % para_name) text = text.replace('\t','') text = text.replace('\n','') self.cntnt.write(text) self.cntnt.write('') self.cntnt.write('\n') self.cntnt.write('\n') def center_text(self,style,text,x,y): box_style = self.draw_styles[style] para_name = box_style.get_paragraph_style() pstyle = self.style_list[para_name] font = pstyle.get_font() size = 1.2*(FontScale.string_width(font,text)/72.0) * 2.54 self.cntnt.write('\n' % float(y)) if text != "": self.cntnt.write('') self.cntnt.write('' % para_name) self.cntnt.write('' % para_name) self.cntnt.write(text) self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('') self.cntnt.write('\n') def write_at(self,style,text,x,y): pstyle = self.style_list[style] font = pstyle.get_font() size = 1.2*(FontScale.string_width(font,text)/72.0) * 2.54 self.cntnt.write('\n' % float(y)) if text != "": self.cntnt.write('') self.cntnt.write('' % style) self.cntnt.write('' % style) self.cntnt.write(text) self.cntnt.write('\n') self.cntnt.write('\n') self.cntnt.write('') self.cntnt.write('\n') #-------------------------------------------------------------------------- # # Register plugins # #-------------------------------------------------------------------------- print_label = None try: mprog = Mime.get_application(_apptype) if Utils.search_for(mprog[0]): print_label = _("Open in %(program_name)s") % { 'program_name': mprog[1]} else: print_label = None except: print_label = None register_text_doc(_('Open Document Text'), ODFDoc, 1, 1, 1, ".odt", print_label) register_book_doc(_("Open Document Text"), ODFDoc, 1, 1, 1, ".odt") register_draw_doc(_("Open Document Text"), ODFDoc, 1, 1, ".odt", print_label);