Подкласс qplaintext не расширяется, чтобы заполнить макет

Я не понимаю, почему пример CodeEditor с веб-сайта Qt не работает должным образом. Каждый раз, когда я запускаю код, он отображает его вот так, очень маленький и не расширяющийся, чтобы занять все доступное пространство. Кто-нибудь знает, почему? Я даже попытался установить фиксированный размер, размерную политику и минимальный размер. Не уверен, что мне здесь не хватает.

enter image description here

main.cpp:

#include "mainwindow.h"
#include <QApplication>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    MainWindow w;
    w.show();

    return a.exec();
}

mainwindow.cpp:

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QVBoxLayout>
#include <QLabel>
#include "codeeditor.h"

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent)
{
    // controls
    auto *widget_main = new QWidget(this);
    auto *lay_main = new QVBoxLayout(widget_main);
    auto *label = new QLabel("Test");
    auto *editor = new CodeEditor();

    // layout
    lay_main->addWidget(label);
    lay_main->addWidget(editor);
    lay_main->setContentsMargins(5, 5, 5, 5);
}

MainWindow::~MainWindow()
{
}

codeeditor.h:

#ifndef CODEEDITOR_H
#define CODEEDITOR_H

#include <QPlainTextEdit>
#include <QObject>

class QPaintEvent;
class QResizeEvent;
class QSize;
class QWidget;

class LineNumberArea;

// Main text editor
class CodeEditor : public QPlainTextEdit
{
    Q_OBJECT

    public:
        CodeEditor(QWidget *parent = 0);

        void lineNumberAreaPaintEvent(QPaintEvent *event);
        int lineNumberAreaWidth();

    protected:
        void resizeEvent(QResizeEvent *event) override;

    private slots:
        void updateLineNumberAreaWidth(int newBlockCount);
        void highlightCurrentLine();
        void updateLineNumberArea(const QRect &, int);

    private:
        QWidget *lineNumberArea;
};

// Line number gutter
class LineNumberArea : public QWidget
{
    public:
        LineNumberArea(CodeEditor *editor) : QWidget(editor) {
            codeEditor = editor;
        }

        QSize sizeHint() const override {
            return QSize(codeEditor->lineNumberAreaWidth(), 0);
        }

    protected:
        void paintEvent(QPaintEvent *event) override {
            codeEditor->lineNumberAreaPaintEvent(event);
        }

    private:
        CodeEditor *codeEditor;
};

#endif

codeeditor.cpp:

#include "codeeditor.h"
#include <QtWidgets>
#include <QFontMetrics>


CodeEditor::CodeEditor(QWidget *parent) : QPlainTextEdit(parent)
{
    setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);

    lineNumberArea = new LineNumberArea(this);

    connect(this, SIGNAL(blockCountChanged(int)), this, SLOT(updateLineNumberAreaWidth(int)));
    connect(this, SIGNAL(updateRequest(QRect,int)), this, SLOT(updateLineNumberArea(QRect,int)));
    connect(this, SIGNAL(cursorPositionChanged()), this, SLOT(highlightCurrentLine()));

    updateLineNumberAreaWidth(0);
    highlightCurrentLine();
    setFixedSize(200, 200);
    setMinimumSize(200,200);
}


int CodeEditor::lineNumberAreaWidth()
{
    int digits = 1;
    int max = qMax(1, blockCount());
    while (max >= 10) {
        max /= 10;
        ++digits;
    }

    //int space = 3 + fontMetrics().horizontalAdvance(QLatin1Char('9')) * digits;
    int space = 3 + 12 * digits;

    return space;
}


void CodeEditor::updateLineNumberAreaWidth(int /* newBlockCount */)
{
    setViewportMargins(lineNumberAreaWidth(), 0, 0, 0);
}


void CodeEditor::updateLineNumberArea(const QRect &rect, int dy)
{
    if (dy)
        lineNumberArea->scroll(0, dy);
    else
        lineNumberArea->update(0, rect.y(), lineNumberArea->width(), rect.height());

    if (rect.contains(viewport()->rect()))
        updateLineNumberAreaWidth(0);
}


