QDockWidget не изменяет размер должным образом при взаимодействии с командами GUI QMainWindow - PullRequest
0 голосов
/ 12 марта 2019

У меня проблема с неправильным изменением размера QDockWidget. В частности, когда я запускаю графический интерфейс, QDockWidget выглядит как на изображении ниже, что неправильно . Также я настраиваю размер QDockWidget во время использования .ui, однако, как только я взаимодействую с .ui (например, используя QPushButton или QCheckBox), QDockWidget снова становится больше:

GUI_not_resizing

Ожидаемое поведение *1014* является приведенным ниже, которое не увеличивает размер внезапно во время взаимодействия с .ui, а скорее остается в положении, как показано ниже:

GUI_correctg

Ниже приведена наиболее важная часть кода, который я использую для этого проекта, и я подписал 3 ошибки отладки, уведомленные компилятором, с // <-- ERROR HERE, если это может быть полезно:

mainwindow.cpp

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    mDockWidget_A = new QDockWidget(QLatin1String("Command Log"));
    mDockWidget_A->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding);
    mNewText = new QPlainTextEdit;
    mNewText->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
    mDockWidget_A->setWidget(mNewText);

    mDockWidget_A->installEventFilter(new QDockResizeEventFilter(mNewText,dynamic_cast<QFluidGridLayout*>(mNewText->layout())));
    addDockWidget(Qt::BottomDockWidgetArea, mDockWidget_A);
}

qdockresizeeventfilter.h

#include <QObject>
#include <QLayout>
#include <QEvent>
#include <QDockWidget>
#include <QResizeEvent>
#include <QCoreApplication>
#include <QMouseEvent>
#include "qfluidgridlayout.h"
#include "mainwindow.h"

class QDockResizeEventFilter : public QObject
{

public:
    friend QMainWindow;
    friend QLayoutPrivate;
    QDockResizeEventFilter(QWidget* dockChild, QFluidGridLayout* layout, QObject* parent = nullptr)
        : QObject(parent), m_dockChild(dockChild), m_layout(layout)
    {

    }

protected:

    bool eventFilter(QObject *p_obj, QEvent *p_event)
    {  
        if (p_event->type() == QEvent::Resize)
        {
            QResizeEvent* resizeEvent   = static_cast<QResizeEvent*>(p_event);
            QMainWindow* mainWindow     = dynamic_cast<QMainWindow*>(p_obj->parent());              
            QDockWidget* dock           = static_cast<QDockWidget*>(p_obj);

            // determine resize direction
            if (resizeEvent->oldSize().height() != resizeEvent->size().height())
            {
                // vertical expansion
                QSize fixedSize(m_layout->widthForHeight(m_dockChild->size().height()), m_dockChild->size().height()); // <-- ERROR HERE
                if (dock->size().width() != fixedSize.width())
                {
                    m_dockChild->setFixedWidth(fixedSize.width());
                    dock->setFixedWidth(fixedSize.width());

                    // cause mainWindow dock layout recalculation
                    QDockWidget* dummy = new QDockWidget;
                    mainWindow->addDockWidget(Qt::BottomDockWidgetArea, dummy);
                    mainWindow->removeDockWidget(dummy);

                    // adding dock widgets causes the separator move event to end
                    // restart it by synthesizing a mouse press event
                    QPoint mousePos = mainWindow->mapFromGlobal(QCursor::pos());
                    mousePos.setY(dock->rect().bottom());
                    QCursor::setPos(mainWindow->mapToGlobal(mousePos));
                    QMouseEvent* grabSeparatorEvent = new QMouseEvent(QMouseEvent::MouseButtonPress,mousePos,Qt::LeftButton,Qt::LeftButton,Qt::NoModifier);
                    qApp->postEvent(mainWindow, grabSeparatorEvent);
                }
            }
            if (resizeEvent->oldSize().width() != resizeEvent->size().width())
            {
                // Do nothing
            }           
        }   
        return false;
    }

private:

    QWidget* m_dockChild;
    QFluidGridLayout* m_layout;
};

#endif // QDockResizeEventFilter_h_

и наконец qfluidgridlayout.h

#ifndef QFluidGridLayout_h_
#define QFluidGridLayout_h_

#include <QLayout>
#include <QGridLayout>
#include <QRect>
#include <QStyle>
#include <QWidgetItem>

class QFluidGridLayout : public QLayout
{
public:

    enum Direction { downToUp, UpToDown };
    QFluidGridLayout(QWidget *parent = nullptr)
        : QLayout(parent)
    {
        setContentsMargins(8,8,8,8);
        setSizeConstraint(QLayout::SetMinAndMaxSize);
    }

