Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Qt 5.7 QML Why are my CheckBox property bindings disappearing?

I have a simple list of CheckBoxes, one for each day of the week. They depend on the value of days, an integer using a mask, 1 bit for each CheckBox.

Assigning to days both with the "clear all" button or the "set all" button works and they update. However, once any of the boxes have been clicked, they no longer respond to changes in the dependent property days.

Why is this? Are they somehow becoming unbound. If so, should i be manually re-binding them, and if so why?

Here's the code,

import QtQuick 2.7
import QtQuick.Controls 1.4
import QtQuick.Layouts 1.3

ApplicationWindow
{
    visible: true
    width: 800
    height: 400

    property int days: 0

    ColumnLayout
    {
        Repeater
        {
            model: ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"]
            CheckBox
            {
                text: modelData
                checked: (days & (1<<index)) != false
                onClicked:
                {
                    if (checked) days |= (1<<index);
                    else days &= ~(1<<index);
                }

            }
        }

        Button 
        {
            text: "clear all"
            onClicked: days = 0
        }

        Button 
        {
            text: "set all"
            onClicked: days = 127
        }
    }
}

which looks like this:

enter image description here

To reproduce the problem, first click on "set all" and "clear all". then click on some checkboxes. Then click on "set all" and "clear all" again. You'll see that the boxes you checked are no longer affected.

thanks.

like image 822
jkj yuio Avatar asked Aug 05 '16 22:08

jkj yuio


2 Answers

OP here.

Selbie's answer is quite correct. But i'd like to post a variation that i prefer.

I have come to the conclusion that CheckBoxes are broken in QT. This is because you will want to bind them to your data model. and you will also want to click them (otherwise what's the point). Clicking on them breaks the connection to the model, so that it must be fixed up manually (see Selbie's answer). To me this a broken design.

My variation uses a Binding so that it doesn't have to be re-established every time you click.

like this:

import QtQuick 2.7
import QtQuick.Controls 1.4
import QtQuick.Layouts 1.3

ApplicationWindow
{
    visible: true
    width: 800
    height: 400

    property int days: 0

    ColumnLayout
    {
        Repeater
        {
            model: ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"]
            CheckBox
            {
                text: modelData
                Binding on checked { value: (days & (1 << index)) != 0 }
                onClicked:
                {
                    if (checked) days |= (1<<index)
                    else days &= ~(1<<index)
                }
            }
        }

        Button 
        {
            text: "clear all"
            onClicked: days = 0
        }

        Button 
        {
            text: "set all"
            onClicked: days = 127
        }
    }
}

Posting this variation for the benefit of others.

like image 197
jkj yuio Avatar answered Nov 15 '22 18:11

jkj yuio


When you manually click the checkbox, the checked property gets reassigned to a hardcoded true instead of the original expression: (days & (1<<index)) != false. Likewise, a manual uncheck of the box forces the checked property to a hardcoded false.

The fix is to simply rebind the checked property using Qt.binding. I've cleaned up your javascript and fixed your bug. You are welcome.

    Repeater
    {
        model: ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"]
        CheckBox
        {
            function isChecked() {
                return ((days & (1 << index)) != 0);
            }

            text: modelData
            checked: isChecked()
            onClicked:
            {
                if (checked) {
                    days |= (1<<index);
                }
                else {
                    days &= ~(1<<index);
                }

                // now rebind the item's checked property
                checked = Qt.binding(isChecked);

            }
        }
    }
like image 39
selbie Avatar answered Nov 15 '22 16:11

selbie