Как сохранить ширину пикселя QPen одинаковой при масштабировании в QGraphicsView

Я написал быструю и неприятную программу, чтобы помочь мне визуализировать аспект проекта, над которым я работаю. Хотя я работаю с Qt с 4.1.1, у меня никогда не было необходимости использовать модуль QGraphics*.

когда я начал играть с программой, я работал на старом компьютере под управлением Windows XP и Qt4.7.0 и Visual Studio 2008. Теперь я просто скопировал файл на свой главный компьютер под управлением Windows 8. Я решил дать Qt5 выстрел, поэтому я установил QtCreator с Qt5.0.2.

когда я компилирую тот же самый код, который я создал на Qt4.7.Машина 0 / XP, я получаю совершенно другой результат.

Это то, что я вижу на моем Qt4.7.0 компиляция:

Zoomed-out view from Qt4.7.0/XP compilation

и это то, что я вижу на моем Qt5.0.2 компиляция: Zoomed-out view from Qt5.0.2/Win8x64 compilation

очевидно, что в чертеже границы каждого прямоугольника есть разное поведение. Кроме того, если я увеличиваю масштаб с помощью колеса мыши, ширина границы прямоугольника становится больше компиляция Qt5, но остается той же (шириной около 1 пикселя) в Qt4.7 сборник.

Как изменить код, чтобы поведение в Qt5 было таким же, как в Qt4.7?

вот полный код:

#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();
}

2 ответов


лучший ответ дается rpsml ниже этого. Но по историческим причинам я оставлю это здесь.

установка ширины пера на ноль сделает его" косметическим " пером с шириной 1.

QPen p = pen();
p.setWidth(0)
setPen(p);

Это немного поздно, но может быть полезным.

предлагаемое решение работает нормально,но ограничивает ширину пера. Альтернативой является использование инструкции

p.setCosmetic(true);

согласно ссылке Qt (http://qt-project.org/doc/qt-5/qpen.html#isCosmetic)

косметические ручки используются для рисования штрихов с постоянной шириной независимо от любых преобразований, применяемых к QPainter, с которым они используются.

и это также объясняет, почему установка ширины до нуля работает:

ручка нулевой ширины по умолчанию является косметической; ручки с ненулевой шириной не являются косметическими.