2018-05-08 16:05:59 +02:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
|
|
/*
|
|
|
|
* This file is part of the LibreOffice project.
|
|
|
|
*
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
*/
|
|
|
|
|
2018-06-01 15:28:26 +02:00
|
|
|
#include <Qt5Menu.hxx>
|
2018-05-24 15:04:02 +02:00
|
|
|
#include <Qt5Menu.moc>
|
2019-05-04 00:03:15 +00:00
|
|
|
|
|
|
|
#include <Qt5Frame.hxx>
|
2019-02-25 17:00:21 +01:00
|
|
|
#include <Qt5Instance.hxx>
|
2019-05-04 00:03:15 +00:00
|
|
|
#include <Qt5MainWindow.hxx>
|
2018-05-08 16:05:59 +02:00
|
|
|
|
2019-05-04 00:03:15 +00:00
|
|
|
#include <QtWidgets/QMenuBar>
|
|
|
|
#include <QtWidgets/QPushButton>
|
2018-05-15 16:29:15 +02:00
|
|
|
|
2018-05-09 14:34:24 +02:00
|
|
|
#include <vcl/svapp.hxx>
|
2018-07-28 15:57:23 +02:00
|
|
|
#include <sal/log.hxx>
|
2019-05-04 00:03:15 +00:00
|
|
|
|
|
|
|
#include <strings.hrc>
|
|
|
|
#include <bitmaps.hlst>
|
2018-05-09 14:34:24 +02:00
|
|
|
|
2018-05-31 12:14:55 +02:00
|
|
|
Qt5Menu::Qt5Menu(bool bMenuBar)
|
|
|
|
: mpVCLMenu(nullptr)
|
|
|
|
, mpParentSalMenu(nullptr)
|
|
|
|
, mpFrame(nullptr)
|
|
|
|
, mbMenuBar(bMenuBar)
|
2018-11-22 12:27:26 +03:00
|
|
|
, mpQMenuBar(nullptr)
|
|
|
|
, mpQMenu(nullptr)
|
2019-05-06 17:18:56 +02:00
|
|
|
, mpCloseButton(nullptr)
|
2018-05-08 16:05:59 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-08-10 15:14:07 +02:00
|
|
|
bool Qt5Menu::VisibleMenuBar() { return true; }
|
2018-05-08 16:05:59 +02:00
|
|
|
|
2019-02-27 15:02:15 +03:00
|
|
|
void Qt5Menu::InsertMenuItem(Qt5MenuItem* pSalMenuItem, unsigned nPos)
|
2018-11-22 12:27:26 +03:00
|
|
|
{
|
|
|
|
sal_uInt16 nId = pSalMenuItem->mnId;
|
|
|
|
OUString aText = mpVCLMenu->GetItemText(nId);
|
|
|
|
NativeItemText(aText);
|
|
|
|
vcl::KeyCode nAccelKey = mpVCLMenu->GetAccelKey(nId);
|
|
|
|
|
2018-11-23 12:08:08 +03:00
|
|
|
pSalMenuItem->mpAction.reset();
|
|
|
|
pSalMenuItem->mpMenu.reset();
|
|
|
|
|
2018-11-22 12:55:06 +03:00
|
|
|
if (mbMenuBar)
|
|
|
|
{
|
2018-11-22 12:27:26 +03:00
|
|
|
// top-level menu
|
2018-11-22 12:55:06 +03:00
|
|
|
if (mpQMenuBar)
|
|
|
|
{
|
2019-02-27 15:02:15 +03:00
|
|
|
QMenu* pQMenu = new QMenu(toQString(aText), nullptr);
|
2018-11-23 12:08:08 +03:00
|
|
|
pSalMenuItem->mpMenu.reset(pQMenu);
|
|
|
|
|
2018-11-22 12:55:06 +03:00
|
|
|
if ((nPos != MENU_APPEND)
|
|
|
|
&& (static_cast<size_t>(nPos) < static_cast<size_t>(mpQMenuBar->actions().size())))
|
|
|
|
{
|
|
|
|
mpQMenuBar->insertMenu(mpQMenuBar->actions()[nPos], pQMenu);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-11-23 12:08:08 +03:00
|
|
|
mpQMenuBar->addMenu(pQMenu);
|
2018-11-22 12:55:06 +03:00
|
|
|
}
|
|
|
|
|
2019-02-27 15:02:15 +03:00
|
|
|
// correct parent menu for generated menu
|
|
|
|
if (pSalMenuItem->mpSubMenu)
|
|
|
|
{
|
|
|
|
pSalMenuItem->mpSubMenu->mpQMenu = pQMenu;
|
|
|
|
}
|
|
|
|
|
2018-11-22 12:55:06 +03:00
|
|
|
connect(pQMenu, &QMenu::aboutToShow, this,
|
|
|
|
[pSalMenuItem] { slotMenuAboutToShow(pSalMenuItem); });
|
|
|
|
connect(pQMenu, &QMenu::aboutToHide, this,
|
|
|
|
[pSalMenuItem] { slotMenuAboutToHide(pSalMenuItem); });
|
|
|
|
}
|
|
|
|
}
|
2019-02-27 15:02:15 +03:00
|
|
|
else if (mpQMenu)
|
2018-11-22 12:27:26 +03:00
|
|
|
{
|
|
|
|
if (pSalMenuItem->mpSubMenu)
|
|
|
|
{
|
|
|
|
// submenu
|
2019-02-27 15:02:15 +03:00
|
|
|
QMenu* pQMenu = new QMenu(toQString(aText), nullptr);
|
|
|
|
pSalMenuItem->mpMenu.reset(pQMenu);
|
2018-11-23 12:08:08 +03:00
|
|
|
|
2018-11-23 15:24:00 +03:00
|
|
|
if ((nPos != MENU_APPEND)
|
2019-02-27 15:02:15 +03:00
|
|
|
&& (static_cast<size_t>(nPos) < static_cast<size_t>(mpQMenu->actions().size())))
|
2018-11-23 15:24:00 +03:00
|
|
|
{
|
2019-02-27 15:02:15 +03:00
|
|
|
mpQMenu->insertMenu(mpQMenu->actions()[nPos], pQMenu);
|
2018-11-23 15:24:00 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-02-27 15:02:15 +03:00
|
|
|
mpQMenu->addMenu(pQMenu);
|
2018-11-23 15:24:00 +03:00
|
|
|
}
|
|
|
|
|
2019-02-27 15:02:15 +03:00
|
|
|
// correct parent menu for generated menu
|
|
|
|
pSalMenuItem->mpSubMenu->mpQMenu = pQMenu;
|
2019-01-11 16:39:55 +03:00
|
|
|
|
|
|
|
ReinitializeActionGroup(nPos);
|
|
|
|
|
|
|
|
// clear all action groups since menu is recreated
|
|
|
|
pSalMenuItem->mpSubMenu->ResetAllActionGroups();
|
2018-11-22 12:55:06 +03:00
|
|
|
|
|
|
|
connect(pQMenu, &QMenu::aboutToShow, this,
|
|
|
|
[pSalMenuItem] { slotMenuAboutToShow(pSalMenuItem); });
|
|
|
|
connect(pQMenu, &QMenu::aboutToHide, this,
|
|
|
|
[pSalMenuItem] { slotMenuAboutToHide(pSalMenuItem); });
|
2018-11-22 12:27:26 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (pSalMenuItem->mnType == MenuItemType::SEPARATOR)
|
2018-11-22 12:55:06 +03:00
|
|
|
{
|
2018-11-30 10:42:38 +03:00
|
|
|
QAction* pAction = new QAction(nullptr);
|
2018-11-23 12:08:08 +03:00
|
|
|
pSalMenuItem->mpAction.reset(pAction);
|
|
|
|
pAction->setSeparator(true);
|
|
|
|
|
2018-11-22 12:55:06 +03:00
|
|
|
if ((nPos != MENU_APPEND)
|
2019-02-27 15:02:15 +03:00
|
|
|
&& (static_cast<size_t>(nPos) < static_cast<size_t>(mpQMenu->actions().size())))
|
2018-11-22 12:55:06 +03:00
|
|
|
{
|
2019-02-27 15:02:15 +03:00
|
|
|
mpQMenu->insertAction(mpQMenu->actions()[nPos], pAction);
|
2018-11-22 12:55:06 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-02-27 15:02:15 +03:00
|
|
|
mpQMenu->addAction(pAction);
|
2018-11-22 12:55:06 +03:00
|
|
|
}
|
2019-01-11 16:39:55 +03:00
|
|
|
|
|
|
|
ReinitializeActionGroup(nPos);
|
2018-11-22 12:55:06 +03:00
|
|
|
}
|
2018-11-22 12:27:26 +03:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// leaf menu
|
2018-11-30 10:42:38 +03:00
|
|
|
QAction* pAction = new QAction(toQString(aText), nullptr);
|
2018-11-23 12:08:08 +03:00
|
|
|
pSalMenuItem->mpAction.reset(pAction);
|
2018-11-22 12:55:06 +03:00
|
|
|
|
|
|
|
if ((nPos != MENU_APPEND)
|
2019-02-27 15:02:15 +03:00
|
|
|
&& (static_cast<size_t>(nPos) < static_cast<size_t>(mpQMenu->actions().size())))
|
2018-11-22 12:55:06 +03:00
|
|
|
{
|
2019-02-27 15:02:15 +03:00
|
|
|
mpQMenu->insertAction(mpQMenu->actions()[nPos], pAction);
|
2018-11-22 12:55:06 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-02-27 15:02:15 +03:00
|
|
|
mpQMenu->addAction(pAction);
|
2018-11-22 12:55:06 +03:00
|
|
|
}
|
|
|
|
|
2019-01-11 16:39:55 +03:00
|
|
|
ReinitializeActionGroup(nPos);
|
|
|
|
|
2019-02-27 15:03:07 +03:00
|
|
|
UpdateActionGroupItem(pSalMenuItem);
|
2019-01-11 16:39:55 +03:00
|
|
|
|
2019-02-27 15:03:07 +03:00
|
|
|
pAction->setShortcut(toQString(nAccelKey.GetName(GetFrame()->GetWindow())));
|
2018-11-22 12:27:26 +03:00
|
|
|
|
|
|
|
connect(pAction, &QAction::triggered, this,
|
|
|
|
[pSalMenuItem] { slotMenuTriggered(pSalMenuItem); });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-25 18:06:11 +03:00
|
|
|
QAction* pAction = pSalMenuItem->getAction();
|
|
|
|
if (pAction)
|
|
|
|
{
|
|
|
|
pAction->setEnabled(pSalMenuItem->mbEnabled);
|
|
|
|
pAction->setVisible(pSalMenuItem->mbVisible);
|
|
|
|
}
|
2018-11-22 12:27:26 +03:00
|
|
|
}
|
|
|
|
|
2019-01-11 16:39:55 +03:00
|
|
|
void Qt5Menu::ReinitializeActionGroup(unsigned nPos)
|
|
|
|
{
|
|
|
|
const unsigned nCount = GetItemCount();
|
|
|
|
|
|
|
|
if (nCount == 0)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nPos == MENU_APPEND)
|
|
|
|
{
|
|
|
|
nPos = nCount - 1;
|
|
|
|
}
|
|
|
|
else if (nPos >= nCount)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Qt5MenuItem* pPrevItem = (nPos > 0) ? GetItemAtPos(nPos - 1) : nullptr;
|
|
|
|
Qt5MenuItem* pCurrentItem = GetItemAtPos(nPos);
|
|
|
|
Qt5MenuItem* pNextItem = (nPos < nCount - 1) ? GetItemAtPos(nPos + 1) : nullptr;
|
|
|
|
|
|
|
|
if (pCurrentItem->mnType == MenuItemType::SEPARATOR)
|
|
|
|
{
|
|
|
|
pCurrentItem->mpActionGroup.reset();
|
|
|
|
|
|
|
|
// if it's inserted into middle of existing group, split it into two groups:
|
|
|
|
// first goes original group, after separator goes new group
|
|
|
|
if (pPrevItem && pPrevItem->mpActionGroup && pNextItem && pNextItem->mpActionGroup
|
|
|
|
&& (pPrevItem->mpActionGroup == pNextItem->mpActionGroup))
|
|
|
|
{
|
|
|
|
std::shared_ptr<QActionGroup> pFirstActionGroup = pPrevItem->mpActionGroup;
|
|
|
|
std::shared_ptr<QActionGroup> pSecondActionGroup(new QActionGroup(nullptr));
|
|
|
|
pSecondActionGroup->setExclusive(true);
|
|
|
|
|
|
|
|
auto actions = pFirstActionGroup->actions();
|
|
|
|
|
|
|
|
for (unsigned idx = nPos + 1; idx < nCount; ++idx)
|
|
|
|
{
|
|
|
|
Qt5MenuItem* pModifiedItem = GetItemAtPos(idx);
|
|
|
|
|
|
|
|
if ((!pModifiedItem) || (!pModifiedItem->mpActionGroup))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
pModifiedItem->mpActionGroup = pSecondActionGroup;
|
|
|
|
auto action = pModifiedItem->getAction();
|
|
|
|
|
|
|
|
if (actions.contains(action))
|
|
|
|
{
|
|
|
|
pFirstActionGroup->removeAction(action);
|
|
|
|
pSecondActionGroup->addAction(action);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!pCurrentItem->mpActionGroup)
|
|
|
|
{
|
|
|
|
// unless element is inserted between two separators, or a separator and an end of vector, use neighbouring group since it's shared
|
|
|
|
if (pPrevItem && pPrevItem->mpActionGroup)
|
|
|
|
{
|
|
|
|
pCurrentItem->mpActionGroup = pPrevItem->mpActionGroup;
|
|
|
|
}
|
|
|
|
else if (pNextItem && pNextItem->mpActionGroup)
|
|
|
|
{
|
|
|
|
pCurrentItem->mpActionGroup = pNextItem->mpActionGroup;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pCurrentItem->mpActionGroup.reset(new QActionGroup(nullptr));
|
|
|
|
pCurrentItem->mpActionGroup->setExclusive(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if there's also a different group after this element, merge it
|
|
|
|
if (pNextItem && pNextItem->mpActionGroup
|
|
|
|
&& (pCurrentItem->mpActionGroup != pNextItem->mpActionGroup))
|
|
|
|
{
|
|
|
|
auto pFirstCheckedAction = pCurrentItem->mpActionGroup->checkedAction();
|
|
|
|
auto pSecondCheckedAction = pNextItem->mpActionGroup->checkedAction();
|
|
|
|
auto actions = pNextItem->mpActionGroup->actions();
|
|
|
|
|
|
|
|
// first move all actions from second group to first one, and if first group already has checked action,
|
|
|
|
// and second group also has a checked action, uncheck action from second group
|
|
|
|
for (auto action : actions)
|
|
|
|
{
|
|
|
|
pNextItem->mpActionGroup->removeAction(action);
|
|
|
|
|
|
|
|
if (pFirstCheckedAction && pSecondCheckedAction && (action == pSecondCheckedAction))
|
|
|
|
{
|
|
|
|
action->setChecked(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
pCurrentItem->mpActionGroup->addAction(action);
|
|
|
|
}
|
|
|
|
|
|
|
|
// now replace all pointers to second group with pointers to first group
|
|
|
|
for (unsigned idx = nPos + 1; idx < nCount; ++idx)
|
|
|
|
{
|
|
|
|
Qt5MenuItem* pModifiedItem = GetItemAtPos(idx);
|
|
|
|
|
|
|
|
if ((!pModifiedItem) || (!pModifiedItem->mpActionGroup))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
pModifiedItem->mpActionGroup = pCurrentItem->mpActionGroup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Qt5Menu::ResetAllActionGroups()
|
|
|
|
{
|
|
|
|
for (unsigned nItem = 0; nItem < GetItemCount(); ++nItem)
|
|
|
|
{
|
|
|
|
Qt5MenuItem* pSalMenuItem = GetItemAtPos(nItem);
|
|
|
|
pSalMenuItem->mpActionGroup.reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-27 15:03:07 +03:00
|
|
|
void Qt5Menu::UpdateActionGroupItem(Qt5MenuItem* pSalMenuItem)
|
|
|
|
{
|
|
|
|
QAction* pAction = pSalMenuItem->getAction();
|
|
|
|
if (!pAction)
|
|
|
|
return;
|
|
|
|
|
|
|
|
bool bChecked = mpVCLMenu->IsItemChecked(pSalMenuItem->mnId);
|
|
|
|
MenuItemBits itemBits = mpVCLMenu->GetItemBits(pSalMenuItem->mnId);
|
|
|
|
|
|
|
|
if (itemBits & MenuItemBits::RADIOCHECK)
|
|
|
|
{
|
|
|
|
pAction->setCheckable(true);
|
|
|
|
|
|
|
|
if (pSalMenuItem->mpActionGroup)
|
|
|
|
{
|
|
|
|
pSalMenuItem->mpActionGroup->addAction(pAction);
|
|
|
|
}
|
|
|
|
|
|
|
|
pAction->setChecked(bChecked);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pAction->setActionGroup(nullptr);
|
|
|
|
|
|
|
|
if (itemBits & MenuItemBits::CHECKABLE)
|
|
|
|
{
|
|
|
|
pAction->setCheckable(true);
|
|
|
|
pAction->setChecked(bChecked);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pAction->setChecked(false);
|
|
|
|
pAction->setCheckable(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-31 12:14:55 +02:00
|
|
|
void Qt5Menu::InsertItem(SalMenuItem* pSalMenuItem, unsigned nPos)
|
2018-05-08 16:05:59 +02:00
|
|
|
{
|
2018-05-09 14:34:24 +02:00
|
|
|
SolarMutexGuard aGuard;
|
2018-05-31 12:14:55 +02:00
|
|
|
Qt5MenuItem* pItem = static_cast<Qt5MenuItem*>(pSalMenuItem);
|
2018-05-09 14:34:24 +02:00
|
|
|
|
2018-05-31 12:14:55 +02:00
|
|
|
if (nPos == MENU_APPEND)
|
|
|
|
maItems.push_back(pItem);
|
2018-05-09 14:34:24 +02:00
|
|
|
else
|
2018-05-31 12:14:55 +02:00
|
|
|
maItems.insert(maItems.begin() + nPos, pItem);
|
2018-05-09 14:34:24 +02:00
|
|
|
|
|
|
|
pItem->mpParentMenu = this;
|
2018-11-22 12:55:06 +03:00
|
|
|
|
|
|
|
InsertMenuItem(pItem, nPos);
|
2018-05-08 16:05:59 +02:00
|
|
|
}
|
|
|
|
|
2018-05-31 12:14:55 +02:00
|
|
|
void Qt5Menu::RemoveItem(unsigned nPos)
|
2018-05-08 16:05:59 +02:00
|
|
|
{
|
2018-05-09 14:34:24 +02:00
|
|
|
SolarMutexGuard aGuard;
|
2019-01-10 15:54:56 +03:00
|
|
|
|
|
|
|
if (nPos < maItems.size())
|
|
|
|
{
|
|
|
|
Qt5MenuItem* pItem = maItems[nPos];
|
|
|
|
pItem->mpAction.reset();
|
|
|
|
pItem->mpMenu.reset();
|
|
|
|
|
|
|
|
maItems.erase(maItems.begin() + nPos);
|
2019-01-11 16:39:55 +03:00
|
|
|
|
|
|
|
// Recalculate action groups if necessary:
|
|
|
|
// if separator between two QActionGroups was removed,
|
|
|
|
// it may be needed to merge them
|
|
|
|
if (nPos > 0)
|
|
|
|
{
|
|
|
|
ReinitializeActionGroup(nPos - 1);
|
|
|
|
}
|
2019-01-10 15:54:56 +03:00
|
|
|
}
|
2018-05-08 16:05:59 +02:00
|
|
|
}
|
|
|
|
|
2019-02-27 15:03:07 +03:00
|
|
|
void Qt5Menu::SetSubMenu(SalMenuItem* pSalMenuItem, SalMenu* pSubMenu, unsigned nPos)
|
2018-05-08 16:05:59 +02:00
|
|
|
{
|
2018-05-23 10:51:30 +02:00
|
|
|
SolarMutexGuard aGuard;
|
2018-05-31 12:14:55 +02:00
|
|
|
Qt5MenuItem* pItem = static_cast<Qt5MenuItem*>(pSalMenuItem);
|
|
|
|
Qt5Menu* pQSubMenu = static_cast<Qt5Menu*>(pSubMenu);
|
2018-05-23 10:51:30 +02:00
|
|
|
|
2019-02-27 15:03:07 +03:00
|
|
|
pItem->mpSubMenu = pQSubMenu;
|
|
|
|
// at this point the pointer to parent menu may be outdated, update it too
|
|
|
|
pItem->mpParentMenu = this;
|
|
|
|
|
|
|
|
if (pQSubMenu != nullptr)
|
|
|
|
{
|
|
|
|
pQSubMenu->mpParentSalMenu = this;
|
|
|
|
pQSubMenu->mpQMenu = pItem->mpMenu.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
// if it's not a menu bar item, then convert it to corresponding item if type if necessary.
|
|
|
|
// If submenu is present and it's an action, convert it to menu.
|
|
|
|
// If submenu is not present and it's a menu, convert it to action.
|
|
|
|
// It may be fine to proceed in any case, but by skipping other cases
|
|
|
|
// amount of unneeded actions taken should be reduced.
|
|
|
|
if (pItem->mpParentMenu->mbMenuBar || (pQSubMenu && pItem->mpMenu)
|
|
|
|
|| ((!pQSubMenu) && pItem->mpAction))
|
|
|
|
{
|
2018-05-23 10:51:30 +02:00
|
|
|
return;
|
2019-02-27 15:03:07 +03:00
|
|
|
}
|
2018-05-23 10:51:30 +02:00
|
|
|
|
2019-02-27 15:03:07 +03:00
|
|
|
InsertMenuItem(pItem, nPos);
|
2018-05-08 16:05:59 +02:00
|
|
|
}
|
|
|
|
|
2018-05-31 12:14:55 +02:00
|
|
|
void Qt5Menu::SetFrame(const SalFrame* pFrame)
|
2018-05-08 16:05:59 +02:00
|
|
|
{
|
2019-02-25 17:00:21 +01:00
|
|
|
auto* pSalInst(static_cast<Qt5Instance*>(GetSalData()->m_pInstance));
|
|
|
|
assert(pSalInst);
|
|
|
|
if (!pSalInst->IsMainThread())
|
2018-11-01 13:57:56 +01:00
|
|
|
{
|
2019-02-25 17:00:21 +01:00
|
|
|
pSalInst->RunInMainThread([this, pFrame]() { SetFrame(pFrame); });
|
|
|
|
return;
|
2018-11-01 13:57:56 +01:00
|
|
|
}
|
|
|
|
|
2018-05-15 16:29:15 +02:00
|
|
|
SolarMutexGuard aGuard;
|
|
|
|
assert(mbMenuBar);
|
2018-05-31 12:14:55 +02:00
|
|
|
mpFrame = const_cast<Qt5Frame*>(static_cast<const Qt5Frame*>(pFrame));
|
2018-05-15 16:29:15 +02:00
|
|
|
|
2018-05-31 12:14:55 +02:00
|
|
|
mpFrame->SetMenu(this);
|
2018-05-15 16:29:15 +02:00
|
|
|
|
2018-08-09 17:14:14 +02:00
|
|
|
Qt5MainWindow* pMainWindow = mpFrame->GetTopLevelWindow();
|
2018-05-31 12:14:55 +02:00
|
|
|
if (pMainWindow)
|
2018-11-01 13:57:56 +01:00
|
|
|
{
|
2018-05-15 16:29:15 +02:00
|
|
|
mpQMenuBar = pMainWindow->menuBar();
|
2019-01-11 16:41:19 +03:00
|
|
|
if (mpQMenuBar)
|
2019-05-06 17:18:56 +02:00
|
|
|
{
|
2019-01-11 16:41:19 +03:00
|
|
|
mpQMenuBar->clear();
|
2019-05-06 17:18:56 +02:00
|
|
|
QPushButton* pButton
|
|
|
|
= static_cast<QPushButton*>(mpQMenuBar->cornerWidget(Qt::TopRightCorner));
|
|
|
|
if (pButton && ((mpCloseButton != pButton) || !maCloseButtonConnection))
|
|
|
|
{
|
|
|
|
maCloseButtonConnection
|
|
|
|
= connect(pButton, &QPushButton::clicked, this, &Qt5Menu::slotCloseDocument);
|
|
|
|
mpCloseButton = pButton;
|
|
|
|
}
|
|
|
|
}
|
2019-01-11 16:41:19 +03:00
|
|
|
|
2019-01-10 15:54:56 +03:00
|
|
|
mpQMenu = nullptr;
|
2018-05-15 16:29:15 +02:00
|
|
|
|
2018-11-01 13:57:56 +01:00
|
|
|
DoFullMenuUpdate(mpVCLMenu);
|
|
|
|
}
|
2018-05-15 16:29:15 +02:00
|
|
|
}
|
|
|
|
|
2019-01-10 15:54:56 +03:00
|
|
|
void Qt5Menu::DoFullMenuUpdate(Menu* pMenuBar)
|
2018-05-16 11:37:09 +02:00
|
|
|
{
|
2019-01-11 16:39:55 +03:00
|
|
|
// clear action groups since menu is rebuilt
|
|
|
|
ResetAllActionGroups();
|
2019-05-04 00:03:15 +00:00
|
|
|
ShowCloseButton(false);
|
2019-01-11 16:39:55 +03:00
|
|
|
|
2018-05-31 12:14:55 +02:00
|
|
|
for (sal_Int32 nItem = 0; nItem < static_cast<sal_Int32>(GetItemCount()); nItem++)
|
2018-05-16 11:37:09 +02:00
|
|
|
{
|
2018-05-31 12:14:55 +02:00
|
|
|
Qt5MenuItem* pSalMenuItem = GetItemAtPos(nItem);
|
2019-02-27 15:02:15 +03:00
|
|
|
InsertMenuItem(pSalMenuItem, nItem);
|
2018-11-23 12:08:08 +03:00
|
|
|
SetItemImage(nItem, pSalMenuItem, pSalMenuItem->maImage);
|
2018-05-23 10:51:30 +02:00
|
|
|
|
2018-05-31 12:14:55 +02:00
|
|
|
if (pSalMenuItem->mpSubMenu != nullptr)
|
2018-05-16 11:37:09 +02:00
|
|
|
{
|
2018-05-23 10:51:30 +02:00
|
|
|
pMenuBar->HandleMenuActivateEvent(pSalMenuItem->mpSubMenu->GetMenu());
|
2019-01-10 15:54:56 +03:00
|
|
|
pSalMenuItem->mpSubMenu->DoFullMenuUpdate(pMenuBar);
|
2018-05-23 10:51:30 +02:00
|
|
|
pMenuBar->HandleMenuDeActivateEvent(pSalMenuItem->mpSubMenu->GetMenu());
|
2018-05-16 11:37:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-05 01:37:28 +02:00
|
|
|
void Qt5Menu::ShowItem(unsigned nPos, bool bShow)
|
|
|
|
{
|
|
|
|
if (nPos < maItems.size())
|
|
|
|
{
|
|
|
|
Qt5MenuItem* pSalMenuItem = GetItemAtPos(nPos);
|
2018-11-23 12:08:08 +03:00
|
|
|
QAction* pAction = pSalMenuItem->getAction();
|
|
|
|
if (pAction)
|
|
|
|
pAction->setVisible(bShow);
|
2018-09-05 01:37:28 +02:00
|
|
|
pSalMenuItem->mbVisible = bShow;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-27 15:03:07 +03:00
|
|
|
void Qt5Menu::SetItemBits(unsigned nPos, MenuItemBits)
|
|
|
|
{
|
|
|
|
if (nPos < maItems.size())
|
|
|
|
{
|
|
|
|
Qt5MenuItem* pSalMenuItem = GetItemAtPos(nPos);
|
|
|
|
UpdateActionGroupItem(pSalMenuItem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-05 01:37:28 +02:00
|
|
|
void Qt5Menu::CheckItem(unsigned nPos, bool bChecked)
|
|
|
|
{
|
|
|
|
if (nPos < maItems.size())
|
|
|
|
{
|
|
|
|
Qt5MenuItem* pSalMenuItem = GetItemAtPos(nPos);
|
2018-11-23 12:08:08 +03:00
|
|
|
QAction* pAction = pSalMenuItem->getAction();
|
|
|
|
if (pAction)
|
2018-12-27 17:27:54 +01:00
|
|
|
{
|
|
|
|
pAction->setCheckable(true);
|
2018-11-23 12:08:08 +03:00
|
|
|
pAction->setChecked(bChecked);
|
2018-12-27 17:27:54 +01:00
|
|
|
}
|
2018-09-05 01:37:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Qt5Menu::EnableItem(unsigned nPos, bool bEnable)
|
|
|
|
{
|
|
|
|
if (nPos < maItems.size())
|
|
|
|
{
|
|
|
|
Qt5MenuItem* pSalMenuItem = GetItemAtPos(nPos);
|
2018-11-23 12:08:08 +03:00
|
|
|
QAction* pAction = pSalMenuItem->getAction();
|
|
|
|
if (pAction)
|
|
|
|
pAction->setEnabled(bEnable);
|
2018-09-05 01:37:28 +02:00
|
|
|
pSalMenuItem->mbEnabled = bEnable;
|
|
|
|
}
|
|
|
|
}
|
2018-05-08 16:05:59 +02:00
|
|
|
|
2018-10-23 12:06:00 +02:00
|
|
|
void Qt5Menu::SetItemText(unsigned, SalMenuItem* pItem, const OUString& rText)
|
2018-09-05 01:37:28 +02:00
|
|
|
{
|
|
|
|
Qt5MenuItem* pSalMenuItem = static_cast<Qt5MenuItem*>(pItem);
|
2018-11-23 12:08:08 +03:00
|
|
|
QAction* pAction = pSalMenuItem->getAction();
|
|
|
|
if (pAction)
|
2018-12-08 03:35:09 +01:00
|
|
|
{
|
|
|
|
OUString aText(rText);
|
|
|
|
NativeItemText(aText);
|
|
|
|
pAction->setText(toQString(aText));
|
|
|
|
}
|
2018-09-05 01:37:28 +02:00
|
|
|
}
|
2018-05-08 16:05:59 +02:00
|
|
|
|
2018-09-05 01:37:28 +02:00
|
|
|
void Qt5Menu::SetItemImage(unsigned, SalMenuItem* pItem, const Image& rImage)
|
|
|
|
{
|
2018-11-23 12:08:08 +03:00
|
|
|
Qt5MenuItem* pSalMenuItem = static_cast<Qt5MenuItem*>(pItem);
|
|
|
|
|
|
|
|
// Save new image to use it in DoFullMenuUpdate
|
|
|
|
pSalMenuItem->maImage = rImage;
|
|
|
|
|
|
|
|
QAction* pAction = pSalMenuItem->getAction();
|
|
|
|
if (!pAction)
|
2018-09-05 01:37:28 +02:00
|
|
|
return;
|
2018-05-08 16:05:59 +02:00
|
|
|
|
2019-05-04 00:03:15 +00:00
|
|
|
pAction->setIcon(QPixmap::fromImage(toQImage(rImage)));
|
2018-09-05 01:37:28 +02:00
|
|
|
}
|
2018-05-08 16:05:59 +02:00
|
|
|
|
2018-09-05 01:37:28 +02:00
|
|
|
void Qt5Menu::SetAccelerator(unsigned, SalMenuItem* pItem, const vcl::KeyCode&,
|
|
|
|
const OUString& rText)
|
|
|
|
{
|
|
|
|
Qt5MenuItem* pSalMenuItem = static_cast<Qt5MenuItem*>(pItem);
|
2018-11-23 12:08:08 +03:00
|
|
|
QAction* pAction = pSalMenuItem->getAction();
|
|
|
|
if (pAction)
|
|
|
|
pAction->setShortcut(QKeySequence(toQString(rText), QKeySequence::PortableText));
|
2018-09-05 01:37:28 +02:00
|
|
|
}
|
2018-05-08 16:05:59 +02:00
|
|
|
|
2018-06-01 15:28:26 +02:00
|
|
|
void Qt5Menu::GetSystemMenuData(SystemMenuData*) {}
|
2018-05-08 16:05:59 +02:00
|
|
|
|
2018-05-25 16:04:24 +02:00
|
|
|
Qt5Menu* Qt5Menu::GetTopLevel()
|
|
|
|
{
|
2018-05-31 12:14:55 +02:00
|
|
|
Qt5Menu* pMenu = this;
|
2018-05-25 16:04:24 +02:00
|
|
|
while (pMenu->mpParentSalMenu)
|
|
|
|
pMenu = pMenu->mpParentSalMenu;
|
|
|
|
return pMenu;
|
|
|
|
}
|
|
|
|
|
2019-03-15 23:29:55 +01:00
|
|
|
void Qt5Menu::ShowMenuBar(bool bVisible)
|
|
|
|
{
|
2019-06-13 02:33:20 +00:00
|
|
|
if (mpQMenuBar)
|
|
|
|
mpQMenuBar->setVisible(bVisible);
|
2019-03-15 23:29:55 +01:00
|
|
|
}
|
|
|
|
|
2018-05-23 17:39:44 +02:00
|
|
|
const Qt5Frame* Qt5Menu::GetFrame() const
|
|
|
|
{
|
|
|
|
SolarMutexGuard aGuard;
|
|
|
|
const Qt5Menu* pMenu = this;
|
2018-05-31 12:14:55 +02:00
|
|
|
while (pMenu && !pMenu->mpFrame)
|
2018-05-23 17:39:44 +02:00
|
|
|
pMenu = pMenu->mpParentSalMenu;
|
|
|
|
return pMenu ? pMenu->mpFrame : nullptr;
|
|
|
|
}
|
|
|
|
|
2018-10-22 14:30:20 +02:00
|
|
|
void Qt5Menu::slotMenuTriggered(Qt5MenuItem* pQItem)
|
2018-05-24 15:04:02 +02:00
|
|
|
{
|
2018-05-31 12:14:55 +02:00
|
|
|
if (pQItem)
|
2018-05-25 16:04:24 +02:00
|
|
|
{
|
|
|
|
Qt5Menu* pSalMenu = pQItem->mpParentMenu;
|
|
|
|
Qt5Menu* pTopLevel = pSalMenu->GetTopLevel();
|
2018-10-22 14:30:20 +02:00
|
|
|
|
2018-11-22 12:55:06 +03:00
|
|
|
Menu* pMenu = pSalMenu->GetMenu();
|
|
|
|
auto mnId = pQItem->mnId;
|
|
|
|
|
|
|
|
pTopLevel->GetMenu()->HandleMenuCommandEvent(pMenu, mnId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Qt5Menu::slotMenuAboutToShow(Qt5MenuItem* pQItem)
|
|
|
|
{
|
|
|
|
if (pQItem)
|
|
|
|
{
|
|
|
|
Qt5Menu* pSalMenu = pQItem->mpSubMenu;
|
|
|
|
Qt5Menu* pTopLevel = pSalMenu->GetTopLevel();
|
|
|
|
|
|
|
|
Menu* pMenu = pSalMenu->GetMenu();
|
|
|
|
|
|
|
|
// following function may update the menu
|
|
|
|
pTopLevel->GetMenu()->HandleMenuActivateEvent(pMenu);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Qt5Menu::slotMenuAboutToHide(Qt5MenuItem* pQItem)
|
|
|
|
{
|
|
|
|
if (pQItem)
|
|
|
|
{
|
|
|
|
Qt5Menu* pSalMenu = pQItem->mpSubMenu;
|
|
|
|
Qt5Menu* pTopLevel = pSalMenu->GetTopLevel();
|
|
|
|
|
|
|
|
Menu* pMenu = pSalMenu->GetMenu();
|
|
|
|
|
|
|
|
pTopLevel->GetMenu()->HandleMenuDeActivateEvent(pMenu);
|
2018-05-25 16:04:24 +02:00
|
|
|
}
|
2018-05-24 15:04:02 +02:00
|
|
|
}
|
|
|
|
|
2018-10-26 23:09:24 +02:00
|
|
|
void Qt5Menu::NativeItemText(OUString& rItemText)
|
|
|
|
{
|
|
|
|
// preserve literal '&'s in menu texts
|
|
|
|
rItemText = rItemText.replaceAll("&", "&&");
|
|
|
|
|
|
|
|
rItemText = rItemText.replace('~', '&');
|
|
|
|
}
|
2018-05-08 16:05:59 +02:00
|
|
|
|
2019-05-04 00:03:15 +00:00
|
|
|
void Qt5Menu::slotCloseDocument()
|
|
|
|
{
|
|
|
|
MenuBar* pVclMenuBar = static_cast<MenuBar*>(mpVCLMenu.get());
|
|
|
|
if (pVclMenuBar)
|
|
|
|
Application::PostUserEvent(pVclMenuBar->GetCloseButtonClickHdl());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Qt5Menu::ShowCloseButton(bool bShow)
|
|
|
|
{
|
|
|
|
if (!mpQMenuBar)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QPushButton* pButton = static_cast<QPushButton*>(mpQMenuBar->cornerWidget(Qt::TopRightCorner));
|
|
|
|
if (!pButton)
|
|
|
|
{
|
|
|
|
QIcon aIcon;
|
|
|
|
if (QIcon::hasThemeIcon("window-close-symbolic"))
|
|
|
|
aIcon = QIcon::fromTheme("window-close-symbolic");
|
|
|
|
else
|
|
|
|
aIcon = QIcon(
|
|
|
|
QPixmap::fromImage((toQImage(Image(StockImage::Yes, SV_RESID_BITMAP_CLOSEDOC)))));
|
|
|
|
pButton = new QPushButton(mpQMenuBar);
|
|
|
|
pButton->setIcon(aIcon);
|
|
|
|
pButton->setFlat(true);
|
2019-05-29 19:45:24 +02:00
|
|
|
pButton->setFocusPolicy(Qt::NoFocus);
|
2019-05-04 00:03:15 +00:00
|
|
|
pButton->setToolTip(toQString(VclResId(SV_HELPTEXT_CLOSEDOCUMENT)));
|
|
|
|
mpQMenuBar->setCornerWidget(pButton, Qt::TopRightCorner);
|
2019-05-06 17:18:56 +02:00
|
|
|
maCloseButtonConnection
|
|
|
|
= connect(pButton, &QPushButton::clicked, this, &Qt5Menu::slotCloseDocument);
|
|
|
|
mpCloseButton = pButton;
|
2019-05-04 00:03:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (bShow)
|
|
|
|
pButton->show();
|
|
|
|
else
|
|
|
|
pButton->hide();
|
|
|
|
}
|
|
|
|
|
2018-05-31 12:14:55 +02:00
|
|
|
Qt5MenuItem::Qt5MenuItem(const SalItemParams* pItemData)
|
2018-09-05 01:37:28 +02:00
|
|
|
: mpParentMenu(nullptr)
|
2018-05-31 12:14:55 +02:00
|
|
|
, mpSubMenu(nullptr)
|
2018-09-05 01:37:28 +02:00
|
|
|
, mnId(pItemData->nId)
|
|
|
|
, mnType(pItemData->eType)
|
|
|
|
, mbVisible(true)
|
|
|
|
, mbEnabled(true)
|
2018-11-23 12:08:08 +03:00
|
|
|
, maImage(pItemData->aImage)
|
2018-05-08 16:05:59 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-11-23 12:08:08 +03:00
|
|
|
QAction* Qt5MenuItem::getAction() const
|
|
|
|
{
|
|
|
|
if (mpMenu)
|
|
|
|
return mpMenu->menuAction();
|
|
|
|
if (mpAction)
|
|
|
|
return mpAction.get();
|
|
|
|
return nullptr;
|
|
|
|
}
|
2018-11-22 12:55:06 +03:00
|
|
|
|
2018-05-08 16:05:59 +02:00
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|