    ~QFluidGridLayout() {
        QLayoutItem *item;
        while ((item = takeAt(0)))
            delete item;
    }

    void addItem(QLayoutItem *item) {
        itemList.append(item);
    }

    Qt::Orientations expandingDirections() const {
        return nullptr;
    }

    bool hasHeightForWidth() const {
        return false;
    }

    int heightForWidth(int width) const {
        int height = doLayout(QRect(0, 0, width, 0), true, true);
        return height;
    }

    bool hasWidthForHeight() const {
        return true;
    }

    int widthForHeight(int height) const {  // <-- ERROR HERE
        int width = doLayout(QRect(0, 0, 0, height), true, false);
        return width;
    }

    int count() const {
        return itemList.size();
    }

    QLayoutItem *itemAt(int index) const {
        return itemList.value(index);
    }

    QSize minimumSize() const {
        QSize size;
        QLayoutItem *item;
        foreach (item, itemList)
            size = size.expandedTo(item->minimumSize());
        size += QSize(2*margin(), 2*margin());
        return size;
    }

    void setGeometry(const QRect &rect) {
        QLayout::setGeometry(rect);
        doLayout(rect); 
    }

    QSize sizeHint() const {
        return minimumSize();
    }

    QLayoutItem *takeAt(int index) {
        if (index >= 0 && index < itemList.size())
            return itemList.takeAt(index);
        else
            return nullptr; }
private:
    int doLayout(const QRect &rect, bool testOnly = false, bool width = false) const
    {
        int left, top, right, bottom;
        getContentsMargins(&left, &top, &right, &bottom); // <-- ERROR HERE
        QRect effectiveRect = rect.adjusted(+left, +top, -right, -bottom);
        int x = effectiveRect.x();
        int y = effectiveRect.y();
        int lineHeight = 0;
        int lineWidth = 0;
        QLayoutItem* item;
        foreach(item,itemList)
        {
            QWidget* widget = item->widget();   
            if (y + item->sizeHint().height() > effectiveRect.bottom() && lineWidth > 0) {
                y = effectiveRect.y();
                x += lineWidth + right;
                lineWidth = 0;
            }
            if (!testOnly) {
                item->setGeometry(QRect(QPoint(x, y), item->sizeHint()));
            }
            y += item->sizeHint().height() + top;
            lineHeight = qMax(lineHeight, item->sizeHint().height());
            lineWidth = qMax(lineWidth, item->sizeHint().width());
        } 
        if (width) {
            return y + lineHeight - rect.y() + bottom;
        }
        else {
            return x + lineWidth - rect.x() + right;
        }
    }
    QList<QLayoutItem *> itemList;
    Direction dir;
};

#endif // QFluidGridLayout_h_

Я часто читал об этой проблеме здесь и в этом посте . Однако я читал о возможности того, что этот конкретный объект может иметь некоторые ошибки , и было рекомендовано перезаписать resiveEvent. Однако ничего из этого не сработало.

Я, наконец, после большого количества исследований нашел этот полезный пост , который почти повторяет мою проблему и который несет большинство из двух классов выше class QFluidGridLayout и class QDockResizeEventFilter.

Хотя я использую тот же подход, я все еще не могу добиться нормального поведения этого объекта.

Я также включаю снимок отладчика:

debugger

Может кто-нибудь объяснить, что я делаю не так? Большое спасибо за то, что пролили свет на этот вопрос.

1 Ответ

0 голосов
/ 16 марта 2019

@ Emanuele, post , который вы видели, он в основном предназначен для подкласса QDockWidget как ребенка, и поэтому это решение должно было быть реализовано вручную.Я думаю, что если вы посмотрите на это альтернативное решение , вы найдете его полезным.

Попробуйте изменить конструктор, добавив resizeDocks({dock}, {100}, Qt::Horizontal);, как в сообщении, чтобы иметь:

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    mDockWidget_A = new QDockWidget(QLatin1String("Command Log"));
    mDockWidget_A->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding);
    mNewText = new QPlainTextEdit;
    mNewText->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
    mDockWidget_A->setWidget(mNewText);
    mDockWidget_A->installEventFilter(new QDockResizeEventFilter(mNewText,dynamic_cast<QFluidGridLayout*>(mNewText->layout())));
    addDockWidget(Qt::BottomDockWidgetArea, mDockWidget_A);

    resizeDocks({mDockWidget_A}, {100}, Qt::Horizontal);

}
...