如何在QML spinbox中使用float

时间:2017-04-14 07:06:01

标签: qt qml qtquick2 qspinbox

我使用的是QML Spinbox,但我无法在其中使用浮点数。 如果我写的是value: 5.0之类的内容,它将显示为5,因此它将显示为int而不是浮点数。

你知道如何继续吗?

非常感谢,祝你有个美好的一天!

4 个答案:

答案 0 :(得分:3)

您可以使用自定义文本

创建一个Spinbox

<强> DoubleSpinBox.qml

import QtQuick 2.0

import QtQuick.Controls 2.1

Item {
    property int decimals: 2
    property real realValue: 0.0
    property real realFrom: 0.0
    property real realTo: 100.0
    property real realStepSize: 1.0

    SpinBox{
        property real factor: Math.pow(10, decimals)
        id: spinbox
        stepSize: realStepSize*factor
        value: realValue*factor
        to : realTo*factor
        from : realFrom*factor
        validator: DoubleValidator {
            bottom: Math.min(spinbox.from, spinbox.to)*spinbox.factor
            top:  Math.max(spinbox.from, spinbox.to)*spinbox.factor
        }

        textFromValue: function(value, locale) {
            return parseFloat(value*1.0/factor).toFixed(decimals);
        }

    }
}

示例:

DoubleSpinBox{
    realValue: 5.0
    realStepSize: 0.01
}

enter image description here

答案 1 :(得分:2)

当前的SpinBox(Controls 2.0-2.4)有一个限制,即它只能接受+/- to的{​​{1}} / from范围(甚至不能接受)。处理一个无符号的int,BTW)。对于0x7FFF FFFF来说,这(可能是很严重的)含义是,对于小数点后所需的每个数字,您都会在小数点前丢失一个数字。例如,如果您需要6位十进制精度,则最大可能值为real。到目前为止,这都影响了此答案中提出的两个解决方案。

如果您尝试将2147.483647to设置为超出from的限制(例如,通过乘以此处建议中的系数),您将变得非常“奇怪”可能令人莫名其妙的行为。如果您尝试直接超过这些限制(例如int),则会出现错误。

查看(并为:投票))QTBUG-67349

我意识到这不是问题的答案,但希望它可以像我一样省下几个小时的挫败感。

到目前为止,我只能提出的唯一解决方法是将文本编辑字段与DoubleValidator一起使用。我将尝试以简化的示例返回这里。

编辑:

Here is my version of a DoubleSpinBox。该代码太长,无法粘贴到此处(IMHO),我宁愿不要维护多个版本。

简而言之,该想法是完全绕过SpinBox to: 0xFFFFFFFF,而仅使用基本Controls.2 SpinBox来实现按钮和整体外观。这意味着不需要按主题(融合/材质/等)自定义。

不幸的是,这涉及到重新实现所有事件处理程序(按钮按下/重复,滚轮滚动,文本编辑)以及其他一些功能。但是(到目前为止),我认为在尝试匹配所有不同主题时,这比使用自定义按钮/等从头重新实现整个事情要好。

它还添加了一些标准SpinBox中不可用的选项...,为什么呢? :)

Default theme Fusion theme Material theme

答案 2 :(得分:0)

这是@eyllanesc修复版。所有其他缺少的属性必须是别名。如果你发现,请报告我的错误。

<强> DoubleSpinBox.qml

import QtQuick 2.0
import QtQuick.Controls 2.2
import QmlUtils 1.0

Item
{
    id: doublespinbox
    width: 140
    height: 40
    property int decimals: 1
    property alias value: valuePreview.value
    property real from: 0
    property real to: 99
    property real stepSize: 1
    property alias editable: spinbox.editable
    property alias font: spinbox.font

    SpinBox
    {
        id: spinbox
        property bool init: false
        property real factor: Math.pow(10, decimals)

        function setValue(preview)
        {
            init = true
            value = preview.value * factor
            init = false
            preview.value = value / factor
        }

        DoubleValuePreview
        {
            id: valuePreview
            onValuePreview: spinbox.setValue(preview)
        }

        anchors.fill: parent
        editable: true
        stepSize: doublespinbox.stepSize * factor
        to : doublespinbox.to * factor
        from : doublespinbox.from * factor

        onValueChanged:
        {
            if (init)
                return

            valuePreview.setValueDirect(value / factor)
        }

        validator: DoubleValidator
        {
            bottom: Math.min(spinbox.from, spinbox.to)
            top: Math.max(spinbox.from, spinbox.to)
        }

        textFromValue: function(value, locale)
        {
            return Number(value / factor).toLocaleString(locale, 'f', doublespinbox.decimals)
        }

        valueFromText: function(text, locale)
        {
            doublespinbox.value = Number.fromLocaleString(locale, text)
            return doublespinbox.value * factor
        }
    }
}

<强> QmlValuePreview.h

#pragma once

#include <qobject.h>
#include <qqmlengine.h>

class QDoubleValueArg : public QObject
{
    Q_OBJECT
public:
    QDoubleValueArg(double value) : Value(value)
    {
        QQmlEngine::setObjectOwnership(this, QQmlEngine::CppOwnership);
    }
public:
    Q_PROPERTY(double value MEMBER Value)
public:
    double Value;
};

class QmlDoubleValuePreview : public QObject
{
    Q_OBJECT
public:
    using QObject::QObject;
public:
    Q_PROPERTY(double value READ getValue WRITE setValue NOTIFY valueChanged)
public:
    Q_INVOKABLE void setValueDirect(double value)
    {
        if (m_value == value)
            return;

        m_value = value;
        emit valueChanged();
    }
public:
    inline double getValue() const { return m_value; }
    inline void setValue(double value)
    {
        if (m_value == value)
            return;

        QDoubleValueArg arg(value);
        emit valuePreview(&arg);
        if (m_value == arg.Value)
            return;

        m_value = arg.Value;
        emit valueChanged();
    }
signals:
    void valueChanged();
    void valuePreview(QDoubleValueArg *preview);
private:
    double m_value = 0;
};

注册时:

#define URI "QmlUtils"
#define VERSION_MAJOR 1
#define VERSION_MINOR 0

void registerTypes()
{
    qRegisterMetaType<QDoubleValueArg *>("QDoubleValueArg *");
    qmlRegisterType<QmlDoubleValuePreview>(URI, VERSION_MAJOR, VERSION_MINOR, "DoubleValuePreview");
}

答案 3 :(得分:0)

我在上面的代码中看到两个问题。

  1. 如果显示像 1.299(period) 这样的数字会出错 DoubleSpinBox 将显示 1.2 而不是 1.3 为了修复它,你应该写

    value = Math.round(preview.value * factor)

在 DoubleSpinBox.qml 函数中 setValue(preview)

  1. 在这一行我们有覆盖绑定

    doublespinbox.value = Number.fromLocaleString(locale, text)

因为我们有 “属性别名值:valuePreview.value” 并且可能“值”会在代码中绑定一些 所以你不能这样做,而且完全足以写这个:

function(text, locale) {
        return Number.fromLocaleString(locale, text) * factor
}
  1. 我想我们也可以删除这一行

    preview.value = 值/因子

在函数 setValue(preview)