/usr/share/qt5/doc/qtqml/qtqml-documents-scope.html is in qtdeclarative5-doc-html 5.9.5-0ubuntu1.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 | <?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html>
<html lang="en">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<!-- scope.qdoc -->
<title>Scope and Naming Resolution | Qt QML 5.9</title>
<link rel="stylesheet" type="text/css" href="style/offline-simple.css" />
<script type="text/javascript">
document.getElementsByTagName("link").item(0).setAttribute("href", "style/offline.css");
// loading style sheet breaks anchors that were jumped to before
// so force jumping to anchor again
setTimeout(function() {
var anchor = location.hash;
// need to jump to different anchor first (e.g. none)
location.hash = "#";
setTimeout(function() {
location.hash = anchor;
}, 0);
}, 0);
</script>
</head>
<body>
<div class="header" id="qtdocheader">
<div class="main">
<div class="main-rounded">
<div class="navigationbar">
<table><tr>
<td >Qt 5.9</td><td ><a href="qtqml-index.html">Qt QML</a></td><td >Scope and Naming Resolution</td></tr></table><table class="buildversion"><tr>
<td id="buildversion" width="100%" align="right">Qt 5.9.5 Reference Documentation</td>
</tr></table>
</div>
</div>
<div class="content">
<div class="line">
<div class="content mainContent">
<div class="sidebar">
<div class="toc">
<h3><a name="toc">Contents</a></h3>
<ul>
<li class="level1"><a href="#javascript-scope">JavaScript Scope</a></li>
<li class="level1"><a href="#type-names-and-imported-javascript-files">Type Names and Imported JavaScript Files</a></li>
<li class="level1"><a href="#binding-scope-object">Binding Scope Object</a></li>
<li class="level1"><a href="#component-scope">Component Scope</a></li>
<li class="level1"><a href="#component-instance-hierarchy">Component Instance Hierarchy</a></li>
<li class="level1"><a href="#overridden-properties">Overridden Properties</a></li>
<li class="level1"><a href="#javascript-global-object">JavaScript Global Object</a></li>
</ul>
</div>
<div class="sidebar-content" id="sidebar-content"></div></div>
<h1 class="title">Scope and Naming Resolution</h1>
<span class="subtitle"></span>
<!-- $$$qtqml-documents-scope.html-description -->
<div class="descr"> <a name="details"></a>
<p>QML property bindings, inline functions, and imported JavaScript files all run in a JavaScript scope. Scope controls which variables an expression can access, and which variable takes precedence when two or more names conflict.</p>
<p>As JavaScript's built-in scope mechanism is very simple, QML enhances it to fit more naturally with the QML language extensions.</p>
<a name="javascript-scope"></a>
<h2 id="javascript-scope">JavaScript Scope</h2>
<p>QML's scope extensions do not interfere with JavaScript's natural scoping. JavaScript programmers can reuse their existing knowledge when programming functions, property bindings or imported JavaScript files in QML.</p>
<p>In the following example, the <code>addConstant()</code> method will add 13 to the parameter passed just as the programmer would expect irrespective of the value of the QML object's <code>a</code> and <code>b</code> properties.</p>
<pre class="cpp">
<span class="type"><a href="qml-qtqml-qtobject.html">QtObject</a></span> {
property <span class="type">int</span> a: <span class="number">3</span>
property <span class="type">int</span> b: <span class="number">9</span>
function addConstant(b) {
var a <span class="operator">=</span> <span class="number">13</span>;
<span class="keyword">return</span> b <span class="operator">+</span> a;
}
}
</pre>
<p>That QML respects JavaScript's normal scoping rules even applies in bindings. This totally evil, abomination of a binding will assign 12 to the QML object's <code>a</code> property.</p>
<pre class="cpp">
<span class="type"><a href="qml-qtqml-qtobject.html">QtObject</a></span> {
property <span class="type">int</span> a
a: { var a <span class="operator">=</span> <span class="number">12</span>; a; }
}
</pre>
<p>Every JavaScript expression, function or file in QML has its own unique variable object. Local variables declared in one will never conflict with local variables declared in another.</p>
<a name="type-names-and-imported-javascript-files"></a>
<h2 id="type-names-and-imported-javascript-files">Type Names and Imported JavaScript Files</h2>
<p><a href="qtqml-documents-topic.html">QML Documents</a> include import statements that define the type names and JavaScript files visible to the document. In addition to their use in the QML declaration itself, type names are used by JavaScript code when accessing <a href="qtqml-syntax-objectattributes.html#attached-properties-and-attached-signal-handlers">attached properties</a> and enumeration values.</p>
<p>The effect of an import applies to every property binding, and JavaScript function in the QML document, even those in nested inline components. The following example shows a simple QML file that accesses some enumeration values and calls an imported JavaScript function.</p>
<pre class="cpp">
import <span class="type">QtQuick</span> <span class="number">2.0</span>
import <span class="string">"code.js"</span> as Code
ListView {
snapMode: ListView<span class="operator">.</span>SnapToItem
delegate: Component {
Text {
elide: Text<span class="operator">.</span>ElideMiddle
text: <span class="string">"A really, really long string that will require eliding."</span>
color: Code<span class="operator">.</span>defaultColor()
}
}
}
</pre>
<a name="binding-scope-object"></a>
<h2 id="binding-scope-object">Binding Scope Object</h2>
<p>An object which has a <a href="qtqml-syntax-propertybinding.html">property binding</a> is known as the binding's <i>scope object</i>. In the following example, the Item object is the binding's scope object.</p>
<pre class="cpp">
Item {
anchors<span class="operator">.</span>left: parent<span class="operator">.</span>left
}
</pre>
<p>Bindings have access to the scope object's properties without qualification. In the previous example, the binding accesses the Item's <code>parent</code> property directly, without needing any form of object prefix. QML introduces a more structured, object-oriented approach to JavaScript, and consequently does not require the use of the JavaScript <code>this</code> property.</p>
<p>Care must be used when accessing <a href="qtqml-syntax-objectattributes.html#attached-properties-and-attached-signal-handlers">attached properties</a> from bindings due to their interaction with the scope object. Conceptually attached properties exist on <i>all</i> objects, even if they only have an effect on a subset of those. Consequently unqualified attached property reads will always resolve to an attached property on the scope object, which is not always what the programmer intended.</p>
<p>For example, the PathView type attaches interpolated value properties to its delegates depending on their position in the path. As PathView only meaningfully attaches these properties to the root object in the delegate, any sub-object that accesses them must explicitly qualify the root object, as shown below.</p>
<pre class="cpp">
PathView {
delegate: Component {
Rectangle {
id: root
Image {
scale: root<span class="operator">.</span>PathView<span class="operator">.</span>scale
}
}
}
}
</pre>
<p>If the Image object omitted the <code>root</code> prefix, it would inadvertently access the unset <code>PathView.scale</code> attached property on itself.</p>
<a name="component-scope"></a>
<h2 id="component-scope">Component Scope</h2>
<p>Each QML component in a QML document defines a logical scope. Each document has at least one root component, but can also have other inline sub-components. The component scope is the union of the object ids within the component and the component's root object's properties.</p>
<pre class="cpp">
Item {
property string title
Text {
id: titletype
text: <span class="string">"<b>"</span> <span class="operator">+</span> title <span class="operator">+</span> <span class="string">"</b>"</span>
font<span class="operator">.</span>pixelSize: <span class="number">22</span>
anchors<span class="operator">.</span>top: parent<span class="operator">.</span>top
}
Text {
text: titletype<span class="operator">.</span>text
font<span class="operator">.</span>pixelSize: <span class="number">18</span>
anchors<span class="operator">.</span>bottom: parent<span class="operator">.</span>bottom
}
}
</pre>
<p>The example above shows a simple QML component that displays a rich text title string at the top, and a smaller copy of the same text at the bottom. The first <code>Text</code> type directly accesses the component's <code>title</code> property when forming the text to display. That the root type's properties are directly accessible makes it trivial to distribute data throughout the component.</p>
<p>The second <code>Text</code> type uses an id to access the first's text directly. IDs are specified explicitly by the QML programmer so they always take precedence over other property names (except for those in the <a href="qtqml-documents-scope.html#javascript-scope">JavaScript Scope</a>). For example, in the unlikely event that the binding's <a href="qtqml-documents-scope.html#binding-scope-object">scope object</a> had a <code>titletype</code> property in the previous example, the <code>titletype</code> id would still take precedence.</p>
<a name="component-instance-hierarchy"></a>
<h2 id="component-instance-hierarchy">Component Instance Hierarchy</h2>
<p>In QML, component instances connect their component scopes together to form a scope hierarchy. Component instances can directly access the component scopes of their ancestors.</p>
<p>The easiest way to demonstrate this is with inline sub-components whose component scopes are implicitly scoped as children of the outer component.</p>
<pre class="cpp">
Item {
property color defaultColor: <span class="string">"blue"</span>
ListView {
delegate: Component {
Rectangle {
color: defaultColor
}
}
}
}
</pre>
<p>The component instance hierarchy allows instances of the delegate component to access the <code>defaultColor</code> property of the <code>Item</code> type. Of course, had the delegate component had a property called <code>defaultColor</code> that would have taken precedence.</p>
<p>The component instance scope hierarchy extends to out-of-line components, too. In the following example, the <code>TitlePage.qml</code> component creates two <code>TitleText</code> instances. Even though the <code>TitleText</code> type is in a separate file, it still has access to the <code>title</code> property when it is used from within the <code>TitlePage</code>. QML is a dynamically scoped language - depending on where it is used, the <code>title</code> property may resolve differently.</p>
<pre class="cpp">
<span class="comment">// TitlePage.qml</span>
import <span class="type">QtQuick</span> <span class="number">2.0</span>
Item {
property string title
TitleText {
size: <span class="number">22</span>
anchors<span class="operator">.</span>top: parent<span class="operator">.</span>top
}
TitleText {
size: <span class="number">18</span>
anchors<span class="operator">.</span>bottom: parent<span class="operator">.</span>bottom
}
}
<span class="comment">// TitleText.qml</span>
import <span class="type">QtQuick</span> <span class="number">2.0</span>
Text {
property <span class="type">int</span> size
text: <span class="string">"<b>"</span> <span class="operator">+</span> title <span class="operator">+</span> <span class="string">"</b>"</span>
font<span class="operator">.</span>pixelSize: size
}
</pre>
<p>Dynamic scoping is very powerful, but it must be used cautiously to prevent the behavior of QML code from becoming difficult to predict. In general it should only be used in cases where the two components are already tightly coupled in another way. When building reusable components, it is preferable to use property interfaces, like this:</p>
<pre class="cpp">
<span class="comment">// TitlePage.qml</span>
import <span class="type">QtQuick</span> <span class="number">2.0</span>
Item {
id: root
property string title
TitleText {
title: root<span class="operator">.</span>title
size: <span class="number">22</span>
anchors<span class="operator">.</span>top: parent<span class="operator">.</span>top
}
TitleText {
title: root<span class="operator">.</span>title
size: <span class="number">18</span>
anchors<span class="operator">.</span>bottom: parent<span class="operator">.</span>bottom
}
}
<span class="comment">// TitleText.qml</span>
import <span class="type">QtQuick</span> <span class="number">2.0</span>
Text {
property string title
property <span class="type">int</span> size
text: <span class="string">"<b>"</span> <span class="operator">+</span> title <span class="operator">+</span> <span class="string">"</b>"</span>
font<span class="operator">.</span>pixelSize: size
}
</pre>
<a name="overridden-properties"></a>
<h2 id="overridden-properties">Overridden Properties</h2>
<p>QML permits property names defined in an object declaration to be overridden by properties declared within another object declaration that extends the first. For example:</p>
<pre class="cpp">
<span class="comment">// Displayable.qml</span>
import <span class="type">QtQuick</span> <span class="number">2.0</span>
Item {
property string title
property string detail
Text {
text: <span class="string">"<b>"</span> <span class="operator">+</span> title <span class="operator">+</span> <span class="string">"</b><br>"</span> <span class="operator">+</span> detail
}
function getTitle() { <span class="keyword">return</span> title }
function setTitle(newTitle) { title <span class="operator">=</span> newTitle }
}
<span class="comment">// Person.qml</span>
import <span class="type">QtQuick</span> <span class="number">2.0</span>
Displayable {
property string title
property string firstName
property string lastName
function fullName() { <span class="keyword">return</span> title <span class="operator">+</span> <span class="string">" "</span> <span class="operator">+</span> firstName <span class="operator">+</span> <span class="string">" "</span> <span class="operator">+</span> lastName }
}
</pre>
<p>Here, the name <code>title</code> is given to both the heading of the output text for Displayable, and also to the honorific title of the Person object.</p>
<p>An overridden property is resolved according to the scope in which it is referenced. Inside the scope of the Person component, or from an external scope that refers to an instance of the Person component, <code>title</code> resolves to the property declared inside Person.qml. The <code>fullName</code> function will refer to the <code>title</code> property declared inside Person.</p>
<p>Inside the Displayable component, however, <code>title</code> refers to the property declared in Displayable.qml. The getTitle() and setTitle() functions, and the binding for the <code>text</code> property of the Text object will all refer to the <code>title</code> property declared in the Displayable component.</p>
<p>Despite sharing the same name, the two properties are entirely separate. An onChanged signal handler for one of the properties will not be triggered by a change to the other property with the same name. An alias to either property will refer to one or the other, but not both.</p>
<a name="javascript-global-object"></a>
<h2 id="javascript-global-object">JavaScript Global Object</h2>
<p>QML disallows type, id and property names that conflict with the properties on the global object to prevent any confusion. Programmers can be confident that <code>Math.min(10, 9)</code> will always work as expected!</p>
<p>See <a href="qtqml-javascript-hostenvironment.html">JavaScript Host Environment</a> for more information.</p>
</div>
<!-- @@@qtqml-documents-scope.html -->
</div>
</div>
</div>
</div>
</div>
<div class="footer">
<p>
<acronym title="Copyright">©</acronym> 2017 The Qt Company Ltd.
Documentation contributions included herein are the copyrights of
their respective owners.<br> The documentation provided herein is licensed under the terms of the <a href="http://www.gnu.org/licenses/fdl.html">GNU Free Documentation License version 1.3</a> as published by the Free Software Foundation.<br> Qt and respective logos are trademarks of The Qt Company Ltd. in Finland and/or other countries worldwide. All other trademarks are property
of their respective owners. </p>
</div>
</body>
</html>
|