A property that is bound to an expression is updated when something in the expression changes. This is called a dependency.
EDIT:
To clarify:
- I'm interested in details on how Qt determines a list of dependencies
- Dependencies on simple bindings such as
x: y
are more or less obvious- The question is about less obvious cases such as
x: myItemId["y"]
andx: myFunction(z)
wheremyFunction(p) { if (p) return myItemId.y }
Sometimes QML engine is able to detect change even if the expression is a function call without arguments, other times it cannot do that (for example mapToItem(item,0,0).x).
Another example of imperfection is that setting JS array item value without reassigning the array itself doesn't normally produce onXxxxxChanged signal or update anything referring to that array value.
An expression with unused result (x: {myForcedDependency; return myActualCalculation()}
) is sometimes suggested to force a dependency.
According to this KDAB article and Qt source code, a binding expression is not only evaluated but any properties "accessed" during that are "captured" in something called a "guard", then every guard properties onXxxxxChanged() signals are connected, but actual details of this process are unclear.
So my questions are:
Are there any defined rules of dependency resolution?
How does it really work?
- How deeply does QQmlEngine/V8 scan "accesses" into functions called by the binding expression and what may prevent it from doing that?
- Is dependency-detection only based on the first attempt at property resolution?
- Are all possible code paths checked even if execution never reached there yet?
- Are non-trivial accesses determined in those cases, such as object["property"] syntax?
- What if some unexecuted code is (currently) erroneous (and does not produce an error but cannot be properly analyzed)?
How can the dependency resolution process be influenced?
- Is there a way to avoid or block a dependency?
- As far as I understand an intermediate "filter" property that only actually changes its value when it's necessary to update is the intended way, correct?
- Is there an intended way to force a dependency?
- Is manually emitting "XxxxxChanged" signal the correct/supported way to force an update?
- Is adding an unused reference a legal/intended way to do it or undefined behavior based on the current implementation quirk?
- Is there a way to avoid or block a dependency?
Any information would be useful, although I did read the official documentation on QML properties, QML bindings and JavaScript expressions and didn't find any concrete explanation - if you refer to the official documentation please quote relevant parts.
Please note that I'm not asking you to test if any of this works on your system, but if it's supposed to work - if it can be relied on