Add a ripple animation for polls bottom button.

This commit is contained in:
John Preston 2020-01-21 14:45:11 +03:00
parent d5bf742912
commit a482b744d2
2 changed files with 75 additions and 15 deletions

View File

@ -232,7 +232,7 @@ QSize Poll::countOptimalSize() {
+ st::msgDateFont->height + st::msgDateFont->height
+ st::historyPollAnswersSkip + st::historyPollAnswersSkip
+ answersHeight + answersHeight
+ st::msgPadding.bottom() + st::historyPollTotalVotesSkip
+ bottomButtonHeight + bottomButtonHeight
+ st::msgDateFont->height + st::msgDateFont->height
+ st::msgPadding.bottom(); + st::msgPadding.bottom();
@ -645,10 +645,10 @@ void Poll::draw(Painter &p, const QRect &r, TextSelection selection, crl::time m
selection); selection);
tshift += height; tshift += height;
} }
tshift += st::msgPadding.bottom();
if (!inlineFooter()) { if (!inlineFooter()) {
paintBottom(p, padding.left(), tshift, paintw, selection); paintBottom(p, padding.left(), tshift, paintw, selection);
} else if (!_totalVotesLabel.isEmpty()) { } else if (!_totalVotesLabel.isEmpty()) {
tshift += st::msgPadding.bottom();
paintInlineFooter(p, padding.left(), tshift, paintw, selection); paintInlineFooter(p, padding.left(), tshift, paintw, selection);
} }
} }
@ -679,7 +679,7 @@ void Poll::paintBottom(
int top, int top,
int paintw, int paintw,
TextSelection selection) const { TextSelection selection) const {
const auto stringtop = top + st::historyPollBottomButtonTop; const auto stringtop = top + st::msgPadding.bottom() + st::historyPollBottomButtonTop;
const auto selected = (selection == FullSelection); const auto selected = (selection == FullSelection);
const auto outbg = _parent->hasOutLayout(); const auto outbg = _parent->hasOutLayout();
const auto &regular = selected ? (outbg ? st::msgOutDateFgSelected : st::msgInDateFgSelected) : (outbg ? st::msgOutDateFg : st::msgInDateFg); const auto &regular = selected ? (outbg ? st::msgOutDateFgSelected : st::msgInDateFgSelected) : (outbg ? st::msgOutDateFg : st::msgInDateFg);
@ -692,8 +692,16 @@ void Poll::paintBottom(
: canSendVotes() : canSendVotes()
? _sendVotesLink ? _sendVotesLink
: nullptr; : nullptr;
const auto over = link ? ClickHandler::showAsActive(link) : false; if (_linkRipple) {
p.setFont(over ? st::semiboldFont->underline() : st::semiboldFont); const auto linkHeight = bottomButtonHeight();
p.setOpacity(st::historyPollRippleOpacity);
_linkRipple->paint(p, left - st::msgPadding.left(), height() - linkHeight, width());
if (_linkRipple->empty()) {
_linkRipple.reset();
}
p.setOpacity(1.);
}
p.setFont(st::semiboldFont);
if (!link) { if (!link) {
p.setPen(regular); p.setPen(regular);
} else { } else {
@ -1092,7 +1100,6 @@ TextState Poll::textState(QPoint point, StateRequest request) const {
} }
tshift += height; tshift += height;
} }
tshift += st::msgPadding.bottom();
if (!showVotersCount()) { if (!showVotersCount()) {
const auto link = showVotes() const auto link = showVotes()
? _showResultsLink ? _showResultsLink
@ -1100,12 +1107,10 @@ TextState Poll::textState(QPoint point, StateRequest request) const {
? _sendVotesLink ? _sendVotesLink
: nullptr; : nullptr;
if (link) { if (link) {
const auto string = showVotes() const auto linkHeight = bottomButtonHeight();
? tr::lng_polls_view_results(tr::now, Ui::Text::Upper) const auto linkTop = height() - linkHeight;
: tr::lng_polls_submit_votes(tr::now, Ui::Text::Upper); if (QRect(0, linkTop, width(), linkHeight).contains(point)) {
const auto stringw = st::semiboldFont->width(string); _lastLinkPoint = point;
const auto stringtop = tshift + st::historyPollBottomButtonTop;
if (QRect(padding.left() + (paintw - stringw) / 2, stringtop, stringw, st::semiboldFont->height).contains(point)) {
result.link = link; result.link = link;
return result; return result;
} }
@ -1125,6 +1130,8 @@ void Poll::clickHandlerPressedChanged(
&Answer::handler); &Answer::handler);
if (i != end(_answers)) { if (i != end(_answers)) {
toggleRipple(*i, pressed); toggleRipple(*i, pressed);
} else if (handler == _sendVotesLink || handler == _showResultsLink) {
toggleLinkRipple(pressed);
} }
} }
@ -1147,10 +1154,55 @@ void Poll::toggleRipple(Answer &answer, bool pressed) {
} }
const auto top = countAnswerTop(answer, innerWidth); const auto top = countAnswerTop(answer, innerWidth);
answer.ripple->add(_lastLinkPoint - QPoint(0, top)); answer.ripple->add(_lastLinkPoint - QPoint(0, top));
} else { } else if (answer.ripple) {
if (answer.ripple) { answer.ripple->lastStop();
answer.ripple->lastStop(); }
}
int Poll::bottomButtonHeight() const {
const auto skip = st::historyPollChoiceRight.height()
- st::historyPollFillingBottom
- st::historyPollFillingHeight
- (st::historyPollChoiceRight.height() - st::historyPollFillingHeight) / 2;
return st::historyPollTotalVotesSkip
- skip
+ st::historyPollBottomButtonSkip
+ st::msgDateFont->height
+ st::msgPadding.bottom();
}
void Poll::toggleLinkRipple(bool pressed) {
if (pressed) {
const auto linkWidth = width();
const auto linkHeight = bottomButtonHeight();
if (!_linkRipple) {
const auto drawMask = [&](QPainter &p) {
const auto radius = st::historyMessageRadius;
p.drawRoundedRect(
0,
0,
linkWidth,
linkHeight,
radius,
radius);
p.fillRect(0, 0, linkWidth, radius * 2, Qt::white);
};
auto mask = isBubbleBottom()
? Ui::RippleAnimation::maskByDrawer(
QSize(linkWidth, linkHeight),
false,
drawMask)
: Ui::RippleAnimation::rectMask({ linkWidth, linkHeight });
_linkRipple = std::make_unique<Ui::RippleAnimation>(
(_parent->hasOutLayout()
? st::historyPollRippleOut
: st::historyPollRippleIn),
std::move(mask),
[=] { history()->owner().requestViewRepaint(_parent); });
} }
_linkRipple->add(_lastLinkPoint - QPoint(0, height() - linkHeight));
} else if (_linkRipple) {
_linkRipple->lastStop();
} }
} }

