/**************************************************************************** ** ** Copyright (C) 2017 The Qt Company Ltd. ** Contact: http://www.qt.io/licensing/ ** ** This file is part of the Qt Labs Platform module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL3$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see http://www.qt.io/terms-conditions. For further ** information use the contact form at http://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPLv3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or later as published by the Free ** Software Foundation and appearing in the file LICENSE.GPL included in ** the packaging of this file. Please review the following information to ** ensure the GNU General Public License version 2.0 requirements will be ** met: http://www.gnu.org/licenses/gpl-2.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qquickplatformmenuitem_p.h" #include "qquickplatformmenu_p.h" #include "qquickplatformmenuitemgroup_p.h" #include "qquickplatformiconloader_p.h" #include #include #include #include #include "widgets/qwidgetplatform_p.h" QT_BEGIN_NAMESPACE /*! \qmltype MenuItem \inherits QtObject \instantiates QQuickPlatformMenuItem \inqmlmodule Qt.labs.platform \since 5.8 \brief A native menu item. The MenuItem type provides a QML API for native platform menu items. \image qtlabsplatform-menu.png A menu item consists of an \l {iconSource}{icon}, \l text, and \l shortcut. \code Menu { id: zoomMenu MenuItem { text: qsTr("Zoom In") shortcut: StandardKey.ZoomIn onTriggered: zoomIn() } MenuItem { text: qsTr("Zoom Out") shortcut: StandardKey.ZoomOut onTriggered: zoomOut() } } \endcode \labs \sa Menu, MenuItemGroup */ /*! \qmlsignal Qt.labs.platform::MenuItem::triggered() This signal is emitted when the menu item is triggered by the user. */ /*! \qmlsignal Qt.labs.platform::MenuItem::hovered() This signal is emitted when the menu item is hovered by the user. */ QQuickPlatformMenuItem::QQuickPlatformMenuItem(QObject *parent) : QObject(parent), m_complete(false), m_enabled(true), m_visible(true), m_separator(false), m_checkable(false), m_checked(false), m_role(QPlatformMenuItem::TextHeuristicRole), m_menu(nullptr), m_subMenu(nullptr), m_group(nullptr), m_iconLoader(nullptr), m_handle(nullptr) { } QQuickPlatformMenuItem::~QQuickPlatformMenuItem() { if (m_menu) m_menu->removeItem(this); if (m_group) m_group->removeItem(this); delete m_iconLoader; m_iconLoader = nullptr; delete m_handle; m_handle = nullptr; } QPlatformMenuItem *QQuickPlatformMenuItem::handle() const { return m_handle; } QPlatformMenuItem *QQuickPlatformMenuItem::create() { if (!m_handle && m_menu && m_menu->handle()) { m_handle = m_menu->handle()->createMenuItem(); // TODO: implement QCocoaMenu::createMenuItem() if (!m_handle) m_handle = QGuiApplicationPrivate::platformTheme()->createPlatformMenuItem(); if (!m_handle) m_handle = QWidgetPlatform::createMenuItem(); if (m_handle) { connect(m_handle, &QPlatformMenuItem::activated, this, &QQuickPlatformMenuItem::triggered); connect(m_handle, &QPlatformMenuItem::hovered, this, &QQuickPlatformMenuItem::hovered); if (m_checkable) connect(m_handle, &QPlatformMenuItem::activated, this, &QQuickPlatformMenuItem::toggle); } } return m_handle; } void QQuickPlatformMenuItem::sync() { if (!m_complete || !create()) return; m_handle->setEnabled(isEnabled()); m_handle->setVisible(isVisible()); m_handle->setIsSeparator(m_separator); m_handle->setCheckable(m_checkable); m_handle->setChecked(m_checked); m_handle->setRole(m_role); m_handle->setText(m_text); m_handle->setFont(m_font); m_handle->setHasExclusiveGroup(m_group && m_group->isExclusive()); if (m_subMenu && m_subMenu->handle()) m_handle->setMenu(m_subMenu->handle()); #if QT_CONFIG(shortcut) QKeySequence sequence; if (m_shortcut.type() == QVariant::Int) sequence = QKeySequence(static_cast(m_shortcut.toInt())); else sequence = QKeySequence::fromString(m_shortcut.toString()); m_handle->setShortcut(sequence.toString()); #endif if (m_menu && m_menu->handle()) m_menu->handle()->syncMenuItem(m_handle); } /*! \readonly \qmlproperty Menu Qt.labs.platform::MenuItem::menu This property holds the menu that the item belongs to, or \c null if the item is not in a menu. */ QQuickPlatformMenu *QQuickPlatformMenuItem::menu() const { return m_menu; } void QQuickPlatformMenuItem::setMenu(QQuickPlatformMenu *menu) { if (m_menu == menu) return; m_menu = menu; emit menuChanged(); } /*! \readonly \qmlproperty Menu Qt.labs.platform::MenuItem::subMenu This property holds the sub-menu that the item contains, or \c null if the item is not a sub-menu item. */ QQuickPlatformMenu *QQuickPlatformMenuItem::subMenu() const { return m_subMenu; } void QQuickPlatformMenuItem::setSubMenu(QQuickPlatformMenu *menu) { if (m_subMenu == menu) return; m_subMenu = menu; sync(); emit subMenuChanged(); } /*! \qmlproperty MenuItemGroup Qt.labs.platform::MenuItem::group This property holds the group that the item belongs to, or \c null if the item is not in a group. */ QQuickPlatformMenuItemGroup *QQuickPlatformMenuItem::group() const { return m_group; } void QQuickPlatformMenuItem::setGroup(QQuickPlatformMenuItemGroup *group) { if (m_group == group) return; bool wasEnabled = isEnabled(); bool wasVisible = isVisible(); if (group) group->addItem(this); m_group = group; sync(); emit groupChanged(); if (isEnabled() != wasEnabled) emit enabledChanged(); if (isVisible() != wasVisible) emit visibleChanged(); } /*! \qmlproperty bool Qt.labs.platform::MenuItem::enabled This property holds whether the item is enabled. The default value is \c true. Disabled items cannot be triggered by the user. They do not disappear from menus, but they are displayed in a way which indicates that they are unavailable. For example, they might be displayed using only shades of gray. When an item is disabled, it is not possible to trigger it through its \l shortcut. */ bool QQuickPlatformMenuItem::isEnabled() const { return m_enabled && (!m_group || m_group->isEnabled()); } void QQuickPlatformMenuItem::setEnabled(bool enabled) { if (m_enabled == enabled) return; bool wasEnabled = isEnabled(); m_enabled = enabled; sync(); if (isEnabled() != wasEnabled) emit enabledChanged(); } /*! \qmlproperty bool Qt.labs.platform::MenuItem::visible This property holds whether the item is visible. The default value is \c true. */ bool QQuickPlatformMenuItem::isVisible() const { return m_visible && (!m_group || m_group->isVisible()); } void QQuickPlatformMenuItem::setVisible(bool visible) { if (m_visible == visible) return; bool wasVisible = isVisible(); m_visible = visible; sync(); if (isVisible() != wasVisible) emit visibleChanged(); } /*! \qmlproperty bool Qt.labs.platform::MenuItem::separator This property holds whether the item is a separator line. The default value is \c false. \sa MenuSeparator */ bool QQuickPlatformMenuItem::isSeparator() const { return m_separator; } void QQuickPlatformMenuItem::setSeparator(bool separator) { if (m_separator == separator) return; m_separator = separator; sync(); emit separatorChanged(); } /*! \qmlproperty bool Qt.labs.platform::MenuItem::checkable This property holds whether the item is checkable. A checkable menu item has an on/off state. For example, in a word processor, a "Bold" menu item may be either on or off. A menu item that is not checkable is a command item that is simply executed, e.g. file save. The default value is \c false. \sa checked, MenuItemGroup */ bool QQuickPlatformMenuItem::isCheckable() const { return m_checkable; } void QQuickPlatformMenuItem::setCheckable(bool checkable) { if (m_checkable == checkable) return; if (m_handle) { if (checkable) connect(m_handle, &QPlatformMenuItem::activated, this, &QQuickPlatformMenuItem::toggle); else disconnect(m_handle, &QPlatformMenuItem::activated, this, &QQuickPlatformMenuItem::toggle); } m_checkable = checkable; sync(); emit checkableChanged(); } /*! \qmlproperty bool Qt.labs.platform::MenuItem::checked This property holds whether the item is checked (on) or unchecked (off). The default value is \c false. \sa checkable, MenuItemGroup */ bool QQuickPlatformMenuItem::isChecked() const { return m_checked; } void QQuickPlatformMenuItem::setChecked(bool checked) { if (m_checked == checked) return; if (checked && !m_checkable) setCheckable(true); m_checked = checked; sync(); emit checkedChanged(); } /*! \qmlproperty enumeration Qt.labs.platform::MenuItem::role This property holds the role of the item. The role determines whether the item should be placed into the application menu on macOS. Available values: \value MenuItem.NoRole The item should not be put into the application menu \value MenuItem.TextHeuristicRole The item should be put in the application menu based on the action's text (default) \value MenuItem.ApplicationSpecificRole The item should be put in the application menu with an application-specific role \value MenuItem.AboutQtRole The item handles the "About Qt" menu item. \value MenuItem.AboutRole The item should be placed where the "About" menu item is in the application menu. The text of the menu item will be set to "About ". The application name is fetched from the \c{Info.plist} file in the application's bundle (See \l{Qt for macOS - Deployment}). \value MenuItem.PreferencesRole The item should be placed where the "Preferences..." menu item is in the application menu. \value MenuItem.QuitRole The item should be placed where the Quit menu item is in the application menu. Specifying the role only has effect on items that are in the immediate menus of a menubar, not in the submenus of those menus. For example, if you have a "File" menu in your menubar and the "File" menu has a submenu, specifying a role for the items in that submenu has no effect. They will never be moved to the application menu. */ QPlatformMenuItem::MenuRole QQuickPlatformMenuItem::role() const { return m_role; } void QQuickPlatformMenuItem::setRole(QPlatformMenuItem::MenuRole role) { if (m_role == role) return; m_role = role; sync(); emit roleChanged(); } /*! \qmlproperty string Qt.labs.platform::MenuItem::text This property holds the menu item's text. */ QString QQuickPlatformMenuItem::text() const { return m_text; } void QQuickPlatformMenuItem::setText(const QString &text) { if (m_text == text) return; m_text = text; sync(); emit textChanged(); } /*! \qmlproperty url Qt.labs.platform::MenuItem::iconSource This property holds the url of the menu item's icon. \code MenuItem { iconName: "edit-undo" iconSource: "qrc:/images/undo.png" } \endcode \sa iconName */ QUrl QQuickPlatformMenuItem::iconSource() const { if (!m_iconLoader) return QUrl(); return m_iconLoader->iconSource(); } void QQuickPlatformMenuItem::setIconSource(const QUrl& source) { if (source == iconSource()) return; iconLoader()->setIconSource(source); emit iconSourceChanged(); } /*! \qmlproperty string Qt.labs.platform::MenuItem::iconName This property holds the theme name of the menu item's icon. \code MenuItem { iconName: "edit-undo" iconSource: "qrc:/images/undo.png" } \endcode \sa iconSource, QIcon::fromTheme() */ QString QQuickPlatformMenuItem::iconName() const { if (!m_iconLoader) return QString(); return m_iconLoader->iconName(); } void QQuickPlatformMenuItem::setIconName(const QString& name) { if (name == iconName()) return; iconLoader()->setIconName(name); emit iconNameChanged(); } /*! \qmlproperty keysequence Qt.labs.platform::MenuItem::shortcut This property holds the menu item's shortcut. The shortcut key sequence can be set to one of the \l{QKeySequence::StandardKey}{standard keyboard shortcuts}, or it can be specified by a string containing a sequence of up to four key presses that are needed to \l{triggered}{trigger} the shortcut. The default value is an empty key sequence. \code MenuItem { shortcut: "Ctrl+E,Ctrl+W" onTriggered: edit.wrapMode = TextEdit.Wrap } \endcode */ QVariant QQuickPlatformMenuItem::shortcut() const { return m_shortcut; } void QQuickPlatformMenuItem::setShortcut(const QVariant& shortcut) { if (m_shortcut == shortcut) return; m_shortcut = shortcut; sync(); emit shortcutChanged(); } /*! \qmlproperty font Qt.labs.platform::MenuItem::font This property holds the menu item's font. \sa text */ QFont QQuickPlatformMenuItem::font() const { return m_font; } void QQuickPlatformMenuItem::setFont(const QFont& font) { if (m_font == font) return; m_font = font; sync(); emit fontChanged(); } /*! \qmlmethod void Qt.labs.platform::MenuItem::toggle() Toggles the \l checked state to its opposite state. */ void QQuickPlatformMenuItem::toggle() { if (m_checkable) setChecked(!m_checked); } void QQuickPlatformMenuItem::classBegin() { } void QQuickPlatformMenuItem::componentComplete() { if (m_handle && m_iconLoader) m_iconLoader->setEnabled(true); m_complete = true; sync(); } QQuickPlatformIconLoader *QQuickPlatformMenuItem::iconLoader() const { if (!m_iconLoader) { QQuickPlatformMenuItem *that = const_cast(this); static int slot = staticMetaObject.indexOfSlot("updateIcon()"); m_iconLoader = new QQuickPlatformIconLoader(slot, that); m_iconLoader->setEnabled(m_complete); } return m_iconLoader; } void QQuickPlatformMenuItem::updateIcon() { if (!m_handle || !m_iconLoader) return; m_handle->setIcon(m_iconLoader->icon()); sync(); } QT_END_NAMESPACE