2021-02-05 05:44:04 +00:00
|
|
|
/*
|
|
|
|
This file is part of Telegram Desktop,
|
|
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
|
|
|
|
For license and copyright information please follow this link:
|
|
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
|
|
*/
|
|
|
|
#include "editor/photo_editor_controls.h"
|
|
|
|
|
2021-03-21 12:38:33 +00:00
|
|
|
#include "editor/controllers/controllers.h"
|
2021-02-13 08:23:21 +00:00
|
|
|
#include "lang/lang_keys.h"
|
|
|
|
#include "ui/image/image_prepare.h"
|
2021-02-05 09:24:26 +00:00
|
|
|
#include "ui/widgets/buttons.h"
|
2021-02-13 08:23:21 +00:00
|
|
|
|
2021-02-05 09:24:26 +00:00
|
|
|
#include "styles/style_editor.h"
|
|
|
|
|
2021-02-05 05:44:04 +00:00
|
|
|
namespace Editor {
|
|
|
|
|
2021-02-13 08:23:21 +00:00
|
|
|
class EdgeButton final : public Ui::RippleButton {
|
|
|
|
public:
|
|
|
|
EdgeButton(
|
|
|
|
not_null<Ui::RpWidget*> parent,
|
|
|
|
const QString &text,
|
|
|
|
int height,
|
|
|
|
bool left,
|
|
|
|
const style::color &bg,
|
|
|
|
const style::color &fg,
|
|
|
|
const style::RippleAnimation &st);
|
|
|
|
|
|
|
|
protected:
|
|
|
|
QImage prepareRippleMask() const override;
|
|
|
|
QPoint prepareRippleStartPosition() const override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
void init();
|
|
|
|
|
|
|
|
const style::color &_fg;
|
|
|
|
Ui::Text::String _text;
|
|
|
|
const int _width;
|
|
|
|
const QRect _rippleRect;
|
|
|
|
const QColor _bg;
|
|
|
|
const bool _left;
|
|
|
|
|
|
|
|
QImage rounded(std::optional<QColor> color) const;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
EdgeButton::EdgeButton(
|
|
|
|
not_null<Ui::RpWidget*> parent,
|
|
|
|
const QString &text,
|
|
|
|
int height,
|
|
|
|
bool left,
|
|
|
|
const style::color &bg,
|
|
|
|
const style::color &fg,
|
|
|
|
const style::RippleAnimation &st)
|
|
|
|
: Ui::RippleButton(parent, st)
|
|
|
|
, _fg(fg)
|
|
|
|
, _text(st::semiboldTextStyle, text.toUpper())
|
|
|
|
, _width(_text.maxWidth()
|
|
|
|
+ st::photoEditorTextButtonPadding.left()
|
|
|
|
+ st::photoEditorTextButtonPadding.right())
|
|
|
|
, _rippleRect(QRect(0, 0, _width, height))
|
|
|
|
, _bg(bg->c)
|
|
|
|
, _left(left) {
|
|
|
|
resize(_width, height);
|
|
|
|
init();
|
|
|
|
}
|
|
|
|
|
|
|
|
void EdgeButton::init() {
|
2021-05-02 09:16:12 +00:00
|
|
|
// const auto bg = rounded(_bg);
|
2021-02-13 08:23:21 +00:00
|
|
|
|
|
|
|
paintRequest(
|
|
|
|
) | rpl::start_with_next([=] {
|
|
|
|
Painter p(this);
|
|
|
|
|
2021-05-02 09:16:12 +00:00
|
|
|
// p.drawImage(QPoint(), bg);
|
2021-02-13 08:23:21 +00:00
|
|
|
|
|
|
|
paintRipple(p, _rippleRect.x(), _rippleRect.y());
|
|
|
|
|
|
|
|
p.setPen(_fg);
|
|
|
|
const auto textTop = (height() - _text.minHeight()) / 2;
|
|
|
|
_text.draw(p, 0, textTop, width(), style::al_center);
|
|
|
|
}, lifetime());
|
|
|
|
}
|
|
|
|
|
|
|
|
QImage EdgeButton::rounded(std::optional<QColor> color) const {
|
|
|
|
auto result = QImage(
|
|
|
|
_rippleRect.size() * cIntRetinaFactor(),
|
|
|
|
QImage::Format_ARGB32_Premultiplied);
|
|
|
|
result.setDevicePixelRatio(cIntRetinaFactor());
|
|
|
|
result.fill(color.value_or(Qt::white));
|
|
|
|
|
|
|
|
using Option = Images::Option;
|
|
|
|
const auto options = Option::Smooth
|
|
|
|
| Option::RoundedLarge
|
|
|
|
| (_left ? Option::RoundedTopLeft : Option::RoundedTopRight)
|
|
|
|
| (_left ? Option::RoundedBottomLeft : Option::RoundedBottomRight);
|
|
|
|
return Images::prepare(std::move(result), 0, 0, options, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
QImage EdgeButton::prepareRippleMask() const {
|
|
|
|
return rounded(std::nullopt);
|
|
|
|
}
|
|
|
|
|
|
|
|
QPoint EdgeButton::prepareRippleStartPosition() const {
|
|
|
|
return mapFromGlobal(QCursor::pos()) - _rippleRect.topLeft();
|
|
|
|
}
|
|
|
|
|
2021-05-02 09:16:12 +00:00
|
|
|
class ButtonBar final : public Ui::RpWidget {
|
2021-02-05 08:58:10 +00:00
|
|
|
public:
|
2021-05-02 09:16:12 +00:00
|
|
|
ButtonBar(
|
|
|
|
not_null<Ui::RpWidget*> parent,
|
|
|
|
const style::color &bg);
|
2021-02-05 08:58:10 +00:00
|
|
|
|
2021-05-02 09:16:12 +00:00
|
|
|
private:
|
|
|
|
QImage _roundedBg;
|
2021-02-05 08:58:10 +00:00
|
|
|
|
|
|
|
};
|
|
|
|
|
2021-05-02 09:16:12 +00:00
|
|
|
ButtonBar::ButtonBar(
|
|
|
|
not_null<Ui::RpWidget*> parent,
|
|
|
|
const style::color &bg)
|
2021-02-05 05:44:04 +00:00
|
|
|
: RpWidget(parent) {
|
2021-05-02 09:16:12 +00:00
|
|
|
sizeValue(
|
|
|
|
) | rpl::start_with_next([=](const QSize &size) {
|
|
|
|
const auto children = RpWidget::children();
|
|
|
|
if (children.empty()) {
|
|
|
|
return;
|
2021-02-05 08:58:10 +00:00
|
|
|
}
|
2021-05-02 09:16:12 +00:00
|
|
|
const auto widgets = ranges::view::all(
|
|
|
|
children
|
|
|
|
) | ranges::view::filter([](not_null<const QObject*> object) {
|
|
|
|
return object->isWidgetType();
|
|
|
|
}) | ranges::view::transform([](not_null<QObject*> object) {
|
|
|
|
return static_cast<Ui::RpWidget*>(object.get());
|
|
|
|
}) | ranges::to_vector;
|
|
|
|
|
|
|
|
const auto residualWidth = size.width()
|
|
|
|
- ranges::accumulate(widgets, 0, ranges::plus(), &QWidget::width);
|
|
|
|
const auto step = residualWidth / float(widgets.size() - 1);
|
|
|
|
|
|
|
|
auto left = 0.;
|
|
|
|
for (const auto &widget : widgets) {
|
|
|
|
widget->moveToLeft(int(left), 0);
|
|
|
|
left += widget->width() + step;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto result = QImage(
|
|
|
|
size * cIntRetinaFactor(),
|
|
|
|
QImage::Format_ARGB32_Premultiplied);
|
|
|
|
result.setDevicePixelRatio(cIntRetinaFactor());
|
|
|
|
result.fill(bg->c);
|
|
|
|
|
|
|
|
const auto options = Images::Option::Smooth
|
|
|
|
| Images::Option::RoundedLarge
|
|
|
|
| Images::Option::RoundedAll;
|
|
|
|
_roundedBg = Images::prepare(std::move(result), 0, 0, options, 0, 0);
|
|
|
|
}, lifetime());
|
|
|
|
|
|
|
|
paintRequest(
|
|
|
|
) | rpl::start_with_next([=] {
|
|
|
|
Painter p(this);
|
|
|
|
|
|
|
|
p.drawImage(QPoint(), _roundedBg);
|
|
|
|
}, lifetime());
|
2021-02-05 08:58:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
PhotoEditorControls::PhotoEditorControls(
|
|
|
|
not_null<Ui::RpWidget*> parent,
|
2021-02-23 10:34:12 +00:00
|
|
|
std::shared_ptr<Controllers> controllers,
|
2021-02-22 08:14:23 +00:00
|
|
|
const PhotoModifications modifications,
|
2021-02-05 08:58:10 +00:00
|
|
|
bool doneControls)
|
|
|
|
: RpWidget(parent)
|
2021-04-03 14:58:49 +00:00
|
|
|
, _bg(st::roundedBg)
|
2021-05-02 09:16:12 +00:00
|
|
|
, _buttonHeight(st::photoEditorButtonBarHeight)
|
|
|
|
, _transformButtons(base::make_unique_q<ButtonBar>(this, _bg))
|
|
|
|
, _paintBottomButtons(base::make_unique_q<ButtonBar>(this, _bg))
|
|
|
|
, _transformCancel(base::make_unique_q<EdgeButton>(
|
|
|
|
_transformButtons,
|
|
|
|
tr::lng_cancel(tr::now),
|
|
|
|
_buttonHeight,
|
|
|
|
true,
|
|
|
|
_bg,
|
|
|
|
st::activeButtonFg,
|
|
|
|
st::photoEditorRotateButton.ripple))
|
2021-02-05 09:24:26 +00:00
|
|
|
, _rotateButton(base::make_unique_q<Ui::IconButton>(
|
2021-03-14 09:39:55 +00:00
|
|
|
_transformButtons,
|
2021-02-05 09:24:26 +00:00
|
|
|
st::photoEditorRotateButton))
|
|
|
|
, _flipButton(base::make_unique_q<Ui::IconButton>(
|
2021-03-14 09:39:55 +00:00
|
|
|
_transformButtons,
|
2021-02-13 04:29:31 +00:00
|
|
|
st::photoEditorFlipButton))
|
|
|
|
, _paintModeButton(base::make_unique_q<Ui::IconButton>(
|
2021-03-14 09:39:55 +00:00
|
|
|
_transformButtons,
|
|
|
|
st::photoEditorPaintModeButton))
|
2021-05-02 09:16:12 +00:00
|
|
|
, _transformDone(base::make_unique_q<EdgeButton>(
|
|
|
|
_transformButtons,
|
|
|
|
tr::lng_box_done(tr::now),
|
|
|
|
_buttonHeight,
|
|
|
|
false,
|
|
|
|
_bg,
|
|
|
|
st::lightButtonFg,
|
|
|
|
st::photoEditorRotateButton.ripple))
|
|
|
|
, _paintCancel(base::make_unique_q<EdgeButton>(
|
|
|
|
_paintBottomButtons,
|
|
|
|
tr::lng_cancel(tr::now),
|
|
|
|
_buttonHeight,
|
|
|
|
true,
|
|
|
|
_bg,
|
|
|
|
st::activeButtonFg,
|
|
|
|
st::photoEditorRotateButton.ripple))
|
2021-03-14 09:39:55 +00:00
|
|
|
, _undoButton(base::make_unique_q<Ui::IconButton>(
|
2021-05-02 09:16:12 +00:00
|
|
|
_paintBottomButtons,
|
2021-03-14 09:39:55 +00:00
|
|
|
st::photoEditorUndoButton))
|
|
|
|
, _redoButton(base::make_unique_q<Ui::IconButton>(
|
2021-05-02 09:16:12 +00:00
|
|
|
_paintBottomButtons,
|
2021-03-14 09:39:55 +00:00
|
|
|
st::photoEditorRedoButton))
|
|
|
|
, _paintModeButtonActive(base::make_unique_q<Ui::IconButton>(
|
2021-05-02 09:16:12 +00:00
|
|
|
_paintBottomButtons,
|
2021-02-13 08:23:21 +00:00
|
|
|
st::photoEditorPaintModeButton))
|
2021-02-23 10:35:23 +00:00
|
|
|
, _stickersButton(controllers->stickersPanelController
|
|
|
|
? base::make_unique_q<Ui::IconButton>(
|
2021-05-02 09:16:12 +00:00
|
|
|
_paintBottomButtons,
|
2021-02-23 10:35:23 +00:00
|
|
|
st::photoEditorStickersButton)
|
|
|
|
: nullptr)
|
2021-05-02 09:16:12 +00:00
|
|
|
, _paintDone(base::make_unique_q<EdgeButton>(
|
|
|
|
_paintBottomButtons,
|
2021-02-13 08:23:21 +00:00
|
|
|
tr::lng_box_done(tr::now),
|
2021-05-02 09:16:12 +00:00
|
|
|
_buttonHeight,
|
2021-02-13 08:23:21 +00:00
|
|
|
false,
|
|
|
|
_bg,
|
|
|
|
st::lightButtonFg,
|
2021-04-03 14:58:49 +00:00
|
|
|
st::photoEditorRotateButton.ripple)) {
|
2021-02-05 08:58:10 +00:00
|
|
|
|
2021-05-02 09:16:12 +00:00
|
|
|
{
|
|
|
|
const auto &padding = st::photoEditorButtonBarPadding;
|
|
|
|
const auto w = st::photoEditorButtonBarWidth
|
|
|
|
- padding.left()
|
|
|
|
- padding.right();
|
|
|
|
_transformButtons->resize(w, _buttonHeight);
|
|
|
|
_paintBottomButtons->resize(w, _buttonHeight);
|
|
|
|
}
|
2021-03-14 09:39:55 +00:00
|
|
|
|
2021-04-03 14:58:49 +00:00
|
|
|
{
|
|
|
|
const auto icon = &st::photoEditorPaintIconActive;
|
|
|
|
_paintModeButtonActive->setIconOverride(icon, icon);
|
|
|
|
}
|
2021-03-14 09:39:55 +00:00
|
|
|
_paintModeButtonActive->setAttribute(Qt::WA_TransparentForMouseEvents);
|
2021-02-05 08:58:10 +00:00
|
|
|
|
2021-03-14 09:39:55 +00:00
|
|
|
rpl::combine(
|
|
|
|
sizeValue(),
|
|
|
|
_mode.value()
|
|
|
|
) | rpl::start_with_next([=](
|
|
|
|
const QSize &size,
|
|
|
|
const PhotoEditorMode &mode) {
|
|
|
|
if (size.isEmpty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-05-02 09:16:12 +00:00
|
|
|
const auto buttonsTop = size.height()
|
2021-05-02 09:24:48 +00:00
|
|
|
- st::photoEditorControlsBottomSkip
|
|
|
|
- _transformButtons->height();
|
|
|
|
|
2021-03-14 09:39:55 +00:00
|
|
|
const auto ¤t = _transformButtons->isHidden()
|
2021-05-02 09:16:12 +00:00
|
|
|
? _paintBottomButtons
|
2021-03-14 09:39:55 +00:00
|
|
|
: _transformButtons;
|
2021-02-05 08:58:10 +00:00
|
|
|
|
2021-03-14 09:39:55 +00:00
|
|
|
current->moveToLeft(
|
|
|
|
(size.width() - current->width()) / 2,
|
2021-02-16 03:45:05 +00:00
|
|
|
buttonsTop);
|
2021-02-05 08:58:10 +00:00
|
|
|
}, lifetime());
|
|
|
|
|
2021-02-23 10:34:12 +00:00
|
|
|
controllers->undoController->setPerformRequestChanges(rpl::merge(
|
2021-03-14 09:42:18 +00:00
|
|
|
_undoButton->clicks() | rpl::map_to(Undo::Undo),
|
|
|
|
_redoButton->clicks() | rpl::map_to(Undo::Redo)));
|
|
|
|
|
2021-02-23 10:34:12 +00:00
|
|
|
controllers->undoController->canPerformChanges(
|
2021-03-14 09:42:18 +00:00
|
|
|
) | rpl::start_with_next([=](const UndoController::EnableRequest &r) {
|
|
|
|
const auto isUndo = (r.command == Undo::Undo);
|
|
|
|
const auto &button = isUndo ? _undoButton : _redoButton;
|
|
|
|
button->setAttribute(Qt::WA_TransparentForMouseEvents, !r.enable);
|
|
|
|
if (!r.enable) {
|
|
|
|
button->clearState();
|
|
|
|
}
|
|
|
|
|
|
|
|
button->setIconOverride(r.enable
|
|
|
|
? nullptr
|
|
|
|
: isUndo
|
|
|
|
? &st::photoEditorUndoButtonInactive
|
|
|
|
: &st::photoEditorRedoButtonInactive);
|
|
|
|
}, lifetime());
|
|
|
|
|
2021-02-23 10:35:23 +00:00
|
|
|
if (_stickersButton) {
|
|
|
|
controllers->stickersPanelController->setShowRequestChanges(
|
|
|
|
_stickersButton->clicks(
|
|
|
|
) | rpl::map_to(std::optional<bool>(std::nullopt)));
|
|
|
|
|
|
|
|
controllers->stickersPanelController->setMoveRequestChanges(
|
2021-05-02 09:16:12 +00:00
|
|
|
_paintBottomButtons->positionValue(
|
2021-02-23 10:35:23 +00:00
|
|
|
) | rpl::map([=](const QPoint &containerPos) {
|
|
|
|
return QPoint(
|
|
|
|
(x() + width()) / 2,
|
|
|
|
y() + containerPos.y() + _stickersButton->y());
|
|
|
|
}));
|
|
|
|
|
|
|
|
controllers->stickersPanelController->panelShown(
|
|
|
|
) | rpl::start_with_next([=](bool shown) {
|
2021-04-03 14:58:49 +00:00
|
|
|
const auto icon = shown
|
|
|
|
? &st::photoEditorStickersIconActive
|
|
|
|
: nullptr;
|
|
|
|
_stickersButton->setIconOverride(icon, icon);
|
2021-02-23 10:35:23 +00:00
|
|
|
}, _stickersButton->lifetime());
|
|
|
|
}
|
|
|
|
|
2021-04-03 14:58:49 +00:00
|
|
|
rpl::single(
|
|
|
|
rpl::empty_value()
|
|
|
|
) | rpl::skip(modifications.flipped ? 0 : 1) | rpl::then(
|
|
|
|
_flipButton->clicks() | rpl::to_empty
|
2021-02-22 08:14:23 +00:00
|
|
|
) | rpl::start_with_next([=] {
|
|
|
|
_flipped = !_flipped;
|
2021-04-03 14:58:49 +00:00
|
|
|
const auto icon = _flipped ? &st::photoEditorFlipIconActive : nullptr;
|
|
|
|
_flipButton->setIconOverride(icon, icon);
|
2021-02-22 08:14:23 +00:00
|
|
|
}, _flipButton->lifetime());
|
|
|
|
|
2021-02-05 08:58:10 +00:00
|
|
|
}
|
2021-02-07 23:10:30 +00:00
|
|
|
|
|
|
|
rpl::producer<int> PhotoEditorControls::rotateRequests() const {
|
2021-02-13 08:23:21 +00:00
|
|
|
return _rotateButton->clicks() | rpl::map_to(90);
|
2021-02-07 23:10:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
rpl::producer<> PhotoEditorControls::flipRequests() const {
|
|
|
|
return _flipButton->clicks() | rpl::to_empty;
|
|
|
|
}
|
|
|
|
|
2021-02-13 04:29:31 +00:00
|
|
|
rpl::producer<> PhotoEditorControls::paintModeRequests() const {
|
|
|
|
return _paintModeButton->clicks() | rpl::to_empty;
|
|
|
|
}
|
|
|
|
|
2021-03-14 09:39:55 +00:00
|
|
|
rpl::producer<> PhotoEditorControls::doneRequests() const {
|
2021-05-02 09:16:12 +00:00
|
|
|
return rpl::merge(
|
|
|
|
_transformDone->clicks() | rpl::to_empty,
|
|
|
|
_paintDone->clicks() | rpl::to_empty);
|
2021-03-14 09:39:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
rpl::producer<> PhotoEditorControls::cancelRequests() const {
|
2021-05-02 09:16:12 +00:00
|
|
|
return rpl::merge(
|
|
|
|
_transformCancel->clicks() | rpl::to_empty,
|
|
|
|
_paintCancel->clicks() | rpl::to_empty);
|
2021-03-14 09:39:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void PhotoEditorControls::applyMode(const PhotoEditorMode &mode) {
|
|
|
|
using Mode = PhotoEditorMode::Mode;
|
|
|
|
_transformButtons->setVisible(mode.mode == Mode::Transform);
|
2021-05-02 09:16:12 +00:00
|
|
|
_paintBottomButtons->setVisible(mode.mode == Mode::Paint);
|
2021-03-14 09:39:55 +00:00
|
|
|
_mode = mode;
|
|
|
|
}
|
|
|
|
|
2021-02-05 05:44:04 +00:00
|
|
|
} // namespace Editor
|