void CodeEditor::resizeEvent(QResizeEvent *e)
{
    QPlainTextEdit::resizeEvent(e);

    QRect cr = contentsRect();
    lineNumberArea->setGeometry(QRect(cr.left(), cr.top(), lineNumberAreaWidth(), cr.height()));
}


void CodeEditor::highlightCurrentLine()
{
    QList<QTextEdit::ExtraSelection> extraSelections;

    if (!isReadOnly()) {
        QTextEdit::ExtraSelection selection;

        QColor lineColor = QColor(Qt::yellow).lighter(160);

        selection.format.setBackground(lineColor);
        selection.format.setProperty(QTextFormat::FullWidthSelection, true);
        selection.cursor = textCursor();
        selection.cursor.clearSelection();
        extraSelections.append(selection);
    }

    setExtraSelections(extraSelections);
}


void CodeEditor::lineNumberAreaPaintEvent(QPaintEvent *event)
{
    QPainter painter(lineNumberArea);
    painter.fillRect(event->rect(), Qt::lightGray);


    QTextBlock block = firstVisibleBlock();
    int blockNumber = block.blockNumber();
    int top = (int) blockBoundingGeometry(block).translated(contentOffset()).top();
    int bottom = top + (int) blockBoundingRect(block).height();

    while (block.isValid() && top <= event->rect().bottom()) {
        if (block.isVisible() && bottom >= event->rect().top()) {
            QString number = QString::number(blockNumber + 1);
            painter.setPen(Qt::black);
            painter.drawText(0, top, lineNumberArea->width(), fontMetrics().height(),
                             Qt::AlignRight, number);
        }

        block = block.next();
        top = bottom;
        bottom = top + (int) blockBoundingRect(block).height();
        ++blockNumber;
    }
}

Пример CodeEditor работает должным образом.

scopchanov 13.09.2018 21:11

И в вашем коде добавьте setCentralWidget(widget_main); в конец конструктора MainWindow.

scopchanov 13.09.2018 21:13
2
2
68
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

QMainWindow - особый виджет, так как имеет предустановленную раскладку.

enter image description here

Значит надо через setCentralWidget() установить главный виджет:

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
{
    auto *widget_main = new QWidget;
    auto *lay_main = new QVBoxLayout(widget_main);
    auto *label = new QLabel("Test");
    auto *editor = new CodeEditor();

    setCentralWidget(widget_main); // <-- +++

    // layout
    lay_main->addWidget(label);
    lay_main->addWidget(editor);
    lay_main->setContentsMargins(5, 5, 5, 5);
}

С другой стороны, если вы собираетесь использовать макеты, вам не следует устанавливать фиксированный размер для виджета, в вашем случае удалите setFixedSize(200, 200), с другой стороны, рекомендуется устанавливать соединения с новым синтаксисом:

CodeEditor::CodeEditor(QWidget *parent) : QPlainTextEdit(parent)
{
    setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);

    lineNumberArea = new LineNumberArea(this);

    connect(this, &QPlainTextEdit::blockCountChanged, this, &CodeEditor::updateLineNumberAreaWidth);
    connect(this, &QPlainTextEdit::updateRequest, this, &CodeEditor::updateLineNumberArea);
    connect(this, &QPlainTextEdit::cursorPositionChanged, this, &CodeEditor::highlightCurrentLine);

    updateLineNumberAreaWidth(0);
    highlightCurrentLine();
    // setFixedSize(200, 200); <-- ---
    setMinimumSize(200,200);
}

enter image description here

Обыграй меня за полминуты :)

scopchanov 13.09.2018 21:17

@scopchanov Это обычно бывает, со мной было :-)

eyllanesc 13.09.2018 21:19

Без проблем. Ваш ответ более исчерпывающий. Я удалил свой. Важно то, что проблема идентифицирована, а OP сохранен. :)

scopchanov 13.09.2018 21:20

спасибо за обновления. похоже, что там образцы кода в Интернете используют старый синтаксис для сигналов. Я внес изменения и удалил фиксированный размер. единственная причина, по которой у меня это было, - это попытаться выяснить, почему он не отображается правильно.

JokerMartini 13.09.2018 23:28

Другие вопросы по теме