217 lines
4.8 KiB
C++
217 lines
4.8 KiB
C++
// This file is part of Desktop App Toolkit,
|
|
// a set of libraries for developing nice desktop applications.
|
|
//
|
|
// For license and copyright information please follow this link:
|
|
// https://github.com/desktop-app/legal/blob/master/LEGAL
|
|
//
|
|
#include "ui/widgets/menu/menu_action.h"
|
|
|
|
#include "ui/effects/ripple_animation.h"
|
|
#include "ui/painter.h"
|
|
|
|
#include <QtGui/QtEvents>
|
|
|
|
namespace Ui::Menu {
|
|
namespace {
|
|
|
|
[[nodiscard]] TextWithEntities ParseMenuItem(const QString &text) {
|
|
auto result = TextWithEntities();
|
|
result.text.reserve(text.size());
|
|
auto afterAmpersand = false;
|
|
for (const auto &ch : text) {
|
|
if (afterAmpersand) {
|
|
afterAmpersand = false;
|
|
if (ch == '&') {
|
|
result.text.append(ch);
|
|
} else {
|
|
result.entities.append(EntityInText{
|
|
EntityType::Underline,
|
|
int(result.text.size()),
|
|
1 });
|
|
result.text.append(ch);
|
|
}
|
|
} else if (ch == '&') {
|
|
afterAmpersand = true;
|
|
} else {
|
|
result.text.append(ch);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
TextParseOptions MenuTextOptions = {
|
|
TextParseLinks, // flags
|
|
0, // maxw
|
|
0, // maxh
|
|
Qt::LayoutDirectionAuto, // dir
|
|
};
|
|
|
|
} // namespace
|
|
|
|
Action::Action(
|
|
not_null<RpWidget*> parent,
|
|
const style::Menu &st,
|
|
not_null<QAction*> action,
|
|
const style::icon *icon,
|
|
const style::icon *iconOver)
|
|
: ItemBase(parent, st)
|
|
, _action(action)
|
|
, _st(st)
|
|
, _icon(icon)
|
|
, _iconOver(iconOver)
|
|
, _height(_st.itemPadding.top()
|
|
+ _st.itemStyle.font->height
|
|
+ _st.itemPadding.bottom()) {
|
|
|
|
setAcceptBoth(true);
|
|
|
|
initResizeHook(parent->sizeValue());
|
|
processAction();
|
|
|
|
enableMouseSelecting();
|
|
|
|
connect(_action, &QAction::changed, [=] { processAction(); });
|
|
}
|
|
|
|
bool Action::hasSubmenu() const {
|
|
return _action->menu() != nullptr;
|
|
}
|
|
|
|
void Action::paintEvent(QPaintEvent *e) {
|
|
Painter p(this);
|
|
paint(p);
|
|
}
|
|
|
|
void Action::paintBackground(Painter &p, bool selected) {
|
|
if (selected && _st.itemBgOver->c.alpha() < 255) {
|
|
p.fillRect(0, 0, width(), _height, _st.itemBg);
|
|
}
|
|
p.fillRect(
|
|
QRect(0, 0, width(), _height),
|
|
selected ? _st.itemBgOver : _st.itemBg);
|
|
}
|
|
|
|
void Action::paintText(Painter &p) {
|
|
_text.drawLeftElided(
|
|
p,
|
|
_st.itemPadding.left(),
|
|
_st.itemPadding.top(),
|
|
_textWidth,
|
|
width());
|
|
}
|
|
|
|
void Action::paint(Painter &p) {
|
|
const auto enabled = isEnabled();
|
|
const auto selected = isSelected();
|
|
paintBackground(p, selected);
|
|
if (enabled) {
|
|
RippleButton::paintRipple(p, 0, 0);
|
|
}
|
|
if (const auto icon = (selected ? _iconOver : _icon)) {
|
|
icon->paint(p, _st.itemIconPosition, width());
|
|
}
|
|
p.setPen(selected ? _st.itemFgOver : (enabled ? _st.itemFg : _st.itemFgDisabled));
|
|
paintText(p);
|
|
if (hasSubmenu()) {
|
|
const auto left = width() - _st.itemPadding.right() - _st.arrow.width();
|
|
const auto top = (_height - _st.arrow.height()) / 2;
|
|
if (enabled) {
|
|
_st.arrow.paint(p, left, top, width());
|
|
} else {
|
|
_st.arrow.paint(
|
|
p,
|
|
left,
|
|
top,
|
|
width(),
|
|
_st.itemFgDisabled->c);
|
|
}
|
|
} else if (!_shortcut.isEmpty()) {
|
|
p.setPen(selected
|
|
? _st.itemFgShortcutOver
|
|
: (enabled ? _st.itemFgShortcut : _st.itemFgShortcutDisabled));
|
|
p.drawTextRight(
|
|
_st.itemPadding.right(),
|
|
_st.itemPadding.top(),
|
|
width(),
|
|
_shortcut);
|
|
}
|
|
}
|
|
|
|
void Action::processAction() {
|
|
setPointerCursor(isEnabled());
|
|
if (_action->text().isEmpty()) {
|
|
_shortcut = QString();
|
|
_text.clear();
|
|
return;
|
|
}
|
|
const auto actionTextParts = _action->text().split('\t');
|
|
const auto actionText = actionTextParts.empty()
|
|
? QString()
|
|
: actionTextParts[0];
|
|
const auto actionShortcut = (actionTextParts.size() > 1)
|
|
? actionTextParts[1]
|
|
: QString();
|
|
_text.setMarkedText(
|
|
_st.itemStyle,
|
|
ParseMenuItem(actionText),
|
|
MenuTextOptions);
|
|
const auto textWidth = _text.maxWidth();
|
|
const auto &padding = _st.itemPadding;
|
|
|
|
const auto additionalWidth = hasSubmenu()
|
|
? padding.right() + _st.arrow.width()
|
|
: (!actionShortcut.isEmpty())
|
|
? (padding.right() + _st.itemStyle.font->width(actionShortcut))
|
|
: 0;
|
|
const auto goodWidth = padding.left()
|
|
+ textWidth
|
|
+ padding.right()
|
|
+ additionalWidth;
|
|
|
|
const auto w = std::clamp(goodWidth, _st.widthMin, _st.widthMax);
|
|
_textWidth = w - (goodWidth - textWidth);
|
|
_shortcut = actionShortcut;
|
|
setMinWidth(w);
|
|
update();
|
|
}
|
|
|
|
bool Action::isEnabled() const {
|
|
return _action->isEnabled();
|
|
}
|
|
|
|
not_null<QAction*> Action::action() const {
|
|
return _action;
|
|
}
|
|
|
|
QPoint Action::prepareRippleStartPosition() const {
|
|
return mapFromGlobal(QCursor::pos());
|
|
}
|
|
|
|
QImage Action::prepareRippleMask() const {
|
|
return Ui::RippleAnimation::rectMask(size());
|
|
}
|
|
|
|
int Action::contentHeight() const {
|
|
return _height;
|
|
}
|
|
|
|
void Action::handleKeyPress(not_null<QKeyEvent*> e) {
|
|
if (!isSelected()) {
|
|
return;
|
|
}
|
|
const auto key = e->key();
|
|
if (key == Qt::Key_Enter || key == Qt::Key_Return) {
|
|
setClicked(TriggeredSource::Keyboard);
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Action::setIcon(
|
|
const style::icon *icon,
|
|
const style::icon *iconOver) {
|
|
_icon = icon;
|
|
_iconOver = iconOver ? iconOver : icon;
|
|
update();
|
|
}
|
|
|
|
} // namespace Ui::Menu
|