Как сохранить ширину пикселя 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 компиляция:
и это то, что я вижу на моем Qt5.0.2 компиляция:
очевидно, что в чертеже границы каждого прямоугольника есть разное поведение. Кроме того, если я увеличиваю масштаб с помощью колеса мыши, ширина границы прямоугольника становится больше компиляция 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, с которым они используются.
и это также объясняет, почему установка ширины до нуля работает:
ручка нулевой ширины по умолчанию является косметической; ручки с ненулевой шириной не являются косметическими.