用PyQt4+Python写一个简单的EPub阅读器(3/3)

最后一篇,这次我们加上几个函数,使得这个阅读器能够按照章节阅读电子书,前两篇我们将完成了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)

最终的效果:

最后,给出源代码地址

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转自:作者简介作者:唐新华 (xhsmart@263.net)软件工程师    信号和槽作为QT的核心机制在QT编...
    njukay阅读 1,368评论 0 49
  • 韩元旭、余橙、沈开洋 Qt介绍 Qt是一个跨平台的C++图形用户界面应用程序框架。它早在1991年奇趣科技公司两位...
    开洋_shen阅读 16,241评论 4 24
  • 信号和槽(Signals and Slots) Qt库第一个认识到在几乎所有情况下,程序员不需要或甚至不想知道所有...
    珞珈村下山阅读 9,912评论 0 23
  • 接上一篇,这一篇我们写GUI。 上一篇提出了图书仓库的概念,更具体的想法是:这个仓库是一个文件夹,所有打开的书都往...
    knarfeh阅读 2,766评论 0 3
  • 海棠果被不少朋友称作“小苹果”,的确若是放大来说的确有着七八分相似之处。不过这种小巧玲珑的果实吃起来量不多,也方便...
    执笔青衫阅读 1,040评论 1 1