首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >如何在放大QGraphicsView时保持QPen像素宽度不变

如何在放大QGraphicsView时保持QPen像素宽度不变
EN

Stack Overflow用户
提问于 2013-06-13 05:24:56
回答 2查看 4.9K关注 0票数 9

我写了一个快速而令人讨厌的程序来帮助我可视化我正在工作的项目的一个方面。尽管我从4.1.1开始就一直在使用Qt,但我从来没有真正需要使用QGraphics*模块。

当我开始使用这个程序时,我正在一台运行Windows XP、Qt4.7.0和Visual Studio2008的旧计算机上工作。现在我已经把文件复制到运行Windows8的主计算机上了,我决定试一试Qt5,所以我安装了Qt5.0.2的QtCreator。

当我编译与我在Qt4.7.0/XP机器上创建的完全相同的代码时,我得到了一个截然不同的结果。

这是我在Qt4.7.0编译中看到的:

这是我在Qt5.0.2编译中看到的:

显然,在绘制每个矩形的边框时会有不同的行为。此外,如果我使用鼠标滚轮放大,矩形边框宽度在Qt5编译中变大,但在Qt4.7编译中保持不变(大约1像素宽)。

如何更改代码,使Qt5中的行为与Qt4.7中的行为相同?

下面是完整的代码:

代码语言:javascript
复制
#include <QApplication>
#include <QGraphicsScene>
#include <QGraphicsView>
#include <QGraphicsRectItem>
#include <QWheelEvent>

#include <QSqlDatabase>
#include <QSqlQuery>
#include <QSqlRecord>
#include <QVariant>
#include <QSqlError>
#include <QMessageBox>

class ItemData : public QSqlRecord{
public:
    ItemData(QSqlRecord const &rec) : QSqlRecord(rec) {}

    qreal left() const { return value(0).toDouble(); }
    qreal top() const { return value(1).toDouble(); }
    qreal width() const { return value(2).toDouble() - left(); }
    qreal height() const { return value(3).toDouble() - top(); }
    QRectF rect() const { return QRectF(left(), top(), width(), height()); }
    QString barcode() const { return value(4).toString(); }
    QString msaName() const { return value(5).toString(); }
    QString msaDescription() const { return value(6).toString(); }
    QString hsaName() const { return value(7).toString(); }
    QString hsaDescription() const { return value(8).toString(); }
};

class DSAItem : public QGraphicsRectItem{
public:
    DSAItem(ItemData const &data, QGraphicsItem *parent = 0)
        :QGraphicsRectItem(parent) {
            setFlags(QGraphicsItem::ItemIsSelectable);
            setRect(data.rect());
            QString tip = "<p><b>%1</b></p><p><b>MLSA</b><br/>%2<br/>%3</p><p><b>HLSA</b><br/>%4<br/>%5</p>";
            setToolTip(tip.arg(data.barcode(), data.msaName(), data.msaDescription(), data.hsaName(), data.hsaDescription()));
            if(data.barcode() == "1010100101" || data.barcode() == "1010100114"){
                colour = QColor(Qt::red);
            } else {
                colour = QColor(Qt::yellow);
            }
            colour.setAlphaF(.5);
            setBrush(QBrush(colour));
    }

    QVariant itemChange(GraphicsItemChange change, QVariant const &value){
        if (change == QGraphicsItem::ItemSelectedHasChanged){
            QColor c = value.toBool() ? QColor(Qt::green) : colour;
            c.setAlphaF(.5);
            setBrush(QBrush(c));
            update(rect());

        }
        return QGraphicsRectItem::itemChange(change, value);
    }

private:
    QColor colour;
};

class View : public QGraphicsView {
public:
    View(QWidget *parent = 0): QGraphicsView(parent){
        populateScene();
        setScene(&scene);
        rotate(-90);
        scale(1.0, -1.0);
        setDragMode(ScrollHandDrag);
    }

protected:
    void wheelEvent(QWheelEvent *e){
        setTransformationAnchor(QGraphicsView::AnchorUnderMouse);

        double scaleFactor = 1.15;
        if (e->delta() > 0){
            scale(scaleFactor, scaleFactor);
        } else {
            scale(1 / scaleFactor, 1 / scaleFactor);
        }
    }

private:
    void populateScene(){
        QSqlDatabase db = QSqlDatabase::addDatabase("QODBC");
        //db credentials here

        QString errorMessage;
        bool ok = db.open();
        if (ok){
            QSqlQuery query(db);
            QString sql = //query string here

            if (query.exec(sql)){
                while(query.next()){
                    scene.addItem(new DSAItem(query.record()));
                }
            } else {
                errorMessage = query.lastError().text();
            }
        } else {
            errorMessage = db.lastError().text();
        }

        if (!errorMessage.isEmpty()){
            QMessageBox::critical(0, "Database Error", errorMessage);
        }
    }

private:
    QGraphicsScene scene;
};

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

    return a.exec();
}
EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2013-06-13 05:39:29

下面的rpsml给出了最好的答案。但由于历史原因,我将把这个问题留在这里。

将笔的宽度设置为0将使其成为宽度为1的“美容”笔。

代码语言:javascript
复制
QPen p = pen();
p.setWidth(0)
setPen(p);
票数 5
EN

Stack Overflow用户

发布于 2014-04-20 03:06:42

这有点晚了,但可能会有所帮助。

提出的解决方案运行良好,但限制了笔的宽度。另一种方法是使用指令

代码语言:javascript
复制
p.setCosmetic(true);

根据Qt参考(http://qt-project.org/doc/qt-5/qpen.html#isCosmetic)

修饰笔用于绘制具有恒定宽度的笔触,而不管应用于它们所使用的QPainter的任何变换。

它还解释了将宽度设置为零的原因:

缺省情况下,零宽度钢笔是修饰的;宽度非零的钢笔是非修饰的。

票数 31
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/17075623

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档