View File

@ -11,6 +11,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "data/data_poll.h" #include "data/data_poll.h"
#include "base/weak_ptr.h" #include "base/weak_ptr.h"
namespace Ui {
class RippleAnimation;
} // namespace Ui
namespace HistoryView { namespace HistoryView {
class Poll : public Media, public base::has_weak_ptr { class Poll : public Media, public base::has_weak_ptr {
@ -137,10 +141,13 @@ private:
void radialAnimationCallback() const; void radialAnimationCallback() const;
void toggleRipple(Answer &answer, bool pressed); void toggleRipple(Answer &answer, bool pressed);
void toggleLinkRipple(bool pressed);
void toggleMultiOption(const QByteArray &option); void toggleMultiOption(const QByteArray &option);
void sendMultiOptions(); void sendMultiOptions();
void showResults(); void showResults();
[[nodiscard]] int bottomButtonHeight() const;
const not_null<PollData*> _poll; const not_null<PollData*> _poll;
int _pollVersion = 0; int _pollVersion = 0;
int _totalVotes = 0; int _totalVotes = 0;
@ -156,6 +163,7 @@ private:
Ui::Text::String _totalVotesLabel; Ui::Text::String _totalVotesLabel;
ClickHandlerPtr _showResultsLink; ClickHandlerPtr _showResultsLink;
ClickHandlerPtr _sendVotesLink; ClickHandlerPtr _sendVotesLink;
mutable std::unique_ptr<Ui::RippleAnimation> _linkRipple;
bool _hasSelected = false; bool _hasSelected = false;
mutable std::unique_ptr<AnswersAnimation> _answersAnimation; mutable std::unique_ptr<AnswersAnimation> _answersAnimation;