summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJanne Koskinen <janne.p.koskinen@digia.com>2013-10-30 12:08:36 +0200
committerThe Qt Project <gerrit-noreply@qt-project.org>2013-12-09 10:30:14 +0100
commit1d48b9ea56e42b2cf963090078ee2162bde23a25 (patch)
tree5ad940eaa8bfc7bd4fd71fbebb9e72c8190cd155
parentc889b401d4668482bdef6c976bfa2d57144cf7ed (diff)
Fixed rounding errors in QtQuick1 to int conversions
qRound introduced errors when qreal is single precision. Added double signature for qround so that string and int don't lose precision in conversion. Task-number: QTBUG-33625 Change-Id: I58582f57d5cd68fcad3fe9efb5fea5935f61b9e3 Reviewed-by: Olivier Goffart <ogoffart@woboq.com> Reviewed-by: Thiago Macieira <thiago.macieira@intel.com>
-rw-r--r--src/declarative/qml/qdeclarativeobjectscriptclass.cpp9
-rw-r--r--src/declarative/qml/qdeclarativestringconverters.cpp10
-rw-r--r--tests/auto/declarative/qdeclarativevaluetypes/data/qtbug-33625.qml12
-rw-r--r--tests/auto/declarative/qdeclarativevaluetypes/tst_qdeclarativevaluetypes.cpp14
4 files changed, 41 insertions, 4 deletions
diff --git a/src/declarative/qml/qdeclarativeobjectscriptclass.cpp b/src/declarative/qml/qdeclarativeobjectscriptclass.cpp
index 99d34b4060..8cfdad29c4 100644
--- a/src/declarative/qml/qdeclarativeobjectscriptclass.cpp
+++ b/src/declarative/qml/qdeclarativeobjectscriptclass.cpp
@@ -328,7 +328,12 @@ void QDeclarativeObjectScriptClass::setProperty(Object *object,
{
return setProperty(toQObject(object), name, value, context());
}
-
+namespace {
+int qRoundDouble(double d)
+{
+ return d >= double(0.0) ? int(d + double(0.5)) : int(d - int(d-1) + double(0.5)) + int(d-1);
+}
+}
void QDeclarativeObjectScriptClass::setProperty(QObject *obj,
const Identifier &name,
const QScriptValue &value,
@@ -405,7 +410,7 @@ void QDeclarativeObjectScriptClass::setProperty(QObject *obj,
} else {
//### expand optimization for other known types
if (lastData->propType == QMetaType::Int && value.isNumber()) {
- int rawValue = qRound(value.toNumber());
+ int rawValue = qRoundDouble(value.toNumber());
int status = -1;
int flags = 0;
void *a[] = { (void *)&rawValue, 0, &status, &flags };
diff --git a/src/declarative/qml/qdeclarativestringconverters.cpp b/src/declarative/qml/qdeclarativestringconverters.cpp
index bb5eacb9a0..738619d862 100644
--- a/src/declarative/qml/qdeclarativestringconverters.cpp
+++ b/src/declarative/qml/qdeclarativestringconverters.cpp
@@ -97,13 +97,19 @@ QVariant QDeclarativeStringConverters::variantFromString(const QString &s)
return QVariant(s);
}
+namespace {
+int qRoundDouble(double d)
+{
+ return d >= double(0.0) ? int(d + double(0.5)) : int(d - int(d-1) + double(0.5)) + int(d-1);
+}
+}
QVariant QDeclarativeStringConverters::variantFromString(const QString &s, int preferredType, bool *ok)
{
switch (preferredType) {
case QMetaType::Int:
- return QVariant(int(qRound(s.toDouble(ok))));
+ return QVariant(int(qRoundDouble(s.toDouble(ok))));
case QMetaType::UInt:
- return QVariant(uint(qRound(s.toDouble(ok))));
+ return QVariant(uint(qRoundDouble(s.toDouble(ok))));
case QMetaType::QColor:
return QVariant::fromValue(colorFromString(s, ok));
#ifndef QT_NO_DATESTRING
diff --git a/tests/auto/declarative/qdeclarativevaluetypes/data/qtbug-33625.qml b/tests/auto/declarative/qdeclarativevaluetypes/data/qtbug-33625.qml
new file mode 100644
index 0000000000..977028fc5d
--- /dev/null
+++ b/tests/auto/declarative/qdeclarativevaluetypes/data/qtbug-33625.qml
@@ -0,0 +1,12 @@
+import QtQuick 1.0
+Item {
+ property string stringtest: "16777237"
+ property int stringint: stringtest
+
+ property int inttest: 16777237
+ property int intint: 0
+
+ Component.onCompleted: {
+ intint = inttest;
+ }
+}
diff --git a/tests/auto/declarative/qdeclarativevaluetypes/tst_qdeclarativevaluetypes.cpp b/tests/auto/declarative/qdeclarativevaluetypes/tst_qdeclarativevaluetypes.cpp
index 87a60e30e5..a21ffc634d 100644
--- a/tests/auto/declarative/qdeclarativevaluetypes/tst_qdeclarativevaluetypes.cpp
+++ b/tests/auto/declarative/qdeclarativevaluetypes/tst_qdeclarativevaluetypes.cpp
@@ -96,6 +96,7 @@ private slots:
void returnValues();
void varAssignment();
void bindingsSpliceCorrectly();
+ void qtbug_33625();
private:
QDeclarativeEngine engine;
@@ -1012,6 +1013,19 @@ void tst_qdeclarativevaluetypes::bindingsSpliceCorrectly()
}
}
+//qreal as float should not limit int to single precision.
+void tst_qdeclarativevaluetypes::qtbug_33625()
+{
+ QDeclarativeComponent component(&engine, TEST_FILE("qtbug-33625.qml"));
+ QObject *object = component.create();
+ QVERIFY(object != 0);
+
+ QCOMPARE(object->property("intint").toInt(), 16777237);
+ QCOMPARE(object->property("stringint").toInt(), 16777237);
+
+ delete object;
+}
+
QTEST_MAIN(tst_qdeclarativevaluetypes)
#include "tst_qdeclarativevaluetypes.moc"