我正在编写文本编辑器并使用 Qt 作为 GUI。我是 Qt 的菜鸟,我在做这件事时遇到了麻烦。
我需要在 QPlainTextEdit 的第 80 列画一条线,但我真的不知道该怎么做。我正在使用 QPainter,但我无法正确使用它,有什么帮助吗?
最佳答案
这是我的做法。诚然,这并非完全微不足道。确定第 80 列位置的输入是:
80 x 浮点平均字符宽度。使用整数值会将舍入误差放大 80 倍。因此请使用
QFontMetricsF
。滚动条的偏移量来自
contentOffset()
。使用horizontalScrollbar()->value()
会很糟糕。后者目前有效,但依赖于特定于实现的细节。 QPlainTextEdit 碰巧将滚动条值映射到像素——谁知道它明天是否会改变。它没有记录,因此属于未指定的行为。QTextDocument
实现了自己的边距,可通过documentMargin()
获得。
另一个陷阱:您必须在派生自 QAbstractScrollArea
的任何类中的 viewport()
上绘画——而 QPlainTextEdit
就是这样做的。如果不这样做,您的 paintEvent
将成为空操作。它已记录在案,但您必须足够聪明才能真正查看文档。我认为这是一个糟糕的 API 极端情况,会做一些意想不到的事情。在所有其他 paintEvent
中,您只需创建 QPainter p
或 QPainter p(this)
即可。
注意:这是经过测试的可编译代码。
//main.cpp
#include <cmath>
#include <QtWidgets>
class Edit : public QPlainTextEdit
{
public:
Edit(QWidget * parent = 0) : QPlainTextEdit(parent) {}
protected:
void paintEvent(QPaintEvent * ev)
{
QPlainTextEdit::paintEvent(ev);
const QRect rect = ev->rect();
const QFont font = currentCharFormat().font();
int x80 = round(QFontMetricsF(font).averageCharWidth() * 80.0)
+ contentOffset().x()
+ document()->documentMargin();
QPainter p(viewport());
p.setPen(QPen("gray"));
p.drawLine(x80, rect.top(), x80, rect.bottom());
qDebug() << x80 << contentOffset() << document()->documentMargin() << font << endl;
}
};
static QString filler()
{
QString str;
for (char c = '0'; c < '9'; ++ c) {
str.append(QString(10, c));
}
return str;
}
int main(int argc, char ** argv)
{
QApplication app(argc, argv);
Edit ed;
QTextCharFormat fmt = ed.currentCharFormat();
fmt.setFontFamily("courier");
fmt.setFontFixedPitch(true);
ed.setCurrentCharFormat(fmt);
ed.setLineWrapMode(QPlainTextEdit::NoWrap);
qDebug() << fmt.font() << endl;
ed.setPlainText(filler());
ed.show();
app.exec();
}
#include "main.moc"
关于c++ - 在 QPlainTextEdit 的第 80 列画一条线,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11019519/