上一篇文章中颠蕴,我們實(shí)現(xiàn)了 PDF 閱讀器的初始界面拉鹃。這一次,新增了閱讀功能,可以實(shí)現(xiàn)基本的翻頁以及縮放等操作员舵。不過,暫時(shí)只可以同時(shí)閱讀一本書萝快。
下圖為效果圖:
下面我來介紹下川蒙,這次主要新增的功能:
選項(xiàng)卡
QTabWidget 可以允許我們?cè)谝粋€(gè)窗口顯示多個(gè)頁面。對(duì)于書庫的這個(gè)選項(xiàng)卡,頁面顯示為 self.table ,即初始界面
self.table(QTableWidget) -> self.tabwidget(QTabWidge)蹈矮。
# 初始化選項(xiàng)卡
self.tabwidget = QTabWidget()
# 添加書庫選項(xiàng)卡
self.tabwidget.addTab(self.table, '書庫')
self.setCentralWidget(self.tabwidget)
# 設(shè)置選項(xiàng)卡可以關(guān)閉
self.tabwidget.setTabsClosable(True)
# 點(diǎn)擊選項(xiàng)卡叉號(hào)時(shí)砰逻,執(zhí)行 removeTabab 操作
self.tabwidget.tabCloseRequested[int].connect(self.remove_tab)
新建選項(xiàng)卡:每次開始閱讀時(shí),新建一個(gè)選項(xiàng)卡泛鸟,名稱為文件名蝠咆。
def read_book(self, fname):
# self.close()
# 內(nèi)存有可能泄露
self.doc = fitz.open(fname)
# metadata = doc.metadata
title = fname.split('/' or '\\')[-1].replace('.pdf', '')
vbox = self.book_area(self.doc.loadPage(0))
self.book_add_tab(title, vbox)
其中,我們要求主選項(xiàng)卡谈况,即書庫選項(xiàng)卡是不可以關(guān)閉的勺美。
def remove_tab(self, index):
if index:
# 當(dāng)前頁數(shù)
self.current_page = 0
self.tabwidget.removeTab(index)
# 正在閱讀的書
self.read_list.pop(index)
閱讀界面的選項(xiàng)卡對(duì)應(yīng)的頁面區(qū)域?yàn)?QScrollArea 递胧,QScrollArea 支持滾輪操作。也就是說赡茸,如果我們縮放 PDF 頁面大小超過 QScrollArea 的大小缎脾,那么就會(huì)自動(dòng)出現(xiàn)滾輪,以便我們?yōu)g覽頁面占卧。其中遗菠,MyArea 類是對(duì) QScrollArea 的重載,綁定了快捷鍵以支持翻頁以及縮放等操作华蜒。
Pixmap -> label -> area(MyArea) -> vbox(QVBoxLayout) -> tab(QWidget) -> self.tabwidget(QTabWidge)辙纬。
def book_add_tab(self, title, vbox):
tab = QWidget()
tab.setLayout(vbox)
# tab 為頁面,title 為標(biāo)簽名稱
self.tabwidget.addTab(tab, title)
def book_area(self, page):
label = self.page_pixmap(page)
# area = QScrollArea()
area = MyArea(self)
area.init(self)
area.setWidget(label)
vbox = QVBoxLayout()
vbox.addWidget(area)
return vbox
下面我們來看看, MyArea 這個(gè)類該如何定義:
MyArea(QScrollArea)
這里,我們定義了 init 方法,用來接受 Reader 主類 的 self 參數(shù), 即通過 self.widget 調(diào)用 Reader 類的實(shí)例方法叭喜。
在 init_action 函數(shù)中,我們新建了四個(gè) QShortCut 實(shí)例贺拣,分別支持快捷鍵實(shí)現(xiàn)縮小、放大捂蕴、下一頁譬涡、上一頁的操作。
class MyArea(QScrollArea):
def init(self, widget):
self.widget = widget
self.init_action()
def init_action(self):
zoom_minus = QShortcut(QKeySequence("Ctrl+-"), self)
zoom_minus.activated.connect(self.minus)
zoom_plus = QShortcut(QKeySequence("Ctrl+="), self)
zoom_plus.activated.connect(self.plus)
switch_left = QShortcut(QKeySequence(Qt.Key_Left), self)
switch_left.activated.connect(self.left)
switch_right = QShortcut(QKeySequence(Qt.Key_Right), self)
switch_right.activated.connect(self.right)
def plus(self):
self.widget.zoom_book(plus=True)
def minus(self):
self.widget.zoom_book(plus=False)
def right(self):
self.widget.switch_page(right=True)
def left(self):
self.widget.switch_page(right=False)
下面啥辨,我們來介紹縮放與翻頁功能的具體實(shí)現(xiàn):
縮放功能
self.size 用來存儲(chǔ)頁面大小涡匀,self.page 正是根據(jù) self.size 來實(shí)現(xiàn)縮放功能。
def zoom_book(self, plus=True):
a, b = self.size
if plus:
a += 0.4
b += 0.4
self.size = (a, b)
self.set_page()
elif not plus and a > 0:
if a >= 1:
a -= 0.4
b -= 0.4
self.size = (a, b)
self.set_page()
Pixmap -> label -> area(MyArea) -> vbox(QVBoxLayout) -> tab(QWidget) -> self.tabwidget(QTabWidge)溉知。
tab 獲取 tab 對(duì)象陨瘩,layout 獲取 vbox 對(duì)象,widget 獲取 area 對(duì)象级乍,直接更改 area 上 label 控件舌劳。
def set_page(self):
# 加載頁面
page = self.doc.loadPage(self.current_page)
# 獲取當(dāng)前 Widget
tab = self.tabwidget.currentWidget()
# 獲取當(dāng)前的 Layout
layout = tab.layout()
# 獲取 Layout 上的控件
widget = layout.itemAt(0).widget()
# 獲取已經(jīng)繪制好的 label 對(duì)象
label = self.page_pixmap(page)
# 將 widget 的內(nèi)容更改為現(xiàn)在的 label 對(duì)象
widget.setWidget(label)
接著我們介紹如何實(shí)現(xiàn)翻頁功能
翻頁功能
這次,我們實(shí)現(xiàn)的 PDF 閱讀器只能同時(shí)閱讀一本書卡者,所以翻頁功能只需由 self.current_page 控制就行蒿囤。
self.doc.pageCount 為總頁數(shù),當(dāng)前頁數(shù)不能為負(fù)數(shù)或者大于總頁數(shù)崇决。更改完 self.current_page 之后材诽,就可以執(zhí)行 self.set_page 操作,直接更改 area 上的 label 控件恒傻。
def set_current_page(self, right):
if right and self.current_page < self.doc.pageCount - 1:
self.current_page += 1
elif not right and self.current_page > 0:
self.current_page -= 1
def switch_page(self, right=True):
self.set_current_page(right)
self.set_page()
關(guān)注微信公眾號(hào):Python高效編程脸侥,微信后臺(tái)回復(fù) 2019330 獲取源碼。