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