gpt4 book ai didi

c++ - 当 QAction 处于领先位置时如何获得 QLineEdit 文本偏移

转载 作者:搜寻专家 更新时间:2023-10-31 02:04:05 24 4
gpt4 key购买 nike

我有一个 QLineEditQAction处于领先地位。我想知道文本的起始位置,但我不知道该怎么做:

QLineEdit *le = new QLineEdit(parent);
le->addAction(QIcon(":/myicon"), QLineEdit::LeadingPosition);
// Now I want to get the text start position
// but both return "QMargins(0, 0, 0, 0) QMargins(0, 0, 0, 0)"
qDebug() << le->textMargins() << le->contentsMargins();

我在 qt 的 github 源代码中搜索以查找是否 addAction()方法在内容或文本边距上做了一些事情但没有成功。

最佳答案

我必须承认(在阅读 OP 问题之前)我不知道 QLineEdit::addAction() 。因此,我写了一个小示例 testQLineEditAction.cc:

#include <QtWidgets>

int main(int argc, char **argv)
{
qDebug() << "Qt Version:" << QT_VERSION_STR;
QApplication app(argc, argv);
// init GUI
QLineEdit qEdit;
qEdit.addAction(QIcon("./document-properties.svg"), QLineEdit::LeadingPosition);
qEdit.addAction(QIcon("./document-save.svg"), QLineEdit::TrailingPosition);
qEdit.show();
// runtime loop
return app.exec();
}

这是它的样子(在 cygwin64 中编译):

Snapshot of testQLineEditAction

之后,我深入了解了 woboq.org 以了解它是如何实现的。

我从 QLineEdit::paintEvent() 开始:

void QLineEdit::paintEvent(QPaintEvent *)
{

...

  QStyleOptionFrame panel;
initStyleOption(&panel);

...

  QRect r = style()->subElementRect(QStyle::SE_LineEditContents, &panel, this);
r.setX(r.x() + d->effectiveLeftTextMargin());
r.setY(r.y() + d->topTextMargin);
r.setRight(r.right() - d->effectiveRightTextMargin());
r.setBottom(r.bottom() - d->bottomTextMargin);

这很有趣:检索内容的矩形,然后通过内部偏移量进行校正。

  QFontMetrics fm = fontMetrics();

...

  QRect lineRect(r.x() + d->horizontalMargin, d->vscroll, r.width() - 2*d->horizontalMargin, fm.height());

关于 d->horizo​​ntalMargin,我不是很确定,但我暂时忽略了它并遵循了 d->effectiveLeftTextMargin() :

int QLineEditPrivate::effectiveLeftTextMargin() const
{
return effectiveTextMargin(leftTextMargin, leftSideWidgetList(), sideWidgetParameters());
}

...

static int effectiveTextMargin(int defaultMargin, const QLineEditPrivate::SideWidgetEntryList &widgets,
const QLineEditPrivate::SideWidgetParameters &parameters)
{
if (widgets.empty())
return defaultMargin;
return defaultMargin + (parameters.margin + parameters.widgetWidth) *
int(std::count_if(widgets.begin(), widgets.end(),
[](const QLineEditPrivate::SideWidgetEntry &e) {
return e.widget->isVisibleTo(e.widget->parentWidget()); }));
}

因此,我得出结论,当确定文本矩形的有效大小时,QLineEditPrivate::effectiveLeftTextMargin() 会考虑操作图标的空间。

遗憾的是,所有这些函数都是private,因此无法从外部访问。在考虑了一段时间如何从外部访问这些并查看文档之后。无论我是否没有监督某些事情,我都想到了直接为此使用 QAction:

#include <QtWidgets>

void inspect(const QString &cmd, QAction &qCmd)
{
qDebug() << (cmd + "->associatedWidgets().size():")
<< qCmd.associatedWidgets().size();
int i = 0;
for (QWidget *const pQWidget : qCmd.associatedWidgets()) {
qDebug() << '[' << i++ << "]:"
<< typeid(*pQWidget).name()
<< "geometry:" << pQWidget->geometry();
}
}

int main(int argc, char **argv)
{
qDebug() << "Qt Version:" << QT_VERSION_STR;
QApplication app(argc, argv);
// init GUI
QLineEdit qEdit;
qEdit.setText("012345678901234567890123456789");
QAction *const pQCmd1
= qEdit.addAction(QIcon("./document-properties.svg"), QLineEdit::LeadingPosition);
QAction *const pQCmd2
= qEdit.addAction(QIcon("./document-save.svg"), QLineEdit::TrailingPosition);
qEdit.show();
qDebug() << "qEdit.geometry():" << qEdit.geometry();
inspect("pQCmd1", *pQCmd1);
inspect("pQCmd2", *pQCmd2);
// runtime loop
return app.exec();
}

控制台输出:

Qt Version: 5.9.4
qEdit.geometry(): QRect(0,0 200x23)
"pQCmd1->associatedWidgets().size():" 2
[ 0 ]: 9QLineEdit geometry: QRect(0,0 200x23)
[ 1 ]: 19QLineEditIconButton geometry: QRect(4,2 22x18)
"pQCmd2->associatedWidgets().size():" 2
[ 0 ]: 9QLineEdit geometry: QRect(0,0 200x23)
[ 1 ]: 19QLineEditIconButton geometry: QRect(174,2 22x18)

为了比较这些值,另一个带有修改图标的快照(在 SVG 中绘制的框架以显示图标大小)已被放大(因子 5):

Magnified Snapshot of testQLineEditAction (factor 5)

QLineEditIconButton 报告位置 (4, 2) 但图标的左框架距离 QLineEdit 的左边框 8 个像素。 QLineEditIconButton 周围肯定有一个框架,也必须考虑它(我没有研究如何检索它)。框架的宽度可能受样式引擎的影响,因此因平台而异。为了使这种尝试健壮且可移植,应从小部件或样式中检索相应的值。这开始成为一个或多或少成功机会的乏味摆弄。

我在尝试回答 SO: How to automatically increase/decrease text size in label in Qt 时遇到过一次类似的情况。


关于 QLineEdit::cursorRect() :

我相信使用 QLineEdit::cursorRect() 充其量也是脆弱的。

我修改了上面的示例以检查这一点:

#include <QtWidgets>

class LineEdit: public QLineEdit {

public:
QRect cursorRect() const { return QLineEdit::cursorRect(); }
};

int main(int argc, char **argv)
{
qDebug() << "Qt Version:" << QT_VERSION_STR;
QApplication app(argc, argv);
// init GUI
LineEdit qEdit;
qEdit.setText("012345678901234567890123456789");
qEdit.addAction(QIcon("./document-properties.svg"), QLineEdit::LeadingPosition);
qEdit.addAction(QIcon("./document-save.svg"), QLineEdit::TrailingPosition);
qEdit.show();
qDebug() << "qEdit.cursorRect():" << qEdit.cursorRect();
// runtime loop
return app.exec();
}

控制台输出:

Qt Version: 5.9.4
qEdit.geometry(): QRect(0,0 200x23)
qEdit.cursorRect(): QRect(253,0 9x16)

有趣的是,光标的 x 位置不仅很高,甚至比 qEdit 的宽度还高。怎么会?我放入 qEdit 的初始文本 "012345678901234567890123456789" 导致光标靠近右侧,从而发生水平滚动。光标位置似乎与虚拟文本宽度有关(包括左侧的裁剪范围)。

关于c++ - 当 QAction 处于领先位置时如何获得 QLineEdit 文本偏移,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54213701/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com