""" @created: 2018-08-19 18:00:00 @author: (c) 2018 Jorj X. McKie Display a PyMuPDF Document using Tkinter ------------------------------------------------------------------------------- Dependencies: ------------- PyMuPDF, PySimpleGUI (requires Python 3), Tkinter, PIL License: -------- GNU GPL V3+ Description ------------ Get filename and start displaying page 1. Please note that all file types of MuPDF are supported (including EPUB e-books and HTML files for example). Pages can be directly jumped to, or buttons can be used for paging. This version contains enhancements: * Use of PIL improves response times by a factor 3 or more. * Zooming is now flexible: only one button serves as a toggle. Arrow keys can be used for moving the window when zooming. We also interpret keyboard events (PageDown / PageUp) and mouse wheel actions to support paging as if a button was clicked. Similarly, we do not include a 'Quit' button. Instead, the ESCAPE key can be used, or cancelling the window. To improve paging performance, we are not directly creating pixmaps from pages, but instead from the fitz.DisplayList of the page. A display list will be stored in a list and looked up by page number. This way, zooming pixmaps and page re-visits will re-use a once-created display list. """ import sys import fitz import sys if sys.version_info[0] >= 3: import PySimpleGUI as sg else: import PySimpleGUI27 as sg import tkinter as tk from PIL import Image, ImageTk import time if len(sys.argv) == 1: fname = sg.PopupGetFile('Document Browser', 'Document file to open', no_window=True, file_types = ( ("PDF Files", "*.pdf"), ("XPS Files", "*.*xps"), ("Epub Files", "*.epub"), ("Fiction Books", "*.fb2"), ("Comic Books", "*.cbz"), ("HTML", "*.htm*") # add more document types here ) ) else: fname = sys.argv[1] if not fname: sg.Popup("Cancelling:", "No filename supplied") raise SystemExit("Cancelled: no filename supplied") doc = fitz.open(fname) page_count = len(doc) # used for response time statistics only fitz_img_time = 0.0 tk_img_time = 0.0 img_count = 1 # allocate storage for page display lists dlist_tab = [None] * page_count title = "PyMuPDF display of '%s', pages: %i" % (fname, page_count) def get_page(pno, zoom = False, max_size = None, first = False): """Return a PNG image for a document page number. """ dlist = dlist_tab[pno] # get display list of page number if not dlist: # create if not yet there dlist_tab[pno] = doc[pno].getDisplayList() dlist = dlist_tab[pno] r = dlist.rect # the page rectangle clip = r # ensure image fits screen: # exploit, but do not exceed width or height zoom_0 = 1 if max_size: zoom_0 = min(1, max_size[0] / r.width, max_size[1] / r.height) if zoom_0 == 1: zoom_0 = min(max_size[0] / r.width, max_size[1] / r.height) mat_0 = fitz.Matrix(zoom_0, zoom_0) if not zoom: # show total page pix = dlist.getPixmap(matrix = mat_0, alpha=False) else: mp = r.tl + (r.br - r.tl) * 0.5 # page rect center w2 = r.width / 2 h2 = r.height / 2 clip = r * 0.5 tl = zoom[0] # old top-left tl.x += zoom[1] * (w2 / 2) tl.x = max(0, tl.x) tl.x = min(w2, tl.x) tl.y += zoom[2] * (h2 / 2) tl.y = max(0, tl.y) tl.y = min(h2, tl.y) clip = fitz.Rect(tl, tl.x + w2, tl.y + h2) mat = mat_0 * fitz.Matrix(2, 2) # zoom matrix pix = dlist.getPixmap(alpha=False, matrix=mat, clip=clip) if first: # first call: tkinter still inactive img = pix.getPNGData() # so use fitz png output else: # else take tk photo image pilimg = Image.frombytes("RGB", [pix.width, pix.height], pix.samples) img = ImageTk.PhotoImage(pilimg) return img, clip.tl # return image, clip position root = tk.Tk() max_width = root.winfo_screenwidth() - 20 max_height = root.winfo_screenheight() - 135 max_size = (max_width, max_height) root.destroy() del root window = sg.Window(title, return_keyboard_events = True, location = (0,0), use_default_focus = False, no_titlebar=False) cur_page = 0 data, clip_pos = get_page(cur_page, zoom = False, max_size = max_size, first = True) image_elem = sg.Image(data = data) goto = sg.InputText(str(cur_page + 1), size=(5, 1), do_not_clear=True, key = "PageNumber") layout = [ [ sg.ReadButton('Next'), sg.ReadButton('Prev'), sg.Text('Page:'), goto, sg.Text('(%i)' % page_count), sg.ReadButton('Zoom'), sg.Text('(toggle on/off, use arrows to navigate while zooming)'), ], [image_elem], ] window.Layout(layout) # now define the buttons / events we want to handle enter_buttons = [chr(13), "Return:13"] quit_buttons = ["Escape:27", chr(27)] next_buttons = ["Next", "Next:34", "MouseWheel:Down"] prev_buttons = ["Prev", "Prior:33", "MouseWheel:Up"] Up = "Up:38" Left = "Left:37" Right = "Right:39" Down = "Down:40" zoom_buttons = ["Zoom", Up, Down, Left, Right] # all the buttons we will handle my_keys = enter_buttons + next_buttons + prev_buttons + zoom_buttons # old page store and zoom toggle old_page = 0 old_zoom = False while True: event, value = window.Read() if event is None and (value is None or value['PageNumber'] is None): break if event in quit_buttons: break zoom_pressed = False zoom = False if event in enter_buttons: try: cur_page = int(value['PageNumber']) - 1 # check if valid while cur_page < 0: cur_page += page_count except: cur_page = 0 # this guy's trying to fool me elif event in next_buttons: cur_page += 1 elif event in prev_buttons: cur_page -= 1 elif event == Up: zoom = (clip_pos, 0, -1) elif event == Down: zoom = (clip_pos, 0, 1) elif event == Left: zoom = (clip_pos, -1, 0) elif event == Right: zoom = (clip_pos, 1, 0) elif event == "Zoom": zoom_pressed = True zoom = (clip_pos, 0, 0) # sanitize page number if cur_page >= page_count: # wrap around cur_page = 0 while cur_page < 0: # pages > 0 look nicer cur_page += page_count if zoom_pressed and old_zoom: zoom = zoom_pressed = old_zoom = False t0 = time.perf_counter() data, clip_pos = get_page(cur_page, zoom = zoom, max_size = max_size, first = False) t1 = time.perf_counter() image_elem.Update(data = data) t2 = time.perf_counter() fitz_img_time += t1 - t0 tk_img_time += t2 - t1 img_count += 1 old_page = cur_page old_zoom = zoom_pressed or zoom # update page number field if event in my_keys: goto.Update(str(cur_page + 1)) # print some response time statistics if img_count > 0: print("response times for '%s'" % doc.name) print("%.4f" % (fitz_img_time/img_count), "sec fitz avg. image time") print("%.4f" % (tk_img_time/img_count), "sec tk avg. image time") print("%.4f" % ((fitz_img_time + tk_img_time)/img_count), "sec avg. total time") print(img_count, "images read") print(page_count, "pages")