tdesktop/Telegram/SourceFiles/history/view/history_view_view_button.cpp

225 lines
5.4 KiB
C++

/*
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 "history/view/history_view_view_button.h"
#include "core/click_handler_types.h"
#include "history/history.h"
#include "history/history_item_components.h"
#include "history/view/history_view_cursor_state.h"
#include "lang/lang_keys.h"
#include "ui/effects/ripple_animation.h"
#include "ui/painter.h"
#include "ui/text/text_utilities.h" // Ui::Text::ToUpper
#include "window/window_session_controller.h"
#include "styles/style_chat.h"
namespace HistoryView {
namespace {
[[nodiscard]] ClickHandlerPtr MakeMediaButtonClickHandler(
not_null<Data::Media*> media) {
const auto start = media->giveawayStart();
const auto results = media->giveawayResults();
Assert(start || results);
const auto peer = media->parent()->history()->peer;
const auto messageId = media->parent()->id;
if (media->parent()->isSending() || media->parent()->hasFailed()) {
return nullptr;
}
const auto maybeStart = start
? *start
: std::optional<Data::GiveawayStart>();
const auto maybeResults = results
? *results
: std::optional<Data::GiveawayResults>();
return std::make_shared<LambdaClickHandler>([=](
ClickContext context) {
const auto my = context.other.value<ClickHandlerContext>();
const auto controller = my.sessionWindow.get();
if (!controller) {
return;
}
ResolveGiveawayInfo(
controller,
peer,
messageId,
maybeStart,
maybeResults);
});
}
[[nodiscard]] QString MakeMediaButtonText(not_null<Data::Media*> media) {
Expects(media->giveawayStart() || media->giveawayResults());
return Ui::Text::Upper(tr::lng_prizes_how_works(tr::now));
}
} // namespace
struct ViewButton::Inner {
Inner(
not_null<Data::Media*> media,
uint8 colorIndex,
Fn<void()> updateCallback);
void updateMask(int height);
void toggleRipple(bool pressed);
const style::margins &margins;
const ClickHandlerPtr link;
const Fn<void()> updateCallback;
uint32 lastWidth : 24 = 0;
uint32 colorIndex : 6 = 0;
uint32 aboveInfo : 1 = 0;
uint32 externalLink : 1 = 0;
QPoint lastPoint;
std::unique_ptr<Ui::RippleAnimation> ripple;
Ui::Text::String text;
};
bool ViewButton::MediaHasViewButton(not_null<Data::Media*> media) {
return media->giveawayStart() || media->giveawayResults();
}
ViewButton::Inner::Inner(
not_null<Data::Media*> media,
uint8 colorIndex,
Fn<void()> updateCallback)
: margins(st::historyViewButtonMargins)
, link(MakeMediaButtonClickHandler(media))
, updateCallback(std::move(updateCallback))
, colorIndex(colorIndex)
, aboveInfo(1)
, text(st::historyViewButtonTextStyle, MakeMediaButtonText(media)) {
}
void ViewButton::Inner::updateMask(int height) {
ripple = std::make_unique<Ui::RippleAnimation>(
st::defaultRippleAnimation,
Ui::RippleAnimation::RoundRectMask(
QSize(lastWidth, height - margins.top() - margins.bottom()),
st::roundRadiusLarge),
updateCallback);
}
void ViewButton::Inner::toggleRipple(bool pressed) {
if (ripple) {
if (pressed) {
ripple->add(lastPoint);
} else {
ripple->lastStop();
}
}
}
ViewButton::ViewButton(
not_null<Data::Media*> media,
uint8 colorIndex,
Fn<void()> updateCallback)
: _inner(std::make_unique<Inner>(
media,
colorIndex,
std::move(updateCallback))) {
}
ViewButton::~ViewButton() {
}
void ViewButton::resized() const {
_inner->updateMask(height());
}
int ViewButton::height() const {
return st::historyViewButtonHeight;
}
bool ViewButton::belowMessageInfo() const {
return !_inner->aboveInfo;
}
void ViewButton::draw(
Painter &p,
const QRect &r,
const Ui::ChatPaintContext &context) {
const auto st = context.st;
const auto stm = context.messageStyle();
const auto selected = context.selected();
const auto cache = context.outbg
? stm->replyCache[st->colorPatternIndex(_inner->colorIndex)].get()
: st->coloredReplyCache(selected, _inner->colorIndex).get();
const auto radius = st::historyPagePreview.radius;
if (_inner->ripple && !_inner->ripple->empty()) {
_inner->ripple->paint(p, r.left(), r.top(), r.width(), &cache->bg);
}
PainterHighQualityEnabler hq(p);
p.setPen(Qt::NoPen);
p.setBrush(cache->bg);
p.drawRoundedRect(r, radius, radius);
p.setPen(cache->icon);
_inner->text.drawElided(
p,
r.left(),
r.top() + (r.height() - _inner->text.minHeight()) / 2,
r.width(),
1,
style::al_top);
if (_inner->externalLink) {
const auto &icon = st::msgBotKbUrlIcon;
const auto padding = st::msgBotKbIconPadding;
icon.paint(
p,
r.left() + r.width() - icon.width() - padding,
r.top() + padding,
r.width(),
cache->icon);
}
if (_inner->lastWidth != r.width()) {
_inner->lastWidth = r.width();
resized();
}
}
const ClickHandlerPtr &ViewButton::link() const {
return _inner->link;
}
bool ViewButton::checkLink(const ClickHandlerPtr &other, bool pressed) {
if (_inner->link != other) {
return false;
}
_inner->toggleRipple(pressed);
return true;
}
bool ViewButton::getState(
QPoint point,
const QRect &g,
not_null<TextState*> outResult) const {
if (!g.contains(point)) {
return false;
}
outResult->link = _inner->link;
_inner->lastPoint = point - g.topLeft();
return true;
}
QRect ViewButton::countRect(const QRect &r) const {
return QRect(
r.left(),
r.top() + r.height() - height(),
r.width(),
height()) - _inner->margins;
}
} // namespace HistoryView