Как правильно создать подкласс QWidget для построения линейки на QPlainTextEdit - PullRequest
0 голосов
/ 15 июня 2019

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

В вертикальной части я могу видеть все числа при изменении линии, но не вижу горизонтальной линии сверху линейки. То, что я пытался показать в качестве примера числами от 1 до 30, но на горизонтальной линейке ничего не появляется. После этого урока я вложил в подкласс QWidget и создал производную class LineNumberTop, где я пытался реализовать оставшуюся часть. Если вы скопируете и вставите небольшой проект, он будет скомпилирован и запущен.

Однако во время реализации мне удалось получить рабочую вертикальную сторону, но не рабочую горизонтальную сторону. Как видно из экрана печати ниже:

exampleRuler

Я думал, что подклассификация в другой раз QWidget будет хорошей идеей, так как это вопрос создания дополнительного класса и повторного использования функций для class LineNumberArea : public QWidget. Но я на этом не остановился и нашел этот дополнительный источник, который был полезен для понимания того, как устанавливать измерения линейки. Однако другой момент, в котором я не совсем уверен на сто процентов, заключается в элементе private конструктора, где я устанавливаю в два раза больше QWidget, и я не уверен, что один перезаписывает другой:

private:
    QWidget *lineNumberArea;
    QWidget *lineNumberAreaTop;

Полный код, который я использую для этого упражнения, следующий:

codeeditor.h

#ifndef CODEEDITOR_H
#define CODEEDITOR_H
#include <QPlainTextEdit>
#include <QObject>
class QPaintEvent;
class QResizeEvent;
class QSize;
class QWidget;
class LineNumberArea;
class LineNumberTop;

class CodeEditor : public QPlainTextEdit
{
    Q_OBJECT

public:
    CodeEditor(QWidget *parent = nullptr);
    void lineNumberAreaPaintEvent(QPaintEvent *event);
    int lineNumberAreaWidth();

    void lineNumberAreaPaintEventTop(QPaintEvent *event);
    int lineNumberAreaWidthTop();

protected:
    void resizeEvent(QResizeEvent *event) override;

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

private:
    QWidget *lineNumberArea;
    QWidget *lineNumberAreaTop;
};

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;
};

class LineNumberTop : public QWidget
{
public:
    LineNumberTop(CodeEditor *editor) : QWidget(editor) {
        codeEditor = editor;
    }
    QSize sizeHint() const override {
        return QSize(codeEditor->lineNumberAreaWidthTop(), 0);
    }
protected:
    void paintEvent(QPaintEvent *event) override {
        codeEditor->lineNumberAreaPaintEventTop(event);
    }
private:
    CodeEditor *codeEditor;
};
#endif

codeeditor.cpp

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

CodeEditor::CodeEditor(QWidget *parent) : QPlainTextEdit(parent)
{
    lineNumberArea = new LineNumberArea(this);
    lineNumberAreaTop = new LineNumberTop(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();
}

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;
    return space;
}

void CodeEditor::lineNumberAreaPaintEventTop(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();
    QString number = QString::number(blockNumber + 1);
    painter.setPen(Qt::black);
    painter.drawText(0, top, lineNumberAreaTop->width(), fontMetrics().height(),
                     Qt::AlignRight, number);
}

int CodeEditor::lineNumberAreaWidthTop()
{
    int digits = 1;
    int space = 3 + fontMetrics().horizontalAdvance(QLatin1Char('20')) * 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()));
    lineNumberAreaTop->setGeometry(QRect(cr.left(), cr.top(), lineNumberAreaWidthTop(), 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;
    }
}

main.cpp

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

int main(int argv, char **args)
{
    QApplication app(argv, args);
    CodeEditor editor;
    editor.setWindowTitle(QObject::tr("Code Editor Example"));
    editor.show();
    return app.exec();
}

Спасибо за указание в правильном направлении для решения этой проблемы.

...