根据内容的变化,从QWizardPage动态添加/删除Finish

时间:2016-06-12 22:08:44

标签: qt pyqt wizard transitions pyqt5

我在license wizard(使用PyQt5)上学习如何创建非线性向导的教程。但是我似乎陷入了一个问题。

我想要一个包含QComboBox的页面,其中所选项目确定包含组合框的当前QWizardPage是否为最终页面。

以下是目前该页面包含的内容:

class CalibrationPageSource(QWizardPage):
    def __init__(self, parent):
        super(CalibrationPageSource, self).__init__(parent)
        self.setTitle('Calibration Wizard')
        self.setSubTitle('Select the source for the calibration')

        layout = QVBoxLayout()
        label = QLabel('''
            <ul>
                <li><b>From calibration file</b>: browse and select an existing YAML calibration file that contains the camera matrix and distortion coefficients (for example from a previous calibration)</li>
                <li><b>From image files</b>: browse and select one or more image files with the calibration pattern visible inside each</li>
                <li><b>From stream</b> - if the calibration node is connected to an active <b><i>Device node</i></b> you can use its image stream to interactively calibrate your device</li>
            </ul>
        ''')
        label.setWordWrap(True)
        layout.addWidget(label)

        layout_sources = QHBoxLayout()
        label_sources = QLabel('Source:')
        self.selection_sources = QComboBox()
        self.selection_sources.addItem('Calibration file')
        self.selection_sources.addItem('Image files')
        self.selection_sources.addItem('Stream')
        self.selection_sources.currentIndexChanged['QString'].connect(self.source_changed)
        self.selection_sources.setCurrentIndex(1)
        layout_sources.addWidget(label_sources)
        layout_sources.addWidget(self.selection_sources)
        layout.addLayout(layout_sources)

        self.setLayout(layout)

    @pyqtSlot(str)
    def source_changed(self, source):
        if source == 'Calibration file':
            self.setFinalPage(True)
            # TODO Add file dialog
        else:
            self.setFinalPage(False)
            # TODO Remove file dialog (if present)

每当self.selection_sources的当前项目更改为Calibration file时,我想跳过使页面最终的向导的其余部分。在这种情况下,我想删除Next按钮。在所有其他情况下(目前只有两个:Image filesStream)我想让向导正常行动而不是最终页面。

我已尝试实施自定义isComplete(...)但问题是,如果选择Next,则会同时停用FinishCalibration file。我可以使用已禁用的Next按钮(而不是完全隐藏它),但在我的情况下,禁用Finish基本上没有意义。我真的很惊讶Next按钮存在。当最终页面到达时,是不是应该完全消失?

任何想法如何解决这个问题?我想过迭代QWizardPage中的项目并手动禁用/隐藏Next按钮,但我希​​望有一种更简单,开箱即用的方法。在当前状态下,Finish的动态插入正在运行,但由于Next按钮,向导的转换未正确设置。

2 个答案:

答案 0 :(得分:2)

这差不多晚了一年,但我想我弄清楚问题是什么。您致电setFinalPage(True),只需在QWizardPage中设置状态标记即可。它不会自动传播回您的QWizard

只有一个信号传播信息:completeChanged。这个名称有点误导,但文档似乎表明,如果你读得恰到好处,它会做你想做的事情:

  

如果您重新实现isComplete(),请确保在completeChanged()的值发生更改时发出isComplete(),以确保QWizard更新其按钮的启用或禁用状态。

事实上,遇到同样的问题,我可以通过

来修复它
if source == 'Calibration file':
    self.setFinalPage(True)
    ...
else:
    self.setFinalPage(False)
    ...
self.completeChanged.emit()

粗线是新的。它需要在两种情况下都被调用,并且会从&#34; Next&#34;切换按钮。到&#34;完成&#34;并在选择不同选项时返回。

答案 1 :(得分:0)

在您的代码中,您已使用QWizardPage.setFinalPage(True)向中间页面添加完成按钮。现在下一个按钮仍然存在。删除它的一种方法(不确定这是否是最好的方法)是通过与QWizard.removePage()一起调用QWizard.nextId()来删除所有后续页面。

示例:

from PyQt5.QtWidgets import *

def end_wizard_after_page_two():
    # add finish button to wizard
    p2.setFinalPage(True)
    # remove all over buttons
    while True:
        id = w.nextId()
        if id == -1:
            break
        w.removePage(id)

app = QApplication([])

# page 1
p1 = QWizardPage()
p1.setTitle('Page 1')

# page 2
p2 = QWizardPage()
p2.setTitle('Page 2')

b = QPushButton('No further pages')
b.clicked.connect(end_wizard_after_page_two)
l = QVBoxLayout(p2)
l.addWidget(b)

# page 3
p3 = QWizardPage()
p3.setTitle('Page 3')

# wizard
w = QWizard()
w.addPage(p1)
w.addPage(p2)
w.addPage(p3)
w.show()

app.exec_()

请参阅此示例中的方法end_wizard_after_page_two()

如果要反转效果,则必须反向执行所有操作(再次添加剩余页面并将setFinalPage添加为False)。