Add send vote / retract vote animation.

This commit is contained in:
John Preston 2018-12-22 22:36:00 +04:00
parent 8e28a229f2
commit f2e1d90c74
3 changed files with 303 additions and 50 deletions

View File

@ -507,7 +507,7 @@ historyPollPercentFont: semiboldFont;
historyPollPercentSkip: 6px; historyPollPercentSkip: 6px;
historyPollPercentTop: 0px; historyPollPercentTop: 0px;
historyPollTotalVotesSkip: 5px; historyPollTotalVotesSkip: 5px;
historyPollFillingMin: 8px; historyPollFillingMin: 4px;
historyPollFillingHeight: 4px; historyPollFillingHeight: 4px;
historyPollFillingRadius: 1px; historyPollFillingRadius: 1px;
historyPollFillingBottom: 2px; historyPollFillingBottom: 2px;
@ -524,6 +524,7 @@ historyPollRadio: Radio(defaultRadio) {
} }
historyPollRadioOpacity: 0.7; historyPollRadioOpacity: 0.7;
historyPollRadioOpacityOver: 1.; historyPollRadioOpacityOver: 1.;
historyPollDuration: 300;
boxAttachEmoji: IconButton(historyAttachEmoji) { boxAttachEmoji: IconButton(historyAttachEmoji) {
width: 30px; width: 30px;

View File

@ -64,6 +64,16 @@ FormattedLargeNumber FormatLargeNumber(int64 number) {
HistoryPoll::Answer::Answer() : text(st::msgMinWidth / 2) { HistoryPoll::Answer::Answer() : text(st::msgMinWidth / 2) {
} }
void HistoryPoll::Answer::fillText(const PollAnswer &original) {
if (!text.isEmpty() && text.originalText() == original.text) {
return;
}
text.setText(
st::historyPollAnswerStyle,
original.text,
Ui::WebpageTextTitleOptions());
}
HistoryPoll::HistoryPoll( HistoryPoll::HistoryPoll(
not_null<Element*> parent, not_null<Element*> parent,
not_null<PollData*> poll) not_null<PollData*> poll)
@ -112,6 +122,10 @@ QSize HistoryPoll::countOptimalSize() {
return { maxWidth, minHeight }; return { maxWidth, minHeight };
} }
bool HistoryPoll::canVote() const {
return !_voted && !_closed;
}
int HistoryPoll::countAnswerHeight( int HistoryPoll::countAnswerHeight(
const Answer &answer, const Answer &answer,
int innerWidth) const { int innerWidth) const {
@ -157,8 +171,10 @@ void HistoryPoll::updateTexts() {
return; return;
} }
_pollVersion = _poll->version; _pollVersion = _poll->version;
_closed = _poll->closed;
const auto willStartAnimation = checkAnimationStart();
_closed = _poll->closed;
_question.setText( _question.setText(
st::historyPollQuestionStyle, st::historyPollQuestionStyle,
_poll->question, _poll->question,
@ -168,22 +184,25 @@ void HistoryPoll::updateTexts() {
lang(_closed ? lng_polls_closed : lng_polls_anonymous)); lang(_closed ? lng_polls_closed : lng_polls_anonymous));
updateAnswers(); updateAnswers();
updateVotes();
if (willStartAnimation) {
startAnimation();
}
} }
void HistoryPoll::updateAnswers() { void HistoryPoll::updateAnswers() {
const auto pairFromAnswer = [](const Answer &a) {
return std::make_pair(a.text.originalText(), a.option);
};
const auto pairFromPollAnswer = [](const PollAnswer &p) {
return std::make_pair(p.text, p.option);
};
const auto changed = !ranges::equal( const auto changed = !ranges::equal(
_answers, _answers,
_poll->answers, _poll->answers,
ranges::equal_to(), ranges::equal_to(),
pairFromAnswer, &Answer::option,
pairFromPollAnswer); &PollAnswer::option);
if (!changed) { if (!changed) {
auto &&answers = ranges::view::zip(_answers, _poll->answers);
for (auto &&[answer, original] : answers) {
answer.fillText(original);
}
return; return;
} }
_answers = ranges::view::all( _answers = ranges::view::all(
@ -191,16 +210,15 @@ void HistoryPoll::updateAnswers() {
) | ranges::view::transform([](const PollAnswer &answer) { ) | ranges::view::transform([](const PollAnswer &answer) {
auto result = Answer(); auto result = Answer();
result.option = answer.option; result.option = answer.option;
result.text.setText( result.fillText(answer);
st::historyPollAnswerStyle,
answer.text,
Ui::WebpageTextTitleOptions());
return result; return result;
}) | ranges::to_vector; }) | ranges::to_vector;
for (auto &answer : _answers) { for (auto &answer : _answers) {
answer.handler = createAnswerClickHandler(answer); answer.handler = createAnswerClickHandler(answer);
} }
_answersAnimation = nullptr;
} }
ClickHandlerPtr HistoryPoll::createAnswerClickHandler( ClickHandlerPtr HistoryPoll::createAnswerClickHandler(
@ -218,6 +236,14 @@ void HistoryPoll::updateVotes() const {
updateAnswerVotes(); updateAnswerVotes();
} }
void HistoryPoll::updateVotesCheckAnimation() const {
const auto willStartAnimation = checkAnimationStart();
updateVotes();
if (willStartAnimation) {
startAnimation();
}
}
void HistoryPoll::updateTotalVotes() const { void HistoryPoll::updateTotalVotes() const {
if (_totalVotes == _poll->totalVoters) { if (_totalVotes == _poll->totalVoters) {
return; return;
@ -279,7 +305,7 @@ void HistoryPoll::draw(Painter &p, const QRect &r, TextSelection selection, Time
if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return; if (width() < st::msgPadding.left() + st::msgPadding.right() + 1) return;
auto paintx = 0, painty = 0, paintw = width(), painth = height(); auto paintx = 0, painty = 0, paintw = width(), painth = height();
updateVotes(); updateVotesCheckAnimation();
const auto outbg = _parent->hasOutLayout(); const auto outbg = _parent->hasOutLayout();
const auto selected = (selection == FullSelection); const auto selected = (selection == FullSelection);
@ -300,10 +326,29 @@ void HistoryPoll::draw(Painter &p, const QRect &r, TextSelection selection, Time
_subtitle.drawLeftElided(p, padding.left(), tshift, paintw, width()); _subtitle.drawLeftElided(p, padding.left(), tshift, paintw, width());
tshift += st::msgDateFont->height + st::historyPollAnswersSkip; tshift += st::msgDateFont->height + st::historyPollAnswersSkip;
for (const auto &answer : _answers) { const auto progress = _answersAnimation
? _answersAnimation->progress.current(ms, 1.)
: 1.;
if (progress == 1.) {
_answersAnimation = nullptr;
}
auto &&answers = ranges::view::zip(
_answers,
ranges::view::ints(0, int(_answers.size())));
for (const auto &[answer, index] : answers) {
const auto animation = _answersAnimation
? &_answersAnimation->data[index]
: nullptr;
if (animation) {
animation->percent.update(progress, anim::linear);
animation->filling.update(progress, anim::linear);
animation->opacity.update(progress, anim::linear);
}
const auto height = paintAnswer( const auto height = paintAnswer(
p, p,
answer, answer,
animation,
padding.left(), padding.left(),
tshift, tshift,
paintw, paintw,
@ -322,18 +367,135 @@ void HistoryPoll::draw(Painter &p, const QRect &r, TextSelection selection, Time
int HistoryPoll::paintAnswer( int HistoryPoll::paintAnswer(
Painter &p, Painter &p,
const Answer &answer, const Answer &answer,
const AnswerAnimation *animation,
int left, int left,
int top, int top,
int width, int width,
int outerWidth, int outerWidth,
TextSelection selection, TextSelection selection,
TimeMs ms) const { TimeMs ms) const {
const auto result = countAnswerHeight(answer, width); const auto height = countAnswerHeight(answer, width);
const auto bottom = top + result; const auto outbg = _parent->hasOutLayout();
const auto aleft = left + st::historyPollAnswerPadding.left();
const auto awidth = width
- st::historyPollAnswerPadding.left()
- st::historyPollAnswerPadding.right();
if (animation) {
const auto opacity = animation->opacity.current();
if (opacity < 1.) {
p.setOpacity(1. - opacity);
paintRadio(p, answer, left, top, selection);
}
if (opacity > 0.) {
const auto percent = QString::number(
int(std::round(animation->percent.current()))) + '%';
const auto percentWidth = st::historyPollPercentFont->width(
percent);
p.setOpacity(opacity);
paintPercent(
p,
percent,
percentWidth,
left,
top,
outerWidth,
selection);
p.setOpacity(sqrt(opacity));
paintFilling(
p,
animation->filling.current(),
left,
top,
width,
height,
selection);
p.setOpacity(1.);
}
} else if (canVote()) {
paintRadio(p, answer, left, top, selection);
} else {
paintPercent(
p,
answer.votesPercent,
answer.votesPercentWidth,
left,
top,
outerWidth,
selection);
paintFilling(
p,
answer.filling,
left,
top,
width,
height,
selection);
}
top += st::historyPollAnswerPadding.top();
p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg);
answer.text.drawLeft(p, aleft, top, awidth, outerWidth);
return height;
}
void HistoryPoll::paintRadio(
Painter &p,
const Answer &answer,
int left,
int top,
TextSelection selection) const {
top += st::historyPollAnswerPadding.top();
const auto outbg = _parent->hasOutLayout(); const auto outbg = _parent->hasOutLayout();
const auto selected = (selection == FullSelection); const auto selected = (selection == FullSelection);
const auto &regular = selected ? (outbg ? st::msgOutDateFgSelected : st::msgInDateFgSelected) : (outbg ? st::msgOutDateFg : st::msgInDateFg);
PainterHighQualityEnabler hq(p);
const auto &st = st::historyPollRadio;
const auto over = ClickHandler::showAsActive(answer.handler);
const auto &regular = selected ? (outbg ? st::msgOutDateFgSelected : st::msgInDateFgSelected) : (outbg ? st::msgOutDateFg : st::msgInDateFg);
auto pen = regular->p;
pen.setWidth(st.thickness);
p.setPen(pen);
p.setBrush(Qt::NoBrush);
const auto o = p.opacity();
p.setOpacity(o * (over ? st::historyPollRadioOpacityOver : st::historyPollRadioOpacity));
p.drawEllipse(QRectF(left, top, st.diameter, st.diameter).marginsRemoved(QMarginsF(st.thickness / 2., st.thickness / 2., st.thickness / 2., st.thickness / 2.)));
p.setOpacity(o);
}
void HistoryPoll::paintPercent(
Painter &p,
const QString &percent,
int percentWidth,
int left,
int top,
int outerWidth,
TextSelection selection) const {
const auto outbg = _parent->hasOutLayout();
const auto selected = (selection == FullSelection);
const auto aleft = left + st::historyPollAnswerPadding.left();
top += st::historyPollAnswerPadding.top();
p.setFont(st::historyPollPercentFont);
p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg);
const auto pleft = aleft - percentWidth - st::historyPollPercentSkip;
p.drawTextLeft(pleft, top + st::historyPollPercentTop, outerWidth, percent, percentWidth);
}
void HistoryPoll::paintFilling(
Painter &p,
float64 filling,
int left,
int top,
int width,
int height,
TextSelection selection) const {
const auto bottom = top + height;
const auto outbg = _parent->hasOutLayout();
const auto selected = (selection == FullSelection);
const auto aleft = left + st::historyPollAnswerPadding.left(); const auto aleft = left + st::historyPollAnswerPadding.left();
const auto awidth = width const auto awidth = width
- st::historyPollAnswerPadding.left() - st::historyPollAnswerPadding.left()
@ -341,46 +503,89 @@ int HistoryPoll::paintAnswer(
top += st::historyPollAnswerPadding.top(); top += st::historyPollAnswerPadding.top();
if (_voted || _closed) { const auto bar = outbg ? (selected ? st::msgWaveformOutActiveSelected : st::msgWaveformOutActive) : (selected ? st::msgWaveformInActiveSelected : st::msgWaveformInActive);
p.setFont(st::historyPollPercentFont); PainterHighQualityEnabler hq(p);
p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg); p.setPen(Qt::NoPen);
const auto left = aleft p.setBrush(bar);
- answer.votesPercentWidth const auto max = awidth - st::historyPollFillingRight;
- st::historyPollPercentSkip; const auto size = anim::interpolate(st::historyPollFillingMin, max, filling);
p.drawTextLeft(left, top + st::historyPollPercentTop, outerWidth, answer.votesPercent, answer.votesPercentWidth); const auto radius = st::historyPollFillingRadius;
} else { const auto ftop = bottom - st::historyPollFillingBottom - st::historyPollFillingHeight;
PainterHighQualityEnabler hq(p); p.drawRoundedRect(aleft, ftop, size, st::historyPollFillingHeight, radius, radius);
const auto &st = st::historyPollRadio; }
const auto over = ClickHandler::showAsActive(answer.handler);
auto pen = regular->p; bool HistoryPoll::answerVotesChanged() const {
pen.setWidth(st.thickness); if (_poll->answers.size() != _answers.size()
p.setPen(pen); || _poll->answers.empty()) {
p.setBrush(Qt::NoBrush); return false;
p.setOpacity(over ? st::historyPollRadioOpacityOver : st::historyPollRadioOpacity);
p.drawEllipse(QRectF(left, top, st.diameter, st.diameter).marginsRemoved(QMarginsF(st.thickness / 2., st.thickness / 2., st.thickness / 2., st.thickness / 2.)));
p.setOpacity(1.);
} }
return !ranges::equal(
_answers,
_poll->answers,
ranges::equal_to(),
&Answer::votes,
&PollAnswer::votes);
}
p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg); void HistoryPoll::saveStateInAnimation() const {
answer.text.drawLeft(p, aleft, top, awidth, outerWidth); if (_answersAnimation) {
return;
}
const auto can = canVote();
_answersAnimation = std::make_unique<AnswersAnimation>();
_answersAnimation->data.reserve(_answers.size());
const auto convert = [&](const Answer &answer) {
auto result = AnswerAnimation();
result.percent = can
? 0.
: (answer.votes * 100. / std::max(_totalVotes, 1));
result.filling = can ? 0. : answer.filling;
result.opacity = can ? 0. : 1.;
return result;
};
ranges::transform(
_answers,
ranges::back_inserter(_answersAnimation->data),
convert);
}
if (_voted || _closed) { bool HistoryPoll::checkAnimationStart() const {
const auto bar = outbg ? (selected ? st::msgWaveformOutActiveSelected : st::msgWaveformOutActive) : (selected ? st::msgWaveformInActiveSelected : st::msgWaveformInActive); if (_poll->answers.size() != _answers.size()) {
PainterHighQualityEnabler hq(p); // Skip initial changes.
p.setPen(Qt::NoPen); return false;
p.setBrush(bar); }
const auto max = awidth - st::historyPollFillingRight; const auto result = (canVote() != (!_poll->voted() && !_poll->closed))
const auto size = anim::interpolate(st::historyPollFillingMin, max, answer.filling); || answerVotesChanged();
const auto radius = st::historyPollFillingRadius; if (result) {
const auto top = bottom - st::historyPollFillingBottom - st::historyPollFillingHeight; saveStateInAnimation();
p.drawRoundedRect(aleft, top, size, st::historyPollFillingHeight, radius, radius);
} }
return result; return result;
} }
void HistoryPoll::startAnimation() const {
if (!_answersAnimation) {
return;
}
const auto can = canVote();
auto &&both = ranges::view::zip(_answers, _answersAnimation->data);
for (auto &&[answer, data] : both) {
data.percent.start(can
? 0.
: answer.votes * 100. / std::max(_totalVotes, 1));
data.filling.start(can ? 0. : answer.filling);
data.opacity.start(can ? 0. : 1.);
}
_answersAnimation->progress.start(
[=] { Auth().data().requestViewRepaint(_parent); },
0.,
1.,
st::historyPollDuration);
}
TextState HistoryPoll::textState(QPoint point, StateRequest request) const { TextState HistoryPoll::textState(QPoint point, StateRequest request) const {
auto result = TextState(_parent); auto result = TextState(_parent);
if (_voted || _closed) { if (!canVote()) {
return result; return result;
} }

View File

@ -37,9 +37,22 @@ public:
~HistoryPoll(); ~HistoryPoll();
private: private:
struct AnswerAnimation {
anim::value percent;
anim::value filling;
anim::value opacity;
};
struct AnswersAnimation {
std::vector<AnswerAnimation> data;
Animation progress;
};
struct Answer { struct Answer {
Answer(); Answer();
void fillText(const PollAnswer &original);
Text text; Text text;
QByteArray option; QByteArray option;
mutable int votes = 0; mutable int votes = 0;
@ -49,9 +62,12 @@ private:
mutable bool chosen = false; mutable bool chosen = false;
ClickHandlerPtr handler; ClickHandlerPtr handler;
}; };
QSize countOptimalSize() override; QSize countOptimalSize() override;
QSize countCurrentSize(int newWidth) override; QSize countCurrentSize(int newWidth) override;
bool canVote() const;
int countAnswerHeight(const Answer &answer, int innerWidth) const; int countAnswerHeight(const Answer &answer, int innerWidth) const;
[[nodiscard]] ClickHandlerPtr createAnswerClickHandler( [[nodiscard]] ClickHandlerPtr createAnswerClickHandler(
const Answer &answer) const; const Answer &answer) const;
@ -65,16 +81,45 @@ private:
const PollAnswer &original, const PollAnswer &original,
int totalVotes, int totalVotes,
int maxVotes) const; int maxVotes) const;
void updateVotesCheckAnimation() const;
int paintAnswer( int paintAnswer(
Painter &p, Painter &p,
const Answer &answer, const Answer &answer,
const AnswerAnimation *animation,
int left, int left,
int top, int top,
int width, int width,
int outerWidth, int outerWidth,
TextSelection selection, TextSelection selection,
TimeMs ms) const; TimeMs ms) const;
void paintRadio(
Painter &p,
const Answer &answer,
int left,
int top,
TextSelection selection) const;
void paintPercent(
Painter &p,
const QString &percent,
int percentWidth,
int left,
int top,
int outerWidth,
TextSelection selection) const;
void paintFilling(
Painter &p,
float64 filling,
int left,
int top,
int width,
int height,
TextSelection selection) const;
bool checkAnimationStart() const;
bool answerVotesChanged() const;
void saveStateInAnimation() const;
void startAnimation() const;
not_null<PollData*> _poll; not_null<PollData*> _poll;
int _pollVersion = 0; int _pollVersion = 0;
@ -87,4 +132,6 @@ private:
std::vector<Answer> _answers; std::vector<Answer> _answers;
mutable Text _totalVotesLabel; mutable Text _totalVotesLabel;
mutable std::unique_ptr<AnswersAnimation> _answersAnimation;
}; };