最后一篇,这次我们加上几个函数,使得这个阅读器能够按照章节阅读电子书,前两篇我们将完成了Book模块,实现了电子书的数据抽象,然后写了GUI,完成了一个简单的功能,在仓库中插入图书,把LibraryTableWidget刷新,使得图书在library呈现,可以看到书名,作者。接下来我们完善这个程序。
- 载入书籍的章节
效果是这样的:
首先,我们在BookView模块中加入这样一个函数:
def load_book(self, book_id):
self.book = Book(book_id)
self.chapter_list.clear()
for chapter in self.book.chapters:
self.chapter_list.addItem(chapter[0])
self.chapter_list.setCurrentRow(0)
根据book_id(也就是epub文件名),刷新chapter_list的内容,把每个章节的名称作为QListWidget的item添加进来。
然后,什么时候让这个函数起作用,怎么让这个函数起作用?我们要求的逻辑是双击Library的Item,然后载入这本书的章节。这里就要用到Qt的信号槽机制了,网上关于Qt信号槽机制的文章很多,书本上也有介绍,大多都是balabala说一大堆,堆概念堆名词,我尝试用几句话来说说这个机制的核心。其实它跟IFTTT(if this then that)这个工具是很像的,一个对象发送信号,另一个对象接收到信号就做出相应的动作,为什么要这样设计呢?其实我们可以尝试思考Qt设计者的初衷,因为这个过程是符合人类思考过程的,能够让编程更简单,如果了解了其他的GUI库的事件处理方式,如Java的监听机制,WFT的消息映射,就能发觉Qt的signal/slot机制有多好了,理解了这一点,再去看这方面的资料就能得心应手。
这里没有谈Qt信号槽的细节,我只是提出一种思考方式,因为我也是初学者,但这种思考方式能让我们在遇到稍特别的细节时会不感到诧异,比如,连接信号和槽的connect函数可以使得信号和信号连接,一个button发送了一个click()信号,于是asignal()信号也发送了,就如一个人头疼,他没有直接叫医生,而是通知了朋友,让朋友帮忙叫医生,又比如一个信号可以和多个槽连接,就如一个人头疼,他发了一条朋友圈,于是当医生的朋友都来了。。。。这些过程都是容易理解的,是符合人类思考过程的。
好了,瞎扯淡环节结束,我们继续。让chapter_list的双击Item事件与book_view的load_book函数绑定,我们可以在LibraryTableWidget中加上:
def create_connections(self):
self.connect(self, SIGNAL("itemDoubleClicked(QTableWidgetItem *)"), self.view_book)
def view_book(self):
book_id = self.library['books'][self.currentRow()]['id']
self.book_view.load_book(book_id)
当然,也需要在LibraryTableWidget, __init__
函数中加上 :
self.create_connections()
完成之后,再运行main.py就能实现上面展示的内容。
- web_view显示图书内容的部分
其实跟上面的方式大同小异,依葫芦画瓢, 我就直接给出代码了:
在Book模块中加上
def get_chapter(self, num):
return self.f.read(self.oebps_folder+self.chapters[num][1])
根据章节数目获得html文件内容
在BookView中加上
def set_chapter(self, num=None):
if num is None:
num = self.chapter_list.currentRow()
if num < 0:
num = len(self.book.chapters) - 1
elif num >= len(self.book.chapters):
num = 0
self.web_view.setHtml(self.book.get_chapter(num).decode(encoding="utf-8"))
使得web_view可以根据章节数目显示电子书中该章节的内容
同样在BookView的__init__
函数中加上:
self.create_connections()
绑定signal/slot, 在BookView中加上create_connection函数:
def create_connections(self):
chlist = self.chapter_list
self.connect(self.next_button, SIGNAL("clicked()"), lambda:
chlist.setCurrentRow(0
if chlist.currentRow() == chlist.count() - 1
else chlist.currentRow() + 1))
self.connect(self.previous_button, SIGNAL("clicked()"), lambda:
chlist.setCurrentRow(chlist.count() - 1
if chlist.currentRow() == 0
else chlist.currentRow() - 1))
self.connect(self.chapter_list, SIGNAL("currentRowChanged(int)"),
self.set_chapter)
page = self.web_view.page()
page.setLinkDelegationPolicy(QWebPage.DelegateAllLinks)
self.connect(page, SIGNAL("linkClicked(const QUrl&)"),
self.link_clicked)
最终的效果:
最后,给出源代码地址