将多个像素图显示到滚动区域的最有效方法是什么?

时间:2012-05-16 11:12:25

标签: python pdf qt4 poppler pixmap

我正在尝试使用PyQt4和python-poppler-qt4创建一个显示PDF文件的应用程序。

到目前为止,我已经设法通过加载使用Poppler生成的pixmaps来显示整个文档,在QLabel上设置并附加到QFrame。 QFrame显示在QScrollArea中。

它看起来非常好,直到实现缩放,这是通过重新生成像素图,以递增的分辨率完成的。这个过程需要将整个文档渲染成像素图,这显然需要时间并导致不必要的滞后。

逻辑希望我应该只显示我所看到的页面的图像(听起来像量子物理学)。我有两个选择:

  1. 使用QLabel创建空白页面,并在滚动区域中显示时将图像加载到它们上;
  2. 只创建一个页面,并在显示之前添加或删除先前或后续页面。
  3. 我不确定我是在正确的轨道上还是有其他选择。

    第一个选项似乎更可行,因为空白页面的可见性决定了必须上传像素图的时间(虽然我不知道如何在隐藏页面时删除该像素图)。然而,我不确定缩放是否会更快,因为600页的文档必须重新生成,尽管有空白页。

    第二个选项肯定会改进缩放,因为缩放时一次必须重新生成1到4页。然而,在第二种情况下,我不确定如何触发页面构造。

    你会建议什么?

2 个答案:

答案 0 :(得分:1)

忘记QLabel并直接绘制图像不是很简单:

from PyQt4.QtGui import *
import sys

app = QApplication(sys.argv)

class Test(QWidget):

    def __init__(self):
        super(Test, self).__init__()
        self.painter = QPainter()
        # placeholder for the real stuff to draw
        self.image = QImage("/tmp/test.jpg")

    def paintEvent(self, evt):
        rect = evt.rect()
        evt.accept()
        print rect
        self.painter.begin(self)
        zoomedImage = self.image   # ... calculate this for your images
        sourceRect = rect          # ... caluclate this ...
        # draw it directly
        self.painter.drawImage(rect, self.image, sourceRect)
        self.painter.end()


t = Test()
t.setGeometry(0,0,600,800)

s = QScrollArea()
s.setWidget(t)

s.setGeometry(0,0,300,400)
s.show()
app.exec_()

答案 1 :(得分:1)

我在问题中使用选项1找到答案:

def moveEvent(self, event):
    self.checkVisibility()
    event.ignore()

def resizeEvent(self, event):
    self.checkVisibility()
    event.ignore()

def checkVisibility(self):
    print "Checking visibility"
    for page in self.getPages():
        if not page.visibleRegion().isEmpty():
            if page.was_visible:
                pass
            else:
                print page.page_number, "became visible"
                page.was_visible = True
                self.applyImageToPage(page)
        else:
            if page.was_visible:
                print page.page_number, "became invisible"
                page.was_visible = False
            else:
                pass
def applyImageToPage(self, page):
    print "applying image to page", page.page_number
    source = self.getSourcePage(self.getPageNumber(page))
    scale = self.display.scale
        # this is where the error occurs
    image = source.renderToImage(72 * scale, 72 * scale)
    pixmap = QtGui.QPixmap.fromImage(image)
    page.setPixmap(pixmap)