添加属性 QtCore.Qt.FramelessWindowHint 后如何从边缘调整窗口大小

Posted

技术标签:

【中文标题】添加属性 QtCore.Qt.FramelessWindowHint 后如何从边缘调整窗口大小【英文标题】:How to resize a window from the edges after adding the property QtCore.Qt.FramelessWindowHint 【发布时间】:2020-07-09 04:12:47 【问题描述】:

晚安。 我已经看到一些具有新的无边框设计的程序,但您仍然可以使用调整大小。 目前我知道要删除我们使用的 pyqt 程序的边界: QtCore.Qt.FramelessWindowHint

要改变窗口的大小,请使用 QSizeGrip。 但是我们如何调整一个没有边框的窗口呢?

这是我用来删除窗口边框的代码,但之后我没有找到有关如何在 pyqt5 中执行此操作的信息。 希望你能帮我举个例子说明如何解决这个问题

from PyQt5.QtWidgets import QMainWindow,QApplication
from PyQt5 import  QtCore

class Main(QMainWindow):
    def __init__(self):
        QMainWindow.__init__(self)


        self.setWindowFlags(QtCore.Qt.FramelessWindowHint)


app = QApplication([])
m = Main()
m.show()
m.resize(800,600)
app.exec_()

【问题讨论】:

【参考方案1】:

如果您使用 QMainWindow,您可以通过调用 statusBar() 来添加 QStatusBar(它会自动添加 QSizeGrip):

如果状态栏不存在,此函数会创建并返回一个空的状态栏。

否则,您可以手动添加夹点,它们的交互会根据它们的位置自动完成。在下面的示例中,我添加了 4 个夹点,每个角一个,然后在每次调整窗口大小时移动它们。

class Main(QMainWindow):
    def __init__(self):
        QMainWindow.__init__(self)

        self.setWindowFlags(QtCore.Qt.FramelessWindowHint)
        self.gripSize = 16
        self.grips = []
        for i in range(4):
            grip = QSizeGrip(self)
            grip.resize(self.gripSize, self.gripSize)
            self.grips.append(grip)

    def resizeEvent(self, event):
        QMainWindow.resizeEvent(self, event)
        rect = self.rect()
        # top left grip doesn't need to be moved...
        # top right
        self.grips[1].move(rect.right() - self.gripSize, 0)
        # bottom right
        self.grips[2].move(
            rect.right() - self.gripSize, rect.bottom() - self.gripSize)
        # bottom left
        self.grips[3].move(0, rect.bottom() - self.gripSize)

更新

基于 cmets,还需要调整大小。为此,一个好的解决方案是创建一个行为类似于 QSizeGrip 的自定义小部件,但仅用于垂直/水平调整大小。

为了更好地实现,我更改了上面的代码,使用gripSize 构造一个“内部”矩形,并基于它更改所有小部件的几何形状,包括角和边。

在这里您可以看到用于几何计算的“外部”矩形和“内部”矩形:

然后您可以为 QSizeGrip 小部件创建所有几何图形(浅蓝色):

对于自定义侧面小部件:

from PyQt5 import QtCore, QtGui, QtWidgets

class SideGrip(QtWidgets.QWidget):
    def __init__(self, parent, edge):
        QtWidgets.QWidget.__init__(self, parent)
        if edge == QtCore.Qt.LeftEdge:
            self.setCursor(QtCore.Qt.SizeHorCursor)
            self.resizeFunc = self.resizeLeft
        elif edge == QtCore.Qt.TopEdge:
            self.setCursor(QtCore.Qt.SizeVerCursor)
            self.resizeFunc = self.resizeTop
        elif edge == QtCore.Qt.RightEdge:
            self.setCursor(QtCore.Qt.SizeHorCursor)
            self.resizeFunc = self.resizeRight
        else:
            self.setCursor(QtCore.Qt.SizeVerCursor)
            self.resizeFunc = self.resizeBottom
        self.mousePos = None

    def resizeLeft(self, delta):
        window = self.window()
        width = max(window.minimumWidth(), window.width() - delta.x())
        geo = window.geometry()
        geo.setLeft(geo.right() - width)
        window.setGeometry(geo)

    def resizeTop(self, delta):
        window = self.window()
        height = max(window.minimumHeight(), window.height() - delta.y())
        geo = window.geometry()
        geo.setTop(geo.bottom() - height)
        window.setGeometry(geo)

    def resizeRight(self, delta):
        window = self.window()
        width = max(window.minimumWidth(), window.width() + delta.x())
        window.resize(width, window.height())

    def resizeBottom(self, delta):
        window = self.window()
        height = max(window.minimumHeight(), window.height() + delta.y())
        window.resize(window.width(), height)

    def mousePressEvent(self, event):
        if event.button() == QtCore.Qt.LeftButton:
            self.mousePos = event.pos()

    def mouseMoveEvent(self, event):
        if self.mousePos is not None:
            delta = event.pos() - self.mousePos
            self.resizeFunc(delta)

    def mouseReleaseEvent(self, event):
        self.mousePos = None


