source: pdfshuffler/trunk/fuentes/.pc/use-pypdf2/pdfshuffler/pdfshuffler.py @ 337

Last change on this file since 337 was 337, checked in by jrpelegrina, 4 years ago

Firs release to xenial

File size: 43.7 KB
Line 
1#!/usr/bin/python
2# -*- coding: utf-8 -*-
3
4"""
5
6 PdfShuffler 0.6.0 - GTK+ based utility for splitting, rearrangement and
7 modification of PDF documents.
8 Copyright (C) 2008-2012 Konstantinos Poulios
9 <https://sourceforge.net/projects/pdfshuffler>
10
11 This file is part of PdfShuffler.
12
13 PdfShuffler is free software; you can redistribute it and/or modify
14 it under the terms of the GNU General Public License as published by
15 the Free Software Foundation; either version 3 of the License, or
16 (at your option) any later version.
17
18 This program is distributed in the hope that it will be useful,
19 but WITHOUT ANY WARRANTY; without even the implied warranty of
20 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
21 GNU General Public License for more details.
22
23 You should have received a copy of the GNU General Public License along
24 with this program; if not, write to the Free Software Foundation, Inc.,
25 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
26
27"""
28
29import os
30import shutil       # for file operations like whole directory deletion
31import sys          # for proccessing of command line args
32import urllib       # for parsing filename information passed by DnD
33import threading
34import tempfile
35from copy import copy
36
37import locale       #for multilanguage support
38import gettext
39gettext.install('pdfshuffler', unicode=1)
40
41
42APPNAME = 'PdfShuffler' # PDF-Shuffler, PDFShuffler, pdfshuffler
43VERSION = '0.6.0'
44WEBSITE = 'http://pdfshuffler.sourceforge.net/'
45LICENSE = 'GNU General Public License (GPL) Version 3.'
46
47try:
48    import pygtk
49    pygtk.require('2.0')
50    import gtk
51    assert gtk.gtk_version >= (2, 10, 0)
52    assert gtk.pygtk_version >= (2, 10, 0)
53except AssertionError:
54    print('You do not have the required versions of GTK+ and PyGTK ' +
55          'installed.\n\n' +
56          'Installed GTK+ version is ' +
57          '.'.join([str(n) for n in gtk.gtk_version]) + '\n' +
58          'Required GTK+ version is 2.10.0 or higher\n\n'
59          'Installed PyGTK version is ' +
60          '.'.join([str(n) for n in gtk.pygtk_version]) + '\n' +
61          'Required PyGTK version is 2.10.0 or higher')
62    sys.exit(1)
63except:
64    print('PyGTK version 2.10.0 or higher is required to run this program.')
65    print('No version of PyGTK was found on your system.')
66    sys.exit(1)
67
68import gobject      # for using custom signals
69import pango        # for adjusting the text alignment in CellRendererText
70import gio          # for inquiring mime types information
71import cairo
72
73import poppler      #for the rendering of pdf pages
74from pyPdf import PdfFileWriter, PdfFileReader
75
76from pdfshuffler_iconview import CellRendererImage
77gobject.type_register(CellRendererImage)
78
79import time
80
81class PdfShuffler:
82    prefs = {
83        'window width': min(700, gtk.gdk.screen_get_default().get_width() / 2),
84        'window height': min(600, gtk.gdk.screen_get_default().get_height() - 50),
85        'window x': 0,
86        'window y': 0,
87        'initial thumbnail size': 300,
88        'initial zoom level': -14,
89    }
90
91    MODEL_ROW_INTERN = 1001
92    MODEL_ROW_EXTERN = 1002
93    TEXT_URI_LIST = 1003
94    MODEL_ROW_MOTION = 1004
95    TARGETS_IV = [('MODEL_ROW_INTERN', gtk.TARGET_SAME_WIDGET, MODEL_ROW_INTERN),
96                  ('MODEL_ROW_EXTERN', gtk.TARGET_OTHER_APP, MODEL_ROW_EXTERN),
97                  ('MODEL_ROW_MOTION', 0, MODEL_ROW_MOTION)]
98    TARGETS_SW = [('text/uri-list', 0, TEXT_URI_LIST),
99                  ('MODEL_ROW_EXTERN', gtk.TARGET_OTHER_APP, MODEL_ROW_EXTERN)]
100
101    def __init__(self):
102        # Create the temporary directory
103        self.tmp_dir = tempfile.mkdtemp("pdfshuffler")
104        os.chmod(self.tmp_dir, 0700)
105
106        icon_theme = gtk.icon_theme_get_default()
107        try:
108            gtk.window_set_default_icon(icon_theme.load_icon("pdfshuffler", 64, 0))
109        except:
110            print(_("Can't load icon. Application is not installed correctly."))
111
112        # Import the user interface file, trying different possible locations
113        ui_path = '/usr/share/pdfshuffler/pdfshuffler.ui'
114        if not os.path.exists(ui_path):
115            ui_path = '/usr/local/share/pdfshuffler/pdfshuffler.ui'
116
117        if not os.path.exists(ui_path):
118            parent_dir = os.path.dirname( \
119                         os.path.dirname(os.path.realpath(__file__)))
120            ui_path = os.path.join(parent_dir, 'data', 'pdfshuffler.ui')
121
122        if not os.path.exists(ui_path):
123            head, tail = os.path.split(parent_dir)
124            while tail != 'lib' and tail != '':
125                head, tail = os.path.split(head)
126            if tail == 'lib':
127                ui_path = os.path.join(head, 'share', 'pdfshuffler', \
128                                       'pdfshuffler.ui')
129
130        self.uiXML = gtk.Builder()
131        self.uiXML.add_from_file(ui_path)
132        self.uiXML.connect_signals(self)
133
134        # Create the main window, and attach delete_event signal to terminating
135        # the application
136        self.window = self.uiXML.get_object('main_window')
137        self.window.set_title(APPNAME)
138        self.window.set_border_width(0)
139        self.window.move(self.prefs['window x'], self.prefs['window y'])
140        self.window.set_default_size(self.prefs['window width'],
141                                     self.prefs['window height'])
142        self.window.connect('delete_event', self.close_application)
143
144        # Create a scrolled window to hold the thumbnails-container
145        self.sw = self.uiXML.get_object('scrolledwindow')
146        self.sw.drag_dest_set(gtk.DEST_DEFAULT_MOTION |
147                              gtk.DEST_DEFAULT_HIGHLIGHT |
148                              gtk.DEST_DEFAULT_DROP |
149                              gtk.DEST_DEFAULT_MOTION,
150                              self.TARGETS_SW,
151                              gtk.gdk.ACTION_COPY |
152                              gtk.gdk.ACTION_MOVE)
153        self.sw.connect('drag_data_received', self.sw_dnd_received_data)
154        self.sw.connect('button_press_event', self.sw_button_press_event)
155        self.sw.connect('scroll_event', self.sw_scroll_event)
156
157        # Create an alignment to keep the thumbnails center-aligned
158        align = gtk.Alignment(0.5, 0.5, 0, 0)
159        self.sw.add_with_viewport(align)
160
161        # Create ListStore model and IconView
162        self.model = gtk.ListStore(str,         # 0.Text descriptor
163                                   gobject.TYPE_PYOBJECT,
164                                                # 1.Cached page image
165                                   int,         # 2.Document number
166                                   int,         # 3.Page number
167                                   float,       # 4.Scale
168                                   str,         # 5.Document filename
169                                   int,         # 6.Rotation angle
170                                   float,       # 7.Crop left
171                                   float,       # 8.Crop right
172                                   float,       # 9.Crop top
173                                   float,       # 10.Crop bottom
174                                   int,         # 11.Page width
175                                   int,         # 12.Page height
176                                   float)       # 13.Resampling factor
177
178        self.zoom_set(self.prefs['initial zoom level'])
179        self.iv_col_width = self.prefs['initial thumbnail size']
180
181        self.iconview = gtk.IconView(self.model)
182        self.iconview.set_item_width(self.iv_col_width + 12)
183
184        self.cellthmb = CellRendererImage()
185        self.iconview.pack_start(self.cellthmb, False)
186        self.iconview.set_attributes(self.cellthmb, image=1,
187            scale=4, rotation=6, cropL=7, cropR=8, cropT=9, cropB=10,
188            width=11, height=12, resample=13)
189
190        self.celltxt = gtk.CellRendererText()
191        self.celltxt.set_property('width', self.iv_col_width)
192        self.celltxt.set_property('wrap-width', self.iv_col_width)
193        self.celltxt.set_property('alignment', pango.ALIGN_CENTER)
194        self.iconview.pack_start(self.celltxt, False)
195        self.iconview.set_attributes(self.celltxt, text=0)
196
197        self.iconview.set_selection_mode(gtk.SELECTION_MULTIPLE)
198        self.iconview.enable_model_drag_source(gtk.gdk.BUTTON1_MASK,
199                                               self.TARGETS_IV,
200                                               gtk.gdk.ACTION_COPY |
201                                               gtk.gdk.ACTION_MOVE)
202        self.iconview.enable_model_drag_dest(self.TARGETS_IV,
203                                             gtk.gdk.ACTION_DEFAULT)
204        self.iconview.connect('drag_begin', self.iv_drag_begin)
205        self.iconview.connect('drag_data_get', self.iv_dnd_get_data)
206        self.iconview.connect('drag_data_received', self.iv_dnd_received_data)
207        self.iconview.connect('drag_data_delete', self.iv_dnd_data_delete)
208        self.iconview.connect('drag_motion', self.iv_dnd_motion)
209        self.iconview.connect('drag_leave', self.iv_dnd_leave_end)
210        self.iconview.connect('drag_end', self.iv_dnd_leave_end)
211        self.iconview.connect('button_press_event', self.iv_button_press_event)
212
213        align.add(self.iconview)
214
215        # Progress bar
216        self.progress_bar = self.uiXML.get_object('progressbar')
217        self.progress_bar_timeout_id = 0
218
219        # Define window callback function and show window
220        self.window.connect('size_allocate', self.on_window_size_request)        # resize
221        self.window.connect('key_press_event', self.on_keypress_event ) # keypress
222        self.window.show_all()
223        self.progress_bar.hide_all()
224
225        # Change iconview color background
226        style = self.sw.get_style().copy()
227        for state in (gtk.STATE_NORMAL, gtk.STATE_PRELIGHT, gtk.STATE_ACTIVE):
228            style.base[state] = style.bg[gtk.STATE_NORMAL]
229        self.iconview.set_style(style)
230
231        # Creating the popup menu
232        self.popup = gtk.Menu()
233        popup_rotate_right = gtk.ImageMenuItem(_('_Rotate Right'))
234        popup_rotate_left = gtk.ImageMenuItem(_('Rotate _Left'))
235        popup_crop = gtk.MenuItem(_('C_rop...'))
236        popup_delete = gtk.ImageMenuItem(gtk.STOCK_DELETE)
237        popup_saveselection = gtk.MenuItem(_('_Export selection...'))
238        popup_rotate_right.connect('activate', self.rotate_page_right)
239        popup_rotate_left.connect('activate', self.rotate_page_left)
240        popup_crop.connect('activate', self.crop_page_dialog)
241        popup_delete.connect('activate', self.clear_selected)
242        popup_saveselection.connect('activate', self.choose_export_pdf_name, True)
243        popup_rotate_right.show()
244        popup_rotate_left.show()
245        popup_crop.show()
246        popup_delete.show()
247        popup_saveselection.show()
248        self.popup.append(popup_rotate_right)
249        self.popup.append(popup_rotate_left)
250        self.popup.append(popup_crop)
251        self.popup.append(popup_delete)
252        self.popup.append(popup_saveselection)
253
254        # Initializing variables
255        self.export_directory = os.getenv('HOME')
256        self.import_directory = self.export_directory
257        self.nfile = 0
258        self.iv_auto_scroll_direction = 0
259        self.iv_auto_scroll_timer = None
260        self.pdfqueue = []
261
262        gobject.type_register(PDF_Renderer)
263        gobject.signal_new('update_thumbnail', PDF_Renderer,
264                           gobject.SIGNAL_RUN_FIRST, gobject.TYPE_NONE,
265                           [gobject.TYPE_INT, gobject.TYPE_PYOBJECT,
266                            gobject.TYPE_FLOAT])
267        self.rendering_thread = 0
268
269        self.set_unsaved(False)
270
271        # Importing documents passed as command line arguments
272        for filename in sys.argv[1:]:
273            self.add_pdf_pages(filename)
274
275    def render(self):
276        if self.rendering_thread:
277            self.rendering_thread.quit = True
278            self.rendering_thread.join()
279        #FIXME: the resample=2. factor has to be dynamic when lazy rendering
280        #       is implemented
281        self.rendering_thread = PDF_Renderer(self.model, self.pdfqueue, 2)
282        self.rendering_thread.connect('update_thumbnail', self.update_thumbnail)
283        self.rendering_thread.start()
284
285        if self.progress_bar_timeout_id:
286            gobject.source_remove(self.progress_bar_timeout_id)
287        self.progress_bar_timout_id = \
288            gobject.timeout_add(50, self.progress_bar_timeout)
289
290    def set_unsaved(self, flag):
291        self.is_unsaved = flag
292        gobject.idle_add(self.retitle)
293
294    def retitle(self):
295        title = ''
296        if len(self.pdfqueue) == 1:
297            title += self.pdfqueue[0].filename
298        elif len(self.pdfqueue) == 0:
299            title += _("No document")
300        else:
301            title += _("Several documents")
302        if self.is_unsaved:
303            title += '*'
304        title += ' - ' + APPNAME
305        self.window.set_title(title)
306
307    def progress_bar_timeout(self):
308        cnt_finished = 0
309        cnt_all = 0
310        for row in self.model:
311            cnt_all += 1
312            if row[1]:
313                cnt_finished += 1
314        fraction = float(cnt_finished)/float(cnt_all)
315
316        self.progress_bar.set_fraction(fraction)
317        self.progress_bar.set_text(_('Rendering thumbnails... [%(i1)s/%(i2)s]')
318                                   % {'i1' : cnt_finished, 'i2' : cnt_all})
319        if fraction >= 0.999:
320            self.progress_bar.hide_all()
321            return False
322        elif not self.progress_bar.flags() & gtk.VISIBLE:
323            self.progress_bar.show_all()
324
325        return True
326 
327    def update_thumbnail(self, object, num, thumbnail, resample):
328        row = self.model[num]
329        row[13] = resample
330        row[4] = self.zoom_scale
331        row[1] = thumbnail
332
333    def on_window_size_request(self, window, event):
334        """Main Window resize - workaround for autosetting of
335           iconview cols no."""
336
337        #add 12 because of: http://bugzilla.gnome.org/show_bug.cgi?id=570152
338        col_num = 9 * window.get_size()[0] \
339            / (10 * (self.iv_col_width + self.iconview.get_column_spacing() * 2))
340        self.iconview.set_columns(col_num)
341
342    def update_geometry(self, iter):
343        """Recomputes the width and height of the rotated page and saves
344           the result in the ListStore"""
345
346        if not self.model.iter_is_valid(iter):
347            return
348
349        nfile, npage, rotation = self.model.get(iter, 2, 3, 6)
350        crop = self.model.get(iter, 7, 8, 9, 10)
351        page = self.pdfqueue[nfile-1].document.get_page(npage-1)
352        w0, h0 = page.get_size()
353
354        rotation = int(rotation) % 360
355        rotation = ((rotation + 45) / 90) * 90
356        if rotation == 90 or rotation == 270:
357            w1, h1 = h0, w0
358        else:
359            w1, h1 = w0, h0
360
361        self.model.set(iter, 11, w1, 12, h1)
362
363    def reset_iv_width(self, renderer=None):
364        """Reconfigures the width of the iconview columns"""
365
366        if not self.model.get_iter_first(): #just checking if model is empty
367            return
368
369        max_w = 10 + int(max(row[4]*row[11]*(1.-row[7]-row[8]) \
370                             for row in self.model))
371        if max_w != self.iv_col_width:
372            self.iv_col_width = max_w
373            self.celltxt.set_property('width', self.iv_col_width)
374            self.celltxt.set_property('wrap-width', self.iv_col_width)
375            self.iconview.set_item_width(self.iv_col_width + 12) #-1)
376            self.on_window_size_request(self.window, None)
377
378    def on_keypress_event(self, widget, event):
379        """Keypress events in Main Window"""
380
381        #keyname = gtk.gdk.keyval_name(event.keyval)
382        if event.keyval == 65535:   # Delete keystroke
383            self.clear_selected()
384
385    def close_application(self, widget, event=None, data=None):
386        """Termination"""
387
388        if self.rendering_thread:
389            self.rendering_thread.quit = True
390            self.rendering_thread.join()
391
392        if os.path.isdir(self.tmp_dir):
393            shutil.rmtree(self.tmp_dir)
394        if gtk.main_level():
395            gtk.main_quit()
396        else:
397            sys.exit(0)
398        return False
399
400    def add_pdf_pages(self, filename,
401                            firstpage=None, lastpage=None,
402                            angle=0, crop=[0.,0.,0.,0.]):
403        """Add pages of a pdf document to the model"""
404
405        res = False
406        # Check if the document has already been loaded
407        pdfdoc = None
408        for it_pdfdoc in self.pdfqueue:
409            if os.path.isfile(it_pdfdoc.filename) and \
410               os.path.samefile(filename, it_pdfdoc.filename) and \
411               os.path.getmtime(filename) is it_pdfdoc.mtime:
412                pdfdoc = it_pdfdoc
413                break
414
415        if not pdfdoc:
416            pdfdoc = PDF_Doc(filename, self.nfile, self.tmp_dir)
417            self.import_directory = os.path.split(filename)[0]
418            self.export_directory = self.import_directory
419            if pdfdoc.nfile != 0 and pdfdoc != []:
420                self.nfile = pdfdoc.nfile
421                self.pdfqueue.append(pdfdoc)
422            else:
423                return res
424
425        n_start = 1
426        n_end = pdfdoc.npage
427        if firstpage:
428           n_start = min(n_end, max(1, firstpage))
429        if lastpage:
430           n_end = max(n_start, min(n_end, lastpage))
431
432        for npage in range(n_start, n_end + 1):
433            descriptor = ''.join([pdfdoc.shortname, '\n', _('page'), ' ', str(npage)])
434            page = pdfdoc.document.get_page(npage-1)
435            w, h = page.get_size()
436            iter = self.model.append((descriptor,         # 0
437                                      None,               # 1
438                                      pdfdoc.nfile,       # 2
439                                      npage,              # 3
440                                      self.zoom_scale,    # 4
441                                      pdfdoc.filename,    # 5
442                                      angle,              # 6
443                                      crop[0],crop[1],    # 7-8
444                                      crop[2],crop[3],    # 9-10
445                                      w,h,                # 11-12
446                                      2.              ))  # 13 FIXME
447            self.update_geometry(iter)
448            res = True
449
450        self.reset_iv_width()
451        gobject.idle_add(self.retitle)
452        if res:
453            gobject.idle_add(self.render)
454        return res
455
456    def choose_export_pdf_name(self, widget=None, only_selected=False):
457        """Handles choosing a name for exporting """
458
459        chooser = gtk.FileChooserDialog(title=_('Export ...'),
460                                        action=gtk.FILE_CHOOSER_ACTION_SAVE,
461                                        buttons=(gtk.STOCK_CANCEL,
462                                                 gtk.RESPONSE_CANCEL,
463                                                 gtk.STOCK_SAVE,
464                                                 gtk.RESPONSE_OK))
465        chooser.set_do_overwrite_confirmation(True)
466        chooser.set_current_folder(self.export_directory)
467        filter_pdf = gtk.FileFilter()
468        filter_pdf.set_name(_('PDF files'))
469        filter_pdf.add_mime_type('application/pdf')
470        chooser.add_filter(filter_pdf)
471
472        filter_all = gtk.FileFilter()
473        filter_all.set_name(_('All files'))
474        filter_all.add_pattern('*')
475        chooser.add_filter(filter_all)
476
477        while True:
478            response = chooser.run()
479            if response == gtk.RESPONSE_OK:
480                file_out = chooser.get_filename()
481                (path, shortname) = os.path.split(file_out)
482                (shortname, ext) = os.path.splitext(shortname)
483                if ext.lower() != '.pdf':
484                    file_out = file_out + '.pdf'
485                try:
486                    self.export_to_file(file_out, only_selected)
487                    self.export_directory = path
488                    self.set_unsaved(False)
489                except Exception, e:
490                    chooser.destroy()
491                    error_msg_dlg = gtk.MessageDialog(flags=gtk.DIALOG_MODAL,
492                                                      type=gtk.MESSAGE_ERROR,
493                                                      message_format=str(e),
494                                                      buttons=gtk.BUTTONS_OK)
495                    response = error_msg_dlg.run()
496                    if response == gtk.RESPONSE_OK:
497                        error_msg_dlg.destroy()
498                    return
499            break
500        chooser.destroy()
501
502    def export_to_file(self, file_out, only_selected=False):
503        """Export to file"""
504
505        selection = self.iconview.get_selected_items()
506        pdf_output = PdfFileWriter()
507        pdf_input = []
508        for pdfdoc in self.pdfqueue:
509            pdfdoc_inp = PdfFileReader(file(pdfdoc.copyname, 'rb'))
510            if pdfdoc_inp.getIsEncrypted():
511                try: # Workaround for lp:#355479
512                    stat = pdfdoc_inp.decrypt('')
513                except:
514                    stat = 0
515                if (stat!=1):
516                    errmsg = _('File %s is encrypted.\n'
517                               'Support for encrypted files has not been implemented yet.\n'
518                               'File export failed.') % pdfdoc.filename
519                    raise Exception, errmsg
520                #FIXME
521                #else
522                #   ask for password and decrypt file
523            pdf_input.append(pdfdoc_inp)
524
525        for row in self.model:
526
527            if only_selected and row.path not in selection:
528                continue
529
530            # add pages from input to output document
531            nfile = row[2]
532            npage = row[3]
533            current_page = copy(pdf_input[nfile-1].getPage(npage-1))
534            angle = row[6]
535            angle0 = current_page.get("/Rotate",0)
536            crop = [row[7],row[8],row[9],row[10]]
537            if angle != 0:
538                current_page.rotateClockwise(angle)
539            if crop != [0.,0.,0.,0.]:
540                rotate_times = (((angle + angle0) % 360 + 45) / 90) % 4
541                crop_init = crop
542                if rotate_times != 0:
543                    perm = [0,2,1,3]
544                    for it in range(rotate_times):
545                        perm.append(perm.pop(0))
546                    perm.insert(1,perm.pop(2))
547                    crop = [crop_init[perm[side]] for side in range(4)]
548                #(x1, y1) = current_page.cropBox.lowerLeft
549                #(x2, y2) = current_page.cropBox.upperRight
550                (x1, y1) = [float(xy) for xy in current_page.mediaBox.lowerLeft]
551                (x2, y2) = [float(xy) for xy in current_page.mediaBox.upperRight]
552                x1_new = int(x1 + (x2-x1) * crop[0])
553                x2_new = int(x2 - (x2-x1) * crop[1])
554                y1_new = int(y1 + (y2-y1) * crop[3])
555                y2_new = int(y2 - (y2-y1) * crop[2])
556                #current_page.cropBox.lowerLeft = (x1_new, y1_new)
557                #current_page.cropBox.upperRight = (x2_new, y2_new)
558                current_page.mediaBox.lowerLeft = (x1_new, y1_new)
559                current_page.mediaBox.upperRight = (x2_new, y2_new)
560
561            pdf_output.addPage(current_page)
562
563        # finally, write "output" to document-output.pdf
564        pdf_output.write(file(file_out, 'wb'))
565
566    def on_action_add_doc_activate(self, widget, data=None):
567        """Import doc"""
568
569        chooser = gtk.FileChooserDialog(title=_('Import...'),
570                                        action=gtk.FILE_CHOOSER_ACTION_OPEN,
571                                        buttons=(gtk.STOCK_CANCEL,
572                                                  gtk.RESPONSE_CANCEL,
573                                                  gtk.STOCK_OPEN,
574                                                  gtk.RESPONSE_OK))
575        chooser.set_current_folder(self.import_directory)
576        chooser.set_select_multiple(True)
577
578        filter_all = gtk.FileFilter()
579        filter_all.set_name(_('All files'))
580        filter_all.add_pattern('*')
581        chooser.add_filter(filter_all)
582
583        filter_pdf = gtk.FileFilter()
584        filter_pdf.set_name(_('PDF files'))
585        filter_pdf.add_mime_type('application/pdf')
586        chooser.add_filter(filter_pdf)
587        chooser.set_filter(filter_pdf)
588
589        response = chooser.run()
590        if response == gtk.RESPONSE_OK:
591            for filename in chooser.get_filenames():
592                if os.path.isfile(filename):
593                    # FIXME
594                    f = gio.File(filename)
595                    f_info = f.query_info('standard::content-type')
596                    mime_type = f_info.get_content_type()
597                    expected_mime_type = 'application/pdf'
598
599                    if mime_type == expected_mime_type:
600                        self.add_pdf_pages(filename)
601                    elif mime_type[:34] == 'application/vnd.oasis.opendocument':
602                        print(_('OpenDocument not supported yet!'))
603                    elif mime_type[:5] == 'image':
604                        print(_('Image file not supported yet!'))
605                    else:
606                        print(_('File type not supported!'))
607                else:
608                    print(_('File %s does not exist') % filename)
609        elif response == gtk.RESPONSE_CANCEL:
610            print(_('Closed, no files selected'))
611        chooser.destroy()
612        gobject.idle_add(self.retitle)
613
614    def clear_selected(self, button=None):
615        """Removes the selected elements in the IconView"""
616
617        model = self.iconview.get_model()
618        selection = self.iconview.get_selected_items()
619        if selection:
620            selection.sort(reverse=True)
621            self.set_unsaved(True)
622            for path in selection:
623                iter = model.get_iter(path)
624                model.remove(iter)
625            path = selection[-1]
626            self.iconview.select_path(path)
627            if not self.iconview.path_is_selected(path):
628                if len(model) > 0:      # select the last row
629                    row = model[-1]
630                    path = row.path
631                    self.iconview.select_path(path)
632            self.iconview.grab_focus()
633
634    def iv_drag_begin(self, iconview, context):
635        """Sets custom icon on drag begin for multiple items selected"""
636
637        if len(iconview.get_selected_items()) > 1:
638            iconview.stop_emission('drag_begin')
639            context.set_icon_stock(gtk.STOCK_DND_MULTIPLE, 0, 0)
640
641    def iv_dnd_get_data(self, iconview, context,
642                        selection_data, target_id, etime):
643        """Handles requests for data by drag and drop in iconview"""
644
645        model = iconview.get_model()
646        selection = self.iconview.get_selected_items()
647        selection.sort(key=lambda x: x[0])
648        data = []
649        for path in selection:
650            if selection_data.target == 'MODEL_ROW_INTERN':
651                data.append(str(path[0]))
652            elif selection_data.target == 'MODEL_ROW_EXTERN':
653                iter = model.get_iter(path)
654                nfile, npage, angle = model.get(iter, 2, 3, 6)
655                crop = model.get(iter, 7, 8, 9, 10)
656                pdfdoc = self.pdfqueue[nfile - 1]
657                data.append('\n'.join([pdfdoc.filename,
658                                       str(npage),
659                                       str(angle)] +
660                                       [str(side) for side in crop]))
661        if data:
662            data = '\n;\n'.join(data)
663            selection_data.set(selection_data.target, 8, data)
664
665    def iv_dnd_received_data(self, iconview, context, x, y,
666                             selection_data, target_id, etime):
667        """Handles received data by drag and drop in iconview"""
668
669        model = iconview.get_model()
670        data = selection_data.data
671        if data:
672            data = data.split('\n;\n')
673            drop_info = iconview.get_dest_item_at_pos(x, y)
674            iter_to = None
675            if drop_info:
676                path, position = drop_info
677                ref_to = gtk.TreeRowReference(model,path)
678            else:
679                position = gtk.ICON_VIEW_DROP_RIGHT
680                if len(model) > 0:  #find the iterator of the last row
681                    row = model[-1]
682                    path = row.path
683                    ref_to = gtk.TreeRowReference(model,path)
684            if ref_to:
685                before = (position == gtk.ICON_VIEW_DROP_LEFT
686                          or position == gtk.ICON_VIEW_DROP_ABOVE)
687                #if target_id == self.MODEL_ROW_INTERN:
688                if selection_data.target == 'MODEL_ROW_INTERN':
689                    if before:
690                        data.sort(key=int)
691                    else:
692                        data.sort(key=int,reverse=True)
693                    ref_from_list = [gtk.TreeRowReference(model,path)
694                                     for path in data]
695                    for ref_from in ref_from_list:
696                        path = ref_to.get_path()
697                        iter_to = model.get_iter(path)
698                        path = ref_from.get_path()
699                        iter_from = model.get_iter(path)
700                        row = model[iter_from]
701                        if before:
702                            model.insert_before(iter_to, row)
703                        else:
704                            model.insert_after(iter_to, row)
705                    if context.action == gtk.gdk.ACTION_MOVE:
706                        for ref_from in ref_from_list:
707                            path = ref_from.get_path()
708                            iter_from = model.get_iter(path)
709                            model.remove(iter_from)
710
711                #elif target_id == self.MODEL_ROW_EXTERN:
712                elif selection_data.target == 'MODEL_ROW_EXTERN':
713                    if not before:
714                        data.reverse()
715                    while data:
716                        tmp = data.pop(0).split('\n')
717                        filename = tmp[0]
718                        npage, angle = [int(k) for k in tmp[1:3]]
719                        crop = [float(side) for side in tmp[3:7]]
720                        if self.add_pdf_pages(filename, npage, npage,
721                                                        angle, crop):
722                            if len(model) > 0:
723                                path = ref_to.get_path()
724                                iter_to = model.get_iter(path)
725                                row = model[-1] #the last row
726                                path = row.path
727                                iter_from = model.get_iter(path)
728                                if before:
729                                    model.move_before(iter_from, iter_to)
730                                else:
731                                    model.move_after(iter_from, iter_to)
732                                if context.action == gtk.gdk.ACTION_MOVE:
733                                    context.finish(True, True, etime)
734
735    def iv_dnd_data_delete(self, widget, context):
736        """Deletes dnd items after a successful move operation"""
737
738        model = self.iconview.get_model()
739        selection = self.iconview.get_selected_items()
740        ref_del_list = [gtk.TreeRowReference(model,path) for path in selection]
741        for ref_del in ref_del_list:
742            path = ref_del.get_path()
743            iter = model.get_iter(path)
744            model.remove(iter)
745
746    def iv_dnd_motion(self, iconview, context, x, y, etime):
747        """Handles the drag-motion signal in order to auto-scroll the view"""
748
749        autoscroll_area = 40
750        sw_vadj = self.sw.get_vadjustment()
751        sw_height = self.sw.get_allocation().height
752        if y -sw_vadj.get_value() < autoscroll_area:
753            if not self.iv_auto_scroll_timer:
754                self.iv_auto_scroll_direction = gtk.DIR_UP
755                self.iv_auto_scroll_timer = gobject.timeout_add(150,
756                                                                self.iv_auto_scroll)
757        elif y -sw_vadj.get_value() > sw_height - autoscroll_area:
758            if not self.iv_auto_scroll_timer:
759                self.iv_auto_scroll_direction = gtk.DIR_DOWN
760                self.iv_auto_scroll_timer = gobject.timeout_add(150,
761                                                                self.iv_auto_scroll)
762        elif self.iv_auto_scroll_timer:
763            gobject.source_remove(self.iv_auto_scroll_timer)
764            self.iv_auto_scroll_timer = None
765
766    def iv_dnd_leave_end(self, widget, context, ignored=None):
767        """Ends the auto-scroll during DND"""
768
769        if self.iv_auto_scroll_timer:
770            gobject.source_remove(self.iv_auto_scroll_timer)
771            self.iv_auto_scroll_timer = None
772
773    def iv_auto_scroll(self):
774        """Timeout routine for auto-scroll"""
775
776        sw_vadj = self.sw.get_vadjustment()
777        sw_vpos = sw_vadj.get_value()
778        if self.iv_auto_scroll_direction == gtk.DIR_UP:
779            sw_vpos -= sw_vadj.step_increment
780            sw_vadj.set_value(max(sw_vpos, sw_vadj.lower))
781        elif self.iv_auto_scroll_direction == gtk.DIR_DOWN:
782            sw_vpos += sw_vadj.step_increment
783            sw_vadj.set_value(min(sw_vpos, sw_vadj.upper - sw_vadj.page_size))
784        return True  #call me again
785
786    def iv_button_press_event(self, iconview, event):
787        """Manages mouse clicks on the iconview"""
788
789        if event.button == 3:
790            x = int(event.x)
791            y = int(event.y)
792            time = event.time
793            path = iconview.get_path_at_pos(x, y)
794            selection = iconview.get_selected_items()
795            if path:
796                if path not in selection:
797                    iconview.unselect_all()
798                iconview.select_path(path)
799                iconview.grab_focus()
800                self.popup.popup(None, None, None, event.button, time)
801            return 1
802
803    def sw_dnd_received_data(self, scrolledwindow, context, x, y,
804                             selection_data, target_id, etime):
805        """Handles received data by drag and drop in scrolledwindow"""
806
807        data = selection_data.data
808        if target_id == self.MODEL_ROW_EXTERN:
809            self.model
810            if data:
811                data = data.split('\n;\n')
812            while data:
813                tmp = data.pop(0).split('\n')
814                filename = tmp[0]
815                npage, angle = [int(k) for k in tmp[1:3]]
816                crop = [float(side) for side in tmp[3:7]]
817                if self.add_pdf_pages(filename, npage, npage, angle, crop):
818                    if context.action == gtk.gdk.ACTION_MOVE:
819                        context.finish(True, True, etime)
820        elif target_id == self.TEXT_URI_LIST:
821            uri = data.strip()
822            uri_splitted = uri.split() # we may have more than one file dropped
823            for uri in uri_splitted:
824                filename = self.get_file_path_from_dnd_dropped_uri(uri)
825                if os.path.isfile(filename): # is it file?
826                    self.add_pdf_pages(filename)
827
828    def sw_button_press_event(self, scrolledwindow, event):
829        """Unselects all items in iconview on mouse click in scrolledwindow"""
830
831        if event.button == 1:
832            self.iconview.unselect_all()
833
834    def sw_scroll_event(self, scrolledwindow, event):
835        """Manages mouse scroll events in scrolledwindow"""
836
837        if event.state & gtk.gdk.CONTROL_MASK:
838            if event.direction == gtk.gdk.SCROLL_UP:
839                self.zoom_change(1)
840                return 1
841            elif event.direction == gtk.gdk.SCROLL_DOWN:
842                self.zoom_change(-1)
843                return 1
844
845    def zoom_set(self, level):
846        """Sets the zoom level"""
847        self.zoom_level = max(min(level, 5), -24)
848        self.zoom_scale = 1.1 ** self.zoom_level
849        for row in self.model:
850            row[4] = self.zoom_scale
851        self.reset_iv_width()
852
853    def zoom_change(self, step=5):
854        """Modifies the zoom level"""
855        self.zoom_set(self.zoom_level + step)
856
857    def zoom_in(self, widget=None):
858        """Increases the zoom level by 5 steps"""
859        self.zoom_change(5)
860
861    def zoom_out(self, widget=None, step=5):
862        """Reduces the zoom level by 5 steps"""
863        self.zoom_change(-5)
864
865    def get_file_path_from_dnd_dropped_uri(self, uri):
866        """Extracts the path from an uri"""
867
868        path = urllib.url2pathname(uri) # escape special chars
869        path = path.strip('\r\n\x00')   # remove \r\n and NULL
870
871        # get the path to file
872        if path.startswith('file:\\\\\\'): # windows
873            path = path[8:]  # 8 is len('file:///')
874        elif path.startswith('file://'):   # nautilus, rox
875            path = path[7:]  # 7 is len('file://')
876        elif path.startswith('file:'):     # xffm
877            path = path[5:]  # 5 is len('file:')
878        return path
879
880    def rotate_page_right(self, widget, data=None):
881        self.rotate_page(90)
882
883    def rotate_page_left(self, widget, data=None):
884        self.rotate_page(-90)
885
886    def rotate_page(self, angle):
887        """Rotates the selected page in the IconView"""
888
889        model = self.iconview.get_model()
890        selection = self.iconview.get_selected_items()
891        if len(selection) > 0:
892            self.set_unsaved(True)
893        rotate_times = (((-angle) % 360 + 45) / 90) % 4
894        if rotate_times is not 0:
895            for path in selection:
896                iter = model.get_iter(path)
897                nfile = model.get_value(iter, 2)
898                npage = model.get_value(iter, 3)
899
900                crop = [0.,0.,0.,0.]
901                perm = [0,2,1,3]
902                for it in range(rotate_times):
903                    perm.append(perm.pop(0))
904                perm.insert(1,perm.pop(2))
905                crop = [model.get_value(iter, 7 + perm[side]) for side in range(4)]
906                for side in range(4):
907                    model.set_value(iter, 7 + side, crop[side])
908
909                new_angle = model.get_value(iter, 6) + int(angle)
910                new_angle = new_angle % 360
911                model.set_value(iter, 6, new_angle)
912                self.update_geometry(iter)
913        self.reset_iv_width()
914
915    def crop_page_dialog(self, widget):
916        """Opens a dialog box to define margins for page cropping"""
917
918        sides = ('L', 'R', 'T', 'B')
919        side_names = {'L':_('Left'), 'R':_('Right'),
920                      'T':_('Top'), 'B':_('Bottom') }
921        opposite_sides = {'L':'R', 'R':'L', 'T':'B', 'B':'T' }
922
923        def set_crop_value(spinbutton, side):
924           opp_side = opposite_sides[side]
925           pos = sides.index(opp_side)
926           adj = spin_list[pos].get_adjustment()
927           adj.set_upper(99.0 - spinbutton.get_value())
928
929        model = self.iconview.get_model()
930        selection = self.iconview.get_selected_items()
931
932        crop = [0.,0.,0.,0.]
933        if selection:
934            path = selection[0]
935            pos = model.get_iter(path)
936            crop = [model.get_value(pos, 7 + side) for side in range(4)]
937
938        dialog = gtk.Dialog(title=(_('Crop Selected Pages')),
939                            parent=self.window,
940                            flags=gtk.DIALOG_MODAL,
941                            buttons=(gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL,
942                                     gtk.STOCK_OK, gtk.RESPONSE_OK))
943        dialog.set_size_request(340, 250)
944        dialog.set_default_response(gtk.RESPONSE_OK)
945
946        frame = gtk.Frame(_('Crop Margins'))
947        dialog.vbox.pack_start(frame, False, False, 20)
948
949        vbox = gtk.VBox(False, 0)
950        frame.add(vbox)
951
952        spin_list = []
953        units = 2 * [_('% of width')] + 2 * [_('% of height')]
954        for side in sides:
955            hbox = gtk.HBox(True, 0)
956            vbox.pack_start(hbox, False, False, 5)
957
958            label = gtk.Label(side_names[side])
959            label.set_alignment(0, 0.0)
960            hbox.pack_start(label, True, True, 20)
961
962            adj = gtk.Adjustment(100.*crop.pop(0), 0.0, 99.0, 1.0, 5.0, 0.0)
963            spin = gtk.SpinButton(adj, 0, 1)
964            spin.set_activates_default(True)
965            spin.connect('value-changed', set_crop_value, side)
966            spin_list.append(spin)
967            hbox.pack_start(spin, False, False, 30)
968
969            label = gtk.Label(units.pop(0))
970            label.set_alignment(0, 0.0)
971            hbox.pack_start(label, True, True, 0)
972
973        dialog.show_all()
974        result = dialog.run()
975
976        if result == gtk.RESPONSE_OK:
977            modified = False
978            crop = [spin.get_value()/100. for spin in spin_list]
979            for path in selection:
980                pos = model.get_iter(path)
981                for it in range(4):
982                    old_val = model.get_value(pos, 7 + it)
983                    model.set_value(pos, 7 + it, crop[it])
984                    if crop[it] != old_val:
985                        modified = True
986                self.update_geometry(pos)
987            if modified:
988                self.set_unsaved(True)
989            self.reset_iv_width()
990        elif result == gtk.RESPONSE_CANCEL:
991            print(_('Dialog closed'))
992        dialog.destroy()
993
994    def about_dialog(self, widget, data=None):
995        about_dialog = gtk.AboutDialog()
996        try:
997            about_dialog.set_transient_for(self.window)
998            about_dialog.set_modal(True)
999        except:
1000            pass
1001        # FIXME
1002        about_dialog.set_name(APPNAME)
1003        about_dialog.set_version(VERSION)
1004        about_dialog.set_comments(_(
1005            '%s is a tool for rearranging and modifying PDF files. ' \
1006            'Developed using GTK+ and Python') % APPNAME)
1007        about_dialog.set_authors(['Konstantinos Poulios',])
1008        about_dialog.set_website_label(WEBSITE)
1009        about_dialog.set_logo_icon_name('pdfshuffler')
1010        about_dialog.set_license(LICENSE)
1011        about_dialog.connect('response', lambda w, *args: w.destroy())
1012        about_dialog.connect('delete_event', lambda w, *args: w.destroy())
1013        about_dialog.show_all()
1014
1015
1016class PDF_Doc:
1017    """Class handling PDF documents"""
1018
1019    def __init__(self, filename, nfile, tmp_dir):
1020
1021        self.filename = os.path.abspath(filename)
1022        (self.path, self.shortname) = os.path.split(self.filename)
1023        (self.shortname, self.ext) = os.path.splitext(self.shortname)
1024        f = gio.File(filename)
1025        mime_type = f.query_info('standard::content-type').get_content_type()
1026        expected_mime_type = 'application/pdf'
1027        file_prefix = 'file://'
1028
1029        if mime_type == expected_mime_type:
1030            self.nfile = nfile + 1
1031            self.mtime = os.path.getmtime(filename)
1032            self.copyname = os.path.join(tmp_dir, '%02d_' % self.nfile +
1033                                                  self.shortname + '.pdf')
1034            shutil.copy(self.filename, self.copyname)
1035            self.document = poppler.document_new_from_file (file_prefix + self.copyname, None)
1036            self.npage = self.document.get_n_pages()
1037        else:
1038            self.nfile = 0
1039            self.npage = 0
1040
1041
1042class PDF_Renderer(threading.Thread,gobject.GObject):
1043
1044    def __init__(self, model, pdfqueue, resample=1.):
1045        threading.Thread.__init__(self)
1046        gobject.GObject.__init__(self)
1047        self.model = model
1048        self.pdfqueue = pdfqueue
1049        self.resample = resample
1050        self.quit = False
1051
1052    def run(self):
1053        for idx, row in enumerate(self.model):
1054            if self.quit:
1055                return
1056            if not row[1]:
1057                try:
1058                    nfile = row[2]
1059                    npage = row[3]
1060                    pdfdoc = self.pdfqueue[nfile - 1]
1061                    page = pdfdoc.document.get_page(npage-1)
1062                    w, h = page.get_size()
1063                    thumbnail = cairo.ImageSurface(cairo.FORMAT_ARGB32,
1064                                                   int(w/self.resample),
1065                                                   int(h/self.resample))
1066                    cr = cairo.Context(thumbnail)
1067                    if self.resample != 1.:
1068                        cr.scale(1./self.resample, 1./self.resample)
1069                    page.render(cr)
1070                    time.sleep(0.003)
1071                    gobject.idle_add(self.emit,'update_thumbnail',
1072                                     idx, thumbnail, self.resample,
1073                                     priority=gobject.PRIORITY_LOW)
1074                except Exception,e:
1075                    print e
1076
1077
1078def main():
1079    """This function starts PdfShuffler"""
1080    gobject.threads_init()
1081    PdfShuffler()
1082    gtk.main()
1083
1084if __name__ == '__main__':
1085    main()
1086
Note: See TracBrowser for help on using the repository browser.