Невозможно динамически сбросить максимальную высоту виджетов в QSplitter

#python #pyside

#python #pyside

Вопрос:

Я бы хотел, чтобы QSplitterHandle изменить размер только виджета над ним, путем:

  1. Изначально фиксируя максимальную и минимальную высоту всех виджетов ( QFrame фактически) в QSplitter .
  2. При 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. Политики размера, похоже, не работают. Я не могу найти политику размера, которая сама по себе предотвращает изменение размера виджета.