ComboBox.qml 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710
  1. /****************************************************************************
  2. **
  3. ** Copyright (C) 2016 The Qt Company Ltd.
  4. ** Contact: https://www.qt.io/licensing/
  5. **
  6. ** This file is part of the Qt Quick Controls module of the Qt Toolkit.
  7. **
  8. ** $QT_BEGIN_LICENSE:LGPL$
  9. ** Commercial License Usage
  10. ** Licensees holding valid commercial Qt licenses may use this file in
  11. ** accordance with the commercial license agreement provided with the
  12. ** Software or, alternatively, in accordance with the terms contained in
  13. ** a written agreement between you and The Qt Company. For licensing terms
  14. ** and conditions see https://www.qt.io/terms-conditions. For further
  15. ** information use the contact form at https://www.qt.io/contact-us.
  16. **
  17. ** GNU Lesser General Public License Usage
  18. ** Alternatively, this file may be used under the terms of the GNU Lesser
  19. ** General Public License version 3 as published by the Free Software
  20. ** Foundation and appearing in the file LICENSE.LGPL3 included in the
  21. ** packaging of this file. Please review the following information to
  22. ** ensure the GNU Lesser General Public License version 3 requirements
  23. ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
  24. **
  25. ** GNU General Public License Usage
  26. ** Alternatively, this file may be used under the terms of the GNU
  27. ** General Public License version 2.0 or (at your option) the GNU General
  28. ** Public license version 3 or any later version approved by the KDE Free
  29. ** Qt Foundation. The licenses are as published by the Free Software
  30. ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
  31. ** included in the packaging of this file. Please review the following
  32. ** information to ensure the GNU General Public License requirements will
  33. ** be met: https://www.gnu.org/licenses/gpl-2.0.html and
  34. ** https://www.gnu.org/licenses/gpl-3.0.html.
  35. **
  36. ** $QT_END_LICENSE$
  37. **
  38. ****************************************************************************/
  39. import QtQuick 2.2
  40. import QtQuick.Controls 1.2
  41. import QtQuick.Controls.Private 1.0
  42. /*!
  43. \qmltype ComboBox
  44. \inqmlmodule QtQuick.Controls
  45. \since 5.1
  46. \ingroup controls
  47. \brief Provides a drop-down list functionality.
  48. \image combobox.png
  49. Add items to the ComboBox by assigning it a ListModel, or a list of strings
  50. to the \l model property.
  51. \qml
  52. ComboBox {
  53. width: 200
  54. model: [ "Banana", "Apple", "Coconut" ]
  55. }
  56. \endqml
  57. In this example we are demonstrating how to use a ListModel with a combo box.
  58. \qml
  59. ComboBox {
  60. currentIndex: 2
  61. model: ListModel {
  62. id: cbItems
  63. ListElement { text: "Banana"; color: "Yellow" }
  64. ListElement { text: "Apple"; color: "Green" }
  65. ListElement { text: "Coconut"; color: "Brown" }
  66. }
  67. width: 200
  68. onCurrentIndexChanged: console.debug(cbItems.get(currentIndex).text + ", " + cbItems.get(currentIndex).color)
  69. }
  70. \endqml
  71. You can make a combo box editable by setting the \l editable property. An editable combo box will
  72. autocomplete its text based on what is available in the model.
  73. In the next example we demonstrate how you can append content to an editable combo box by
  74. reacting to the \l accepted signal. Note that you have to explicitly prevent duplicates.
  75. \qml
  76. ComboBox {
  77. editable: true
  78. model: ListModel {
  79. id: model
  80. ListElement { text: "Banana"; color: "Yellow" }
  81. ListElement { text: "Apple"; color: "Green" }
  82. ListElement { text: "Coconut"; color: "Brown" }
  83. }
  84. onAccepted: {
  85. if (find(currentText) === -1) {
  86. model.append({text: editText})
  87. currentIndex = find(editText)
  88. }
  89. }
  90. }
  91. \endqml
  92. You can create a custom appearance for a ComboBox by
  93. assigning a \l {ComboBoxStyle}.
  94. */
  95. Control {
  96. id: comboBox
  97. /*! \qmlproperty model ComboBox::model
  98. The model to populate the ComboBox from.
  99. Changing the model after initialization will reset \l currentIndex to \c 0.
  100. */
  101. property alias model: popupItems.model
  102. /*! The model role used for populating the ComboBox. */
  103. property string textRole: ""
  104. /*! \qmlproperty int ComboBox::currentIndex
  105. The index of the currently selected item in the ComboBox.
  106. Setting currentIndex to \c -1 will reset the selection and clear the text
  107. label. If \l editable is \c true, you may also need to manually clear \l editText.
  108. \sa model
  109. */
  110. property alias currentIndex: popup.__selectedIndex
  111. /*! \qmlproperty string ComboBox::currentText
  112. The text of the currently selected item in the ComboBox.
  113. \note Since \c currentText depends on \c currentIndex, there's no way to ensure \c currentText
  114. will be up to date whenever a \c onCurrentIndexChanged handler is called.
  115. */
  116. readonly property alias currentText: popup.currentText
  117. /*! This property holds whether the combo box can be edited by the user.
  118. The default value is \c false.
  119. \since QtQuick.Controls 1.1
  120. */
  121. property bool editable: false
  122. /*! \qmlproperty string ComboBox::editText
  123. \since QtQuick.Controls 1.1
  124. This property specifies text being manipulated by the user for an editable combo box.
  125. */
  126. property alias editText: input.text
  127. /*! \qmlproperty enumeration ComboBox::inputMethodHints
  128. \since QtQuick.Controls 1.5
  129. Provides hints to the input method about the expected content of the combo box and how it
  130. should operate.
  131. The value is a bit-wise combination of flags or \c Qt.ImhNone if no hints are set.
  132. Flags that alter behavior are:
  133. \list
  134. \li Qt.ImhHiddenText - Characters should be hidden, as is typically used when entering passwords.
  135. \li Qt.ImhSensitiveData - Typed text should not be stored by the active input method
  136. in any persistent storage like predictive user dictionary.
  137. \li Qt.ImhNoAutoUppercase - The input method should not try to automatically switch to upper case
  138. when a sentence ends.
  139. \li Qt.ImhPreferNumbers - Numbers are preferred (but not required).
  140. \li Qt.ImhPreferUppercase - Upper case letters are preferred (but not required).
  141. \li Qt.ImhPreferLowercase - Lower case letters are preferred (but not required).
  142. \li Qt.ImhNoPredictiveText - Do not use predictive text (i.e. dictionary lookup) while typing.
  143. \li Qt.ImhDate - The text editor functions as a date field.
  144. \li Qt.ImhTime - The text editor functions as a time field.
  145. \endlist
  146. Flags that restrict input (exclusive flags) are:
  147. \list
  148. \li Qt.ImhDigitsOnly - Only digits are allowed.
  149. \li Qt.ImhFormattedNumbersOnly - Only number input is allowed. This includes decimal point and minus sign.
  150. \li Qt.ImhUppercaseOnly - Only upper case letter input is allowed.
  151. \li Qt.ImhLowercaseOnly - Only lower case letter input is allowed.
  152. \li Qt.ImhDialableCharactersOnly - Only characters suitable for phone dialing are allowed.
  153. \li Qt.ImhEmailCharactersOnly - Only characters suitable for email addresses are allowed.
  154. \li Qt.ImhUrlCharactersOnly - Only characters suitable for URLs are allowed.
  155. \endlist
  156. Masks:
  157. \list
  158. \li Qt.ImhExclusiveInputMask - This mask yields nonzero if any of the exclusive flags are used.
  159. \endlist
  160. */
  161. property alias inputMethodHints: input.inputMethodHints
  162. /*! This property specifies whether the combobox should gain active focus when pressed.
  163. The default value is \c false. */
  164. property bool activeFocusOnPress: false
  165. /*! \qmlproperty bool ComboBox::pressed
  166. This property holds whether the button is being pressed. */
  167. readonly property bool pressed: mouseArea.effectivePressed || popup.__popupVisible
  168. /*! \qmlproperty bool ComboBox::hovered
  169. This property indicates whether the control is being hovered.
  170. */
  171. readonly property bool hovered: mouseArea.containsMouse || input.containsMouse
  172. /*! \qmlproperty int ComboBox::count
  173. \since QtQuick.Controls 1.1
  174. This property holds the number of items in the combo box.
  175. */
  176. readonly property alias count: popupItems.count
  177. /*! \qmlmethod string ComboBox::textAt(int index)
  178. Returns the text for a given \a index.
  179. If an invalid index is provided, \c null is returned
  180. \since QtQuick.Controls 1.1
  181. */
  182. function textAt (index) {
  183. if (index >= count || index < 0)
  184. return null;
  185. return popupItems.objectAt(index).text;
  186. }
  187. /*! \qmlmethod int ComboBox::find(string text)
  188. Finds and returns the index of a given \a text
  189. If no match is found, \c -1 is returned. The search is case sensitive.
  190. \since QtQuick.Controls 1.1
  191. */
  192. function find (text) {
  193. return input.find(text, Qt.MatchExactly)
  194. }
  195. /*!
  196. \qmlproperty Validator ComboBox::validator
  197. \since QtQuick.Controls 1.1
  198. Allows you to set a text validator for an editable ComboBox.
  199. When a validator is set,
  200. the text field will only accept input which leaves the text property in
  201. an intermediate state. The accepted signal will only be sent
  202. if the text is in an acceptable state when enter is pressed.
  203. Currently supported validators are \l[QtQuick]{IntValidator},
  204. \l[QtQuick]{DoubleValidator}, and \l[QtQuick]{RegExpValidator}. An
  205. example of using validators is shown below, which allows input of
  206. integers between 11 and 31 into the text field:
  207. \note This property is only applied when \l editable is \c true
  208. \qml
  209. import QtQuick 2.2
  210. import QtQuick.Controls 1.2
  211. ComboBox {
  212. editable: true
  213. model: 10
  214. validator: IntValidator {bottom: 0; top: 10;}
  215. focus: true
  216. }
  217. \endqml
  218. \sa acceptableInput, accepted, editable
  219. */
  220. property alias validator: input.validator
  221. /*!
  222. \since QtQuick.Controls 1.3
  223. This property contains the edit \l Menu for working
  224. with text selection. Set it to \c null if no menu
  225. is wanted.
  226. \note The menu is only in use when \l editable is \c true
  227. */
  228. property Component menu: input.editMenu.defaultMenu
  229. /*!
  230. \qmlproperty bool ComboBox::acceptableInput
  231. \since QtQuick.Controls 1.1
  232. Returns \c true if the combo box contains acceptable
  233. text in the editable text field.
  234. If a validator was set, this property will return \c
  235. true if the current text satisfies the validator or mask as
  236. a final string (not as an intermediate string).
  237. \sa validator, accepted
  238. */
  239. readonly property alias acceptableInput: input.acceptableInput
  240. /*!
  241. \qmlproperty bool ComboBox::selectByMouse
  242. \since QtQuick.Controls 1.3
  243. This property determines if the user can select the text in
  244. the editable text field with the mouse.
  245. The default value is \c true.
  246. */
  247. property bool selectByMouse: true
  248. /*!
  249. \qmlproperty bool ComboBox::inputMethodComposing
  250. \since QtQuick.Controls 1.3
  251. This property holds whether an editable ComboBox has partial text input from an input method.
  252. While it is composing an input method may rely on mouse or key events from the ComboBox
  253. to edit or commit the partial text. This property can be used to determine when to disable
  254. events handlers that may interfere with the correct operation of an input method.
  255. */
  256. readonly property bool inputMethodComposing: !!input.inputMethodComposing
  257. /*!
  258. \qmlsignal ComboBox::accepted()
  259. \since QtQuick.Controls 1.1
  260. This signal is emitted when the Return or Enter key is pressed on an
  261. \l editable combo box. If the confirmed string is not currently in the model,
  262. the currentIndex will be set to -1 and the \l currentText will be updated
  263. accordingly.
  264. \note If there is a \l validator set on the combobox,
  265. the signal will only be emitted if the input is in an acceptable state.
  266. The corresponding handler is \c onAccepted.
  267. */
  268. signal accepted
  269. /*!
  270. \qmlsignal ComboBox::activated(int index)
  271. \since QtQuick.Controls 1.1
  272. This signal is similar to currentIndex changed, but will only
  273. be emitted if the combo box index was changed by the user, not
  274. when set programmatically.
  275. \a index is the activated model index, or \c -1 if a new string is
  276. accepted.
  277. The corresponding handler is \c onActivated.
  278. */
  279. signal activated(int index)
  280. /*!
  281. \qmlmethod void ComboBox::selectAll()
  282. \since QtQuick.Controls 1.1
  283. Causes all \l editText to be selected.
  284. */
  285. function selectAll() {
  286. input.selectAll()
  287. }
  288. /*! \internal */
  289. function __selectPrevItem() {
  290. input.blockUpdate = true
  291. if (currentIndex > 0) {
  292. currentIndex--;
  293. input.text = popup.currentText;
  294. activated(currentIndex);
  295. }
  296. input.blockUpdate = false;
  297. }
  298. /*! \internal */
  299. function __selectNextItem() {
  300. input.blockUpdate = true;
  301. if (currentIndex < popupItems.count - 1) {
  302. currentIndex++;
  303. input.text = popup.currentText;
  304. activated(currentIndex);
  305. }
  306. input.blockUpdate = false;
  307. }
  308. /*! \internal */
  309. property var __popup: popup
  310. style: Settings.styleComponent(Settings.style, "ComboBoxStyle.qml", comboBox)
  311. activeFocusOnTab: true
  312. Accessible.name: editable ? editText : currentText
  313. Accessible.role: Accessible.ComboBox
  314. Accessible.editable: editable
  315. MouseArea {
  316. id: mouseArea
  317. property bool overridePressed: false
  318. readonly property bool effectivePressed: (pressed || overridePressed) && containsMouse
  319. anchors.fill: parent
  320. hoverEnabled: Settings.hoverEnabled
  321. onPressed: {
  322. if (comboBox.activeFocusOnPress)
  323. forceActiveFocus()
  324. if (!Settings.hasTouchScreen)
  325. popup.toggleShow()
  326. else
  327. overridePressed = true
  328. }
  329. onCanceled: overridePressed = false
  330. onClicked: {
  331. if (Settings.hasTouchScreen)
  332. popup.toggleShow()
  333. overridePressed = false
  334. }
  335. onWheel: {
  336. if (wheel.angleDelta.y > 0) {
  337. __selectPrevItem();
  338. } else if (wheel.angleDelta.y < 0){
  339. __selectNextItem();
  340. }
  341. }
  342. }
  343. Component.onCompleted: {
  344. if (currentIndex === -1)
  345. currentIndex = 0
  346. popup.ready = true
  347. popup.resolveTextValue(textRole)
  348. }
  349. Keys.onPressed: {
  350. // Perform one-character based lookup for non-editable combo box
  351. if (!editable && event.text.length > 0) {
  352. var index = input.find(event.text, Qt.MatchStartsWith);
  353. if (index >= 0 && index !== currentIndex) {
  354. currentIndex = index;
  355. activated(currentIndex);
  356. }
  357. }
  358. }
  359. TextInputWithHandles {
  360. id: input
  361. visible: editable
  362. enabled: editable
  363. focus: true
  364. clip: contentWidth > width
  365. control: comboBox
  366. cursorHandle: __style ? __style.__cursorHandle : undefined
  367. selectionHandle: __style ? __style.__selectionHandle : undefined
  368. anchors.fill: parent
  369. anchors.leftMargin: __style ? __style.padding.left : 0
  370. anchors.topMargin: __style ? __style.padding.top : 0
  371. anchors.rightMargin: __style ? __panel.dropDownButtonWidth + __style.padding.right : 0
  372. anchors.bottomMargin: __style ? __style.padding.bottom: 0
  373. verticalAlignment: Text.AlignVCenter
  374. font: __panel && __panel.font !== undefined ? __panel.font : TextSingleton.font
  375. renderType: __style ? __style.renderType : Text.NativeRendering
  376. color: __panel ? __panel.textColor : "black"
  377. selectionColor: __panel ? __panel.selectionColor : "blue"
  378. selectedTextColor: __panel ? __panel.selectedTextColor : "white"
  379. onAccepted: {
  380. var idx = input.find(editText, Qt.MatchFixedString)
  381. if (idx > -1) {
  382. editTextMatches = true;
  383. currentIndex = idx;
  384. editText = textAt(idx);
  385. } else {
  386. editTextMatches = false;
  387. currentIndex = -1;
  388. popup.currentText = editText;
  389. }
  390. comboBox.accepted();
  391. }
  392. property bool blockUpdate: false
  393. property string prevText
  394. property bool editTextMatches: true
  395. function find (text, searchType) {
  396. for (var i = 0 ; i < popupItems.count ; ++i) {
  397. var currentString = popupItems.objectAt(i).text
  398. if (searchType === Qt.MatchExactly) {
  399. if (text === currentString)
  400. return i;
  401. } else if (searchType === Qt.CaseSensitive) {
  402. if (currentString.indexOf(text) === 0)
  403. return i;
  404. } else if (searchType === Qt.MatchFixedString) {
  405. if (currentString.toLowerCase().indexOf(text.toLowerCase()) === 0
  406. && currentString.length === text.length)
  407. return i;
  408. } else if (currentString.toLowerCase().indexOf(text.toLowerCase()) === 0) {
  409. return i
  410. }
  411. }
  412. return -1;
  413. }
  414. // Finds first entry and shortest entry. Used by editable combo
  415. function tryComplete (inputText) {
  416. var candidate = "";
  417. var shortestString = "";
  418. for (var i = 0 ; i < popupItems.count ; ++i) {
  419. var currentString = popupItems.objectAt(i).text;
  420. if (currentString.toLowerCase().indexOf(inputText.toLowerCase()) === 0) {
  421. if (candidate.length) { // Find smallest possible match
  422. var cmp = 0;
  423. // We try to complete the shortest string that matches our search
  424. if (currentString.length < candidate.length)
  425. candidate = currentString
  426. while (cmp < Math.min(currentString.length, shortestString.length)
  427. && shortestString[cmp].toLowerCase() === currentString[cmp].toLowerCase())
  428. cmp++;
  429. shortestString = shortestString.substring(0, cmp);
  430. } else { // First match, select as current index and find other matches
  431. candidate = currentString;
  432. shortestString = currentString;
  433. }
  434. }
  435. }
  436. if (candidate.length)
  437. return inputText + candidate.substring(inputText.length, candidate.length);
  438. return inputText;
  439. }
  440. property bool allowComplete: false
  441. Keys.forwardTo: comboBox
  442. Keys.onPressed: allowComplete = (event.key !== Qt.Key_Backspace && event.key !== Qt.Key_Delete);
  443. onTextChanged: {
  444. if (editable && !blockUpdate && allowComplete && text.length > 0) {
  445. var completed = input.tryComplete(text)
  446. if (completed.length > text.length) {
  447. var oldtext = input.text;
  448. input.text = completed;
  449. input.select(text.length, oldtext.length);
  450. }
  451. }
  452. prevText = text
  453. }
  454. }
  455. Binding {
  456. target: input
  457. property: "text"
  458. value: popup.currentText
  459. when: input.editTextMatches
  460. }
  461. onTextRoleChanged: popup.resolveTextValue(textRole)
  462. ExclusiveGroup { id: eg }
  463. Menu {
  464. id: popup
  465. objectName: "popup"
  466. style: isPopup ? __style.__popupStyle : __style.__dropDownStyle
  467. property string currentText: selectedText
  468. onSelectedTextChanged: popup.currentText = selectedText
  469. property string selectedText
  470. on__SelectedIndexChanged: {
  471. if (__selectedIndex === -1)
  472. popup.currentText = ""
  473. else
  474. updateSelectedText()
  475. }
  476. property string textRole: ""
  477. property bool ready: false
  478. property bool isPopup: !editable && !!__panel && __panel.popup
  479. property int y: isPopup ? (comboBox.__panel.height - comboBox.__panel.implicitHeight) / 2.0 : comboBox.__panel.height
  480. __minimumWidth: comboBox.width
  481. __visualItem: comboBox
  482. property bool modelIsArray: false
  483. Instantiator {
  484. id: popupItems
  485. active: false
  486. property bool updatingModel: false
  487. onModelChanged: {
  488. popup.modelIsArray = !!model ? model.constructor === Array : false
  489. if (active) {
  490. if (updatingModel && popup.__selectedIndex === 0) {
  491. // We still want to update the currentText
  492. popup.updateSelectedText()
  493. } else {
  494. updatingModel = true
  495. popup.__selectedIndex = 0
  496. }
  497. }
  498. popup.resolveTextValue(comboBox.textRole)
  499. }
  500. MenuItem {
  501. text: popup.textRole === '' ?
  502. modelData :
  503. ((popup.modelIsArray ? modelData[popup.textRole] : model[popup.textRole]) || '')
  504. onTriggered: {
  505. if (index !== currentIndex)
  506. activated(index)
  507. comboBox.editText = text
  508. }
  509. checkable: true
  510. exclusiveGroup: eg
  511. }
  512. onObjectAdded: {
  513. popup.insertItem(index, object)
  514. if (!updatingModel && index === popup.__selectedIndex)
  515. popup.selectedText = object["text"]
  516. }
  517. onObjectRemoved: popup.removeItem(object)
  518. }
  519. function resolveTextValue(initialTextRole) {
  520. if (!ready || !model) {
  521. popupItems.active = false
  522. return;
  523. }
  524. var get = model['get'];
  525. if (!get && popup.modelIsArray && !!model[0]) {
  526. if (model[0].constructor !== String && model[0].constructor !== Number)
  527. get = function(i) { return model[i]; }
  528. }
  529. var modelMayHaveRoles = get !== undefined
  530. textRole = initialTextRole
  531. if (textRole === "" && modelMayHaveRoles && get(0)) {
  532. // No text role set, check whether model has a suitable role
  533. // If 'text' is found, or there's only one role, pick that.
  534. var listElement = get(0)
  535. var roleName = ""
  536. var roleCount = 0
  537. for (var role in listElement) {
  538. if (listElement[role].constructor === Function)
  539. continue;
  540. if (role === "text") {
  541. roleName = role
  542. break
  543. } else if (!roleName) {
  544. roleName = role
  545. }
  546. ++roleCount
  547. }
  548. if (roleCount > 1 && roleName !== "text") {
  549. console.warn("No suitable 'textRole' found for ComboBox.")
  550. } else {
  551. textRole = roleName
  552. }
  553. }
  554. if (!popupItems.active)
  555. popupItems.active = true
  556. else
  557. updateSelectedText()
  558. }
  559. function toggleShow() {
  560. if (popup.__popupVisible) {
  561. popup.__dismissAndDestroy()
  562. } else {
  563. if (items[__selectedIndex])
  564. items[__selectedIndex].checked = true
  565. __currentIndex = comboBox.currentIndex
  566. if (Qt.application.layoutDirection === Qt.RightToLeft)
  567. __popup(Qt.rect(comboBox.width, y, 0, 0), isPopup ? __selectedIndex : 0)
  568. else
  569. __popup(Qt.rect(0, y, 0, 0), isPopup ? __selectedIndex : 0)
  570. }
  571. }
  572. function updateSelectedText() {
  573. var selectedItem;
  574. if (__selectedIndex !== -1 && (selectedItem = items[__selectedIndex])) {
  575. input.editTextMatches = true
  576. selectedText = Qt.binding(function () { return selectedItem.text })
  577. if (currentText !== selectedText) // __selectedIndex went form -1 to 0
  578. selectedTextChanged()
  579. }
  580. }
  581. }
  582. // The key bindings below will only be in use when popup is
  583. // not visible. Otherwise, native popup key handling will take place:
  584. Keys.onSpacePressed: {
  585. if (!editable)
  586. popup.toggleShow()
  587. else
  588. event.accepted = false
  589. }
  590. Keys.onUpPressed: __selectPrevItem()
  591. Keys.onDownPressed: __selectNextItem()
  592. }