MenuBar.qml 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  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.Styles 1.1
  42. import QtQuick.Controls.Private 1.0
  43. /*!
  44. \qmltype MenuBar
  45. \inqmlmodule QtQuick.Controls
  46. \since 5.1
  47. \ingroup applicationwindow
  48. \ingroup controls
  49. \brief Provides a horizontal menu bar.
  50. \image menubar.png
  51. MenuBar can be added to an \l ApplicationWindow, providing menu options
  52. to access additional functionality of the application.
  53. \code
  54. ApplicationWindow {
  55. ...
  56. menuBar: MenuBar {
  57. Menu {
  58. title: "File"
  59. MenuItem { text: "Open..." }
  60. MenuItem { text: "Close" }
  61. }
  62. Menu {
  63. title: "Edit"
  64. MenuItem { text: "Cut" }
  65. MenuItem { text: "Copy" }
  66. MenuItem { text: "Paste" }
  67. }
  68. }
  69. }
  70. \endcode
  71. \sa ApplicationWindow::menuBar
  72. */
  73. MenuBarPrivate {
  74. id: root
  75. /*! \qmlproperty Component MenuBar::style
  76. \since QtQuick.Controls.Styles 1.2
  77. The style Component for this control.
  78. \sa {MenuBarStyle}
  79. */
  80. property Component style: Settings.styleComponent(Settings.style, "MenuBarStyle.qml", root)
  81. /*! \internal */
  82. property QtObject __style: styleLoader.item
  83. __isNative: !__style.hasOwnProperty("__isNative") || __style.__isNative
  84. /*! \internal */
  85. __contentItem: Loader {
  86. id: topLoader
  87. sourceComponent: __menuBarComponent
  88. active: !root.__isNative
  89. focus: true
  90. Keys.forwardTo: [item]
  91. property real preferredWidth: parent && active ? parent.width : 0
  92. property bool altPressed: item ? item.__altPressed : false
  93. Loader {
  94. id: styleLoader
  95. property alias __control: topLoader.item
  96. sourceComponent: root.style
  97. onStatusChanged: {
  98. if (status === Loader.Error)
  99. console.error("Failed to load Style for", root)
  100. }
  101. }
  102. }
  103. /*! \internal */
  104. property Component __menuBarComponent: Loader {
  105. id: menuBarLoader
  106. Accessible.role: Accessible.MenuBar
  107. onStatusChanged: if (status === Loader.Error) console.error("Failed to load panel for", root)
  108. visible: status === Loader.Ready
  109. sourceComponent: d.style ? d.style.background : undefined
  110. width: implicitWidth || root.__contentItem.preferredWidth
  111. height: Math.max(row.height + d.heightPadding, item ? item.implicitHeight : 0)
  112. Binding {
  113. // Make sure the styled menu bar is in the background
  114. target: menuBarLoader.item
  115. property: "z"
  116. value: menuMouseArea.z - 1
  117. }
  118. QtObject {
  119. id: d
  120. property Style style: __style
  121. property int openedMenuIndex: -1
  122. property bool preselectMenuItem: false
  123. property real heightPadding: style ? style.padding.top + style.padding.bottom : 0
  124. property bool altPressed: false
  125. property bool altPressedAgain: false
  126. property var mnemonicsMap: ({})
  127. function openMenuAtIndex(index) {
  128. if (openedMenuIndex === index)
  129. return;
  130. var oldIndex = openedMenuIndex
  131. openedMenuIndex = index
  132. if (oldIndex !== -1) {
  133. var menu = root.menus[oldIndex]
  134. if (menu.__popupVisible)
  135. menu.__dismissAndDestroy()
  136. }
  137. if (openedMenuIndex !== -1) {
  138. menu = root.menus[openedMenuIndex]
  139. if (menu.enabled) {
  140. if (menu.__usingDefaultStyle)
  141. menu.style = d.style.menuStyle
  142. var xPos = row.LayoutMirroring.enabled ? menuItemLoader.width : 0
  143. menu.__popup(Qt.rect(xPos, menuBarLoader.height - d.heightPadding, 0, 0), 0)
  144. if (preselectMenuItem)
  145. menu.__currentIndex = 0
  146. }
  147. }
  148. }
  149. function dismissActiveFocus(event, reason) {
  150. if (reason) {
  151. altPressedAgain = false
  152. altPressed = false
  153. openMenuAtIndex(-1)
  154. root.__contentItem.parent.forceActiveFocus()
  155. } else {
  156. event.accepted = false
  157. }
  158. }
  159. function maybeOpenFirstMenu(event) {
  160. if (altPressed && openedMenuIndex === -1) {
  161. preselectMenuItem = true
  162. openMenuAtIndex(0)
  163. } else {
  164. event.accepted = false
  165. }
  166. }
  167. }
  168. property alias __altPressed: d.altPressed // Needed for the menu contents
  169. focus: true
  170. Keys.onPressed: {
  171. var action = null
  172. if (event.key === Qt.Key_Alt) {
  173. if (!d.altPressed)
  174. d.altPressed = true
  175. else
  176. d.altPressedAgain = true
  177. } else if (d.altPressed && (action = d.mnemonicsMap[event.text.toUpperCase()])) {
  178. d.preselectMenuItem = true
  179. action.trigger()
  180. event.accepted = true
  181. }
  182. }
  183. Keys.onReleased: d.dismissActiveFocus(event, d.altPressedAgain && d.openedMenuIndex === -1)
  184. Keys.onEscapePressed: d.dismissActiveFocus(event, d.openedMenuIndex === -1)
  185. Keys.onUpPressed: d.maybeOpenFirstMenu(event)
  186. Keys.onDownPressed: d.maybeOpenFirstMenu(event)
  187. Keys.onLeftPressed: {
  188. if (d.openedMenuIndex > 0) {
  189. var idx = d.openedMenuIndex - 1
  190. while (idx >= 0 && !(root.menus[idx].enabled && root.menus[idx].visible))
  191. idx--
  192. if (idx >= 0) {
  193. d.preselectMenuItem = true
  194. d.openMenuAtIndex(idx)
  195. }
  196. } else {
  197. event.accepted = false;
  198. }
  199. }
  200. Keys.onRightPressed: {
  201. if (d.openedMenuIndex !== -1 && d.openedMenuIndex < root.menus.length - 1) {
  202. var idx = d.openedMenuIndex + 1
  203. while (idx < root.menus.length && !(root.menus[idx].enabled && root.menus[idx].visible))
  204. idx++
  205. if (idx < root.menus.length) {
  206. d.preselectMenuItem = true
  207. d.openMenuAtIndex(idx)
  208. }
  209. } else {
  210. event.accepted = false;
  211. }
  212. }
  213. Keys.forwardTo: d.openedMenuIndex !== -1 ? [root.menus[d.openedMenuIndex].__contentItem] : []
  214. Row {
  215. id: row
  216. x: d.style ? d.style.padding.left : 0
  217. y: d.style ? d.style.padding.top : 0
  218. width: parent.width - (d.style ? d.style.padding.left + d.style.padding.right : 0)
  219. LayoutMirroring.enabled: Qt.application.layoutDirection === Qt.RightToLeft
  220. Repeater {
  221. id: itemsRepeater
  222. model: root.menus
  223. Loader {
  224. id: menuItemLoader
  225. Accessible.role: Accessible.MenuItem
  226. Accessible.name: StyleHelpers.removeMnemonics(opts.text)
  227. Accessible.onPressAction: d.openMenuAtIndex(opts.index)
  228. property var styleData: QtObject {
  229. id: opts
  230. readonly property int index: __menuItemIndex
  231. readonly property string text: !!__menuItem && __menuItem.title
  232. readonly property bool enabled: !!__menuItem && __menuItem.enabled
  233. readonly property bool selected: menuMouseArea.hoveredItem === menuItemLoader
  234. readonly property bool open: !!__menuItem && __menuItem.__popupVisible || d.openedMenuIndex === index
  235. readonly property bool underlineMnemonic: d.altPressed
  236. }
  237. height: Math.max(menuBarLoader.height - d.heightPadding,
  238. menuItemLoader.item ? menuItemLoader.item.implicitHeight : 0)
  239. readonly property var __menuItem: modelData
  240. readonly property int __menuItemIndex: index
  241. sourceComponent: d.style ? d.style.itemDelegate : null
  242. visible: __menuItem.visible
  243. Connections {
  244. target: __menuItem
  245. onAboutToHide: {
  246. if (d.openedMenuIndex === index) {
  247. d.openMenuAtIndex(-1)
  248. menuMouseArea.hoveredItem = null
  249. }
  250. }
  251. }
  252. Connections {
  253. target: __menuItem.__action
  254. onTriggered: d.openMenuAtIndex(__menuItemIndex)
  255. }
  256. Component.onCompleted: {
  257. __menuItem.__visualItem = menuItemLoader
  258. var title = __menuItem.title
  259. var ampersandPos = title.indexOf("&")
  260. if (ampersandPos !== -1)
  261. d.mnemonicsMap[title[ampersandPos + 1].toUpperCase()] = __menuItem.__action
  262. }
  263. }
  264. }
  265. }
  266. MouseArea {
  267. id: menuMouseArea
  268. anchors.fill: parent
  269. hoverEnabled: Settings.hoverEnabled
  270. onPositionChanged: updateCurrentItem(mouse)
  271. onPressed: updateCurrentItem(mouse)
  272. onExited: hoveredItem = null
  273. property Item currentItem: null
  274. property Item hoveredItem: null
  275. function updateCurrentItem(mouse) {
  276. var pos = mapToItem(row, mouse.x, mouse.y)
  277. if (pressed || !hoveredItem
  278. || !hoveredItem.contains(Qt.point(pos.x - currentItem.x, pos.y - currentItem.y))) {
  279. hoveredItem = row.childAt(pos.x, pos.y)
  280. if (!hoveredItem)
  281. return false;
  282. currentItem = hoveredItem
  283. if (pressed || d.openedMenuIndex !== -1) {
  284. d.preselectMenuItem = false
  285. d.openMenuAtIndex(currentItem.__menuItemIndex)
  286. }
  287. }
  288. return true;
  289. }
  290. }
  291. }
  292. }