class Main(QtWidgets.QMainWindow):
    _gripSize = 8
    def __init__(self):
        QtWidgets.QMainWindow.__init__(self)

        self.setWindowFlags(QtCore.Qt.FramelessWindowHint)

        self.sideGrips = [
            SideGrip(self, QtCore.Qt.LeftEdge), 
            SideGrip(self, QtCore.Qt.TopEdge), 
            SideGrip(self, QtCore.Qt.RightEdge), 
            SideGrip(self, QtCore.Qt.BottomEdge), 
        ]
        # corner grips should be "on top" of everything, otherwise the side grips
        # will take precedence on mouse events, so we are adding them *after*;
        # alternatively, widget.raise_() can be used
        self.cornerGrips = [QtWidgets.QSizeGrip(self) for i in range(4)]

    @property
    def gripSize(self):
        return self._gripSize

    def setGripSize(self, size):
        if size == self._gripSize:
            return
        self._gripSize = max(2, size)
        self.updateGrips()

    def updateGrips(self):
        self.setContentsMargins(*[self.gripSize] * 4)

        outRect = self.rect()
        # an "inner" rect used for reference to set the geometries of size grips
        inRect = outRect.adjusted(self.gripSize, self.gripSize,
            -self.gripSize, -self.gripSize)

        # top left
        self.cornerGrips[0].setGeometry(
            QtCore.QRect(outRect.topLeft(), inRect.topLeft()))
        # top right
        self.cornerGrips[1].setGeometry(
            QtCore.QRect(outRect.topRight(), inRect.topRight()).normalized())
        # bottom right
        self.cornerGrips[2].setGeometry(
            QtCore.QRect(inRect.bottomRight(), outRect.bottomRight()))
        # bottom left
        self.cornerGrips[3].setGeometry(
            QtCore.QRect(outRect.bottomLeft(), inRect.bottomLeft()).normalized())

        # left edge
        self.sideGrips[0].setGeometry(
            0, inRect.top(), self.gripSize, inRect.height())
        # top edge
        self.sideGrips[1].setGeometry(
            inRect.left(), 0, inRect.width(), self.gripSize)
        # right edge
        self.sideGrips[2].setGeometry(
            inRect.left() + inRect.width(), 
            inRect.top(), self.gripSize, inRect.height())
        # bottom edge
        self.sideGrips[3].setGeometry(
            self.gripSize, inRect.top() + inRect.height(), 
            inRect.width(), self.gripSize)

    def resizeEvent(self, event):
        QtWidgets.QMainWindow.resizeEvent(self, event)
        self.updateGrips()


app = QtWidgets.QApplication([])
m = Main()
m.show()
m.resize(240, 160)
app.exec_()

【讨论】:

那么,难道程序的所有边框都可以调整大小,而不仅仅是角落? 很棒的工作!如果要删除边框,只需将夹点小部件向前移动,以便它们高于所有其他小部件。这可以通过在 def updateGrips(self) --- [grip.raise_(​​) for grip in self.sideGrips + self._cornerGrips] 的末尾设置此代码来完成 --- 然后您可以删除 self.setContentsMargins(*[ self.gripSize] * 4) 从该函数开始。 你用什么软件来绘制可视化这些窗口? @MansoorAhmedMemon 你用三张图片来显示边缘吗?我真的不记得了,但我可能用过inkscape。

以上是关于添加属性 QtCore.Qt.FramelessWindowHint 后如何从边缘调整窗口大小的主要内容,如果未能解决你的问题,请参考以下文章

C#自定义控件中如何动态添加属性

jquery 怎么给标签添加属性

jsTree如何添加自定义属性

TS TypeScript window 添加自定义属性

OC为啥 Category不能添加属性

$message 添加属性