#python #pyside
#python #pyside
Вопрос:
Я бы хотел, чтобы QSplitterHandle
изменить размер только виджета над ним, путем:
- Изначально фиксируя максимальную и минимальную высоту всех виджетов (
QFrame
фактически) вQSplitter
. - При
QSplitterHandle::mousePressEvent
«увеличении» максимальной высоты и приQSplitterHandle::mouseReleaseEvent
сбросе максимальной высоты до текущей.
Код
#! /usr/bin/python
import sys
from PySide import QtGui
from PySide import QtCore
# Custom Splitter Handle
class SplitterHandle(QtGui.QSplitterHandle):
QWIDGET_MAX_HEIGHT = 1000
def __init__(self, index , orientation, parent):
super(SplitterHandle , self).__init__(orientation, parent)
self.index = index
def getWidget( self ):
return self.splitter().widget( self.index )
# Remove height restriction on corresponding widget when dragging starts
def mousePressEvent(self ,event ):
widget = self.getWidget()
widget.setMinimumHeight( 100 ) # FIX
widget.setMaximumHeight( self.QWIDGET_MAX_HEIGHT )
return QtGui.QSplitterHandle.mousePressEvent(self,event)
# Freeze height of corresponding widget after dragging has ends
def mouseReleaseEvent(self ,event ):
widget = self.getWidget()
widget.setMinimumHeight( widget.height() ) # FIX
widget.setMaximumHeight( widget.height() )
return QtGui.QSplitterHandle.mousePressEvent(self,event)
# Custom Splitter
class Splitter(QtGui.QSplitter):
def __init__( self , orientation , parent = None):
super(Splitter , self).__init__(orientation,parent)
self.orientation = orientation
def createHandle( self):
return SplitterHandle( self.count() , self.orientation , self )
class Example(QtGui.QWidget):
def __init__(self):
super(Example, self).__init__()
hbox = QtGui.QVBoxLayout(self)
frame1 = QtGui.QFrame()
frame1.setFrameShape(QtGui.QFrame.StyledPanel)
frame1.setMinimumHeight( 100 )
frame1.setMaximumHeight( 100 )
frame1.setStyleSheet("background-color: red;");
frame2 = QtGui.QFrame()
frame2.setFrameShape(QtGui.QFrame.StyledPanel)
frame2.setMinimumHeight( 100 )
frame2.setMaximumHeight( 100 )
frame2.setStyleSheet("background-color: green;");
frame3 = QtGui.QFrame()
frame3.setFrameShape(QtGui.QFrame.StyledPanel)
frame3.setMinimumHeight( 100 )
frame3.setMaximumHeight( 100 )
frame3.setStyleSheet("background-color: blue;");
frame4 = QtGui.QFrame()
frame4.setFrameShape(QtGui.QFrame.StyledPanel)
frame4.setMinimumHeight( 100 )
frame4.setMaximumHeight( 100 )
frame4.setStyleSheet("background-color: yellow;");
# The "absorber"
frame5 = QtGui.QFrame()
frame5.setStyleSheet("background-color: white;");
########################
splitter = Splitter(QtCore.Qt.Vertical)
splitter.addWidget(frame1)
splitter.addWidget(frame2)
splitter.addWidget(frame3)
splitter.addWidget(frame4)
splitter.addWidget(frame5)
splitter.setChildrenCollapsible( False )
hbox.addWidget(splitter)
self.setGeometry(300, 300, 300, 600)
self.setWindowTitle('--Splitter--')
self.show()
def main():
app = QtGui.QApplication(sys.argv)
ex = Example()
sys.exit(app.exec_())
if __name__ == '__main__':
main()
Смотрите скриншот приложения. Проблема в том, что попытки динамически сбросить высоту виджетов завершаются неудачей.
Ответ №1:
Проблема в том, что вы применяете максимальный размер к неправильному виджету, вызванному тем, что вы передаете неподходящий индекс. Решение состоит в том, чтобы передать его self.count()-1
:
def createHandle( self):
return SplitterHandle(self.count()-1, self.orientation , self)
Комментарии:
1. Спасибо. Я проверю это, когда вернусь домой. Как вы думаете, использование политик размера является лучшим решением или возможно? (Я попробую, когда вернусь домой.)
2. @Olumide На самом деле я не знаю.
3. Я думаю, это вопрос дизайна. Политики размера позволят неограниченно изменять размер виджета до сумасшедших пределов. Комбинация обоих может быть лучшим решением. Я вскоре опубликую свои результаты.
4. Политики размера, похоже, не работают. Я не могу найти политику размера, которая сама по себе предотвращает изменение размера виджета.