mirror of https://github.com/procxx/kepka.git
Display correct poll subtitle and quiz answer.
This commit is contained in:
parent
d57905c2b3
commit
95b2886bad
|
@ -2166,6 +2166,9 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
"lng_launch_exe_dont_ask" = "Don't ask me again";
|
"lng_launch_exe_dont_ask" = "Don't ask me again";
|
||||||
|
|
||||||
"lng_polls_anonymous" = "Anonymous Poll";
|
"lng_polls_anonymous" = "Anonymous Poll";
|
||||||
|
"lng_polls_public" = "Poll";
|
||||||
|
"lng_polls_anonymous_quiz" = "Anonymous Quiz";
|
||||||
|
"lng_polls_public_quiz" = "Quiz";
|
||||||
"lng_polls_closed" = "Final results";
|
"lng_polls_closed" = "Final results";
|
||||||
"lng_polls_votes_count#one" = "{count} vote";
|
"lng_polls_votes_count#one" = "{count} vote";
|
||||||
"lng_polls_votes_count#other" = "{count} votes";
|
"lng_polls_votes_count#other" = "{count} votes";
|
||||||
|
|
|
@ -41,7 +41,10 @@ bool PollData::applyChanges(const MTPDpoll &poll) {
|
||||||
Expects(poll.vid().v == id);
|
Expects(poll.vid().v == id);
|
||||||
|
|
||||||
const auto newQuestion = qs(poll.vquestion());
|
const auto newQuestion = qs(poll.vquestion());
|
||||||
const auto newClosed = poll.is_closed();
|
const auto newFlags = (poll.is_closed() ? Flag::Closed : Flag(0))
|
||||||
|
| (poll.is_public_voters() ? Flag::PublicVotes : Flag(0))
|
||||||
|
| (poll.is_multiple_choice() ? Flag::MultiChoice : Flag(0))
|
||||||
|
| (poll.is_quiz() ? Flag::Quiz : Flag(0));
|
||||||
auto newAnswers = ranges::view::all(
|
auto newAnswers = ranges::view::all(
|
||||||
poll.vanswers().v
|
poll.vanswers().v
|
||||||
) | ranges::view::transform([](const MTPPollAnswer &data) {
|
) | ranges::view::transform([](const MTPPollAnswer &data) {
|
||||||
|
@ -56,14 +59,14 @@ bool PollData::applyChanges(const MTPDpoll &poll) {
|
||||||
) | ranges::to_vector;
|
) | ranges::to_vector;
|
||||||
|
|
||||||
const auto changed1 = (question != newQuestion)
|
const auto changed1 = (question != newQuestion)
|
||||||
|| (closed != newClosed);
|
|| (_flags != newFlags);
|
||||||
const auto changed2 = (answers != newAnswers);
|
const auto changed2 = (answers != newAnswers);
|
||||||
if (!changed1 && !changed2) {
|
if (!changed1 && !changed2) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (changed1) {
|
if (changed1) {
|
||||||
question = newQuestion;
|
question = newQuestion;
|
||||||
closed = newClosed;
|
_flags = newFlags;
|
||||||
}
|
}
|
||||||
if (changed2) {
|
if (changed2) {
|
||||||
std::swap(answers, newAnswers);
|
std::swap(answers, newAnswers);
|
||||||
|
@ -71,6 +74,7 @@ bool PollData::applyChanges(const MTPDpoll &poll) {
|
||||||
if (const auto current = answerByOption(old.option)) {
|
if (const auto current = answerByOption(old.option)) {
|
||||||
current->votes = old.votes;
|
current->votes = old.votes;
|
||||||
current->chosen = old.chosen;
|
current->chosen = old.chosen;
|
||||||
|
current->correct = old.correct;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -104,7 +108,7 @@ bool PollData::applyResults(const MTPPollResults &results) {
|
||||||
void PollData::checkResultsReload(not_null<HistoryItem*> item, crl::time now) {
|
void PollData::checkResultsReload(not_null<HistoryItem*> item, crl::time now) {
|
||||||
if (lastResultsUpdate && lastResultsUpdate + kShortPollTimeout > now) {
|
if (lastResultsUpdate && lastResultsUpdate + kShortPollTimeout > now) {
|
||||||
return;
|
return;
|
||||||
} else if (closed) {
|
} else if (closed()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
lastResultsUpdate = now;
|
lastResultsUpdate = now;
|
||||||
|
@ -137,17 +141,42 @@ bool PollData::applyResultToAnswers(
|
||||||
answer->chosen = voters.is_chosen();
|
answer->chosen = voters.is_chosen();
|
||||||
changed = true;
|
changed = true;
|
||||||
}
|
}
|
||||||
|
if (answer->correct != voters.is_correct()) {
|
||||||
|
answer->correct = voters.is_correct();
|
||||||
|
changed = true;
|
||||||
|
}
|
||||||
} else if (const auto existing = answerByOption(option)) {
|
} else if (const auto existing = answerByOption(option)) {
|
||||||
answer->chosen = existing->chosen;
|
answer->chosen = existing->chosen;
|
||||||
|
answer->correct = existing->correct;
|
||||||
}
|
}
|
||||||
return changed;
|
return changed;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PollData::Flags PollData::flags() const {
|
||||||
|
return _flags;
|
||||||
|
}
|
||||||
|
|
||||||
bool PollData::voted() const {
|
bool PollData::voted() const {
|
||||||
return ranges::find(answers, true, &PollAnswer::chosen) != end(answers);
|
return ranges::find(answers, true, &PollAnswer::chosen) != end(answers);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool PollData::closed() const {
|
||||||
|
return (_flags & Flag::Closed);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PollData::publicVotes() const {
|
||||||
|
return (_flags & Flag::PublicVotes);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PollData::multiChoice() const {
|
||||||
|
return (_flags & Flag::MultiChoice);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PollData::quiz() const {
|
||||||
|
return (_flags & Flag::Quiz);
|
||||||
|
}
|
||||||
|
|
||||||
MTPPoll PollDataToMTP(not_null<const PollData*> poll) {
|
MTPPoll PollDataToMTP(not_null<const PollData*> poll) {
|
||||||
const auto convert = [](const PollAnswer &answer) {
|
const auto convert = [](const PollAnswer &answer) {
|
||||||
return MTP_pollAnswer(
|
return MTP_pollAnswer(
|
||||||
|
|
|
@ -12,6 +12,7 @@ struct PollAnswer {
|
||||||
QByteArray option;
|
QByteArray option;
|
||||||
int votes = 0;
|
int votes = 0;
|
||||||
bool chosen = false;
|
bool chosen = false;
|
||||||
|
bool correct = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
inline bool operator==(const PollAnswer &a, const PollAnswer &b) {
|
inline bool operator==(const PollAnswer &a, const PollAnswer &b) {
|
||||||
|
@ -26,20 +27,34 @@ inline bool operator!=(const PollAnswer &a, const PollAnswer &b) {
|
||||||
struct PollData {
|
struct PollData {
|
||||||
explicit PollData(PollId id);
|
explicit PollData(PollId id);
|
||||||
|
|
||||||
|
enum class Flag {
|
||||||
|
Closed = 0x01,
|
||||||
|
PublicVotes = 0x02,
|
||||||
|
MultiChoice = 0x04,
|
||||||
|
Quiz = 0x08,
|
||||||
|
};
|
||||||
|
friend inline constexpr bool is_flag_type(Flag) { return true; };
|
||||||
|
using Flags = base::flags<Flag>;
|
||||||
|
|
||||||
bool applyChanges(const MTPDpoll &poll);
|
bool applyChanges(const MTPDpoll &poll);
|
||||||
bool applyResults(const MTPPollResults &results);
|
bool applyResults(const MTPPollResults &results);
|
||||||
void checkResultsReload(not_null<HistoryItem*> item, crl::time now);
|
void checkResultsReload(not_null<HistoryItem*> item, crl::time now);
|
||||||
|
|
||||||
PollAnswer *answerByOption(const QByteArray &option);
|
[[nodiscard]] PollAnswer *answerByOption(const QByteArray &option);
|
||||||
const PollAnswer *answerByOption(const QByteArray &option) const;
|
[[nodiscard]] const PollAnswer *answerByOption(
|
||||||
|
const QByteArray &option) const;
|
||||||
|
|
||||||
bool voted() const;
|
[[nodiscard]] Flags flags() const;
|
||||||
|
[[nodiscard]] bool voted() const;
|
||||||
|
[[nodiscard]] bool closed() const;
|
||||||
|
[[nodiscard]] bool publicVotes() const;
|
||||||
|
[[nodiscard]] bool multiChoice() const;
|
||||||
|
[[nodiscard]] bool quiz() const;
|
||||||
|
|
||||||
PollId id = 0;
|
PollId id = 0;
|
||||||
QString question;
|
QString question;
|
||||||
std::vector<PollAnswer> answers;
|
std::vector<PollAnswer> answers;
|
||||||
int totalVoters = 0;
|
int totalVoters = 0;
|
||||||
bool closed = false;
|
|
||||||
QByteArray sendingVote;
|
QByteArray sendingVote;
|
||||||
crl::time lastResultsUpdate = 0;
|
crl::time lastResultsUpdate = 0;
|
||||||
|
|
||||||
|
@ -52,6 +67,8 @@ private:
|
||||||
const MTPPollAnswerVoters &result,
|
const MTPPollAnswerVoters &result,
|
||||||
bool isMinResults);
|
bool isMinResults);
|
||||||
|
|
||||||
|
Flags _flags = Flags();
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
MTPPoll PollDataToMTP(not_null<const PollData*> poll);
|
MTPPoll PollDataToMTP(not_null<const PollData*> poll);
|
||||||
|
|
|
@ -1719,8 +1719,8 @@ void HistoryInner::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
||||||
}
|
}
|
||||||
if (const auto media = item->media()) {
|
if (const auto media = item->media()) {
|
||||||
if (const auto poll = media->poll()) {
|
if (const auto poll = media->poll()) {
|
||||||
if (!poll->closed) {
|
if (!poll->closed()) {
|
||||||
if (poll->voted()) {
|
if (poll->voted() && !poll->quiz()) {
|
||||||
_menu->addAction(tr::lng_polls_retract(tr::now), [=] {
|
_menu->addAction(tr::lng_polls_retract(tr::now), [=] {
|
||||||
session().api().sendPollVotes(itemId, {});
|
session().api().sendPollVotes(itemId, {});
|
||||||
});
|
});
|
||||||
|
|
|
@ -112,6 +112,8 @@ struct Poll::AnswerAnimation {
|
||||||
anim::value percent;
|
anim::value percent;
|
||||||
anim::value filling;
|
anim::value filling;
|
||||||
anim::value opacity;
|
anim::value opacity;
|
||||||
|
bool chosen = false;
|
||||||
|
bool correct = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Poll::AnswersAnimation {
|
struct Poll::AnswersAnimation {
|
||||||
|
@ -132,7 +134,7 @@ struct Poll::SendingAnimation {
|
||||||
struct Poll::Answer {
|
struct Poll::Answer {
|
||||||
Answer();
|
Answer();
|
||||||
|
|
||||||
void fillText(const PollAnswer &original);
|
void fillData(not_null<PollData*> poll, const PollAnswer &original);
|
||||||
|
|
||||||
Ui::Text::String text;
|
Ui::Text::String text;
|
||||||
QByteArray option;
|
QByteArray option;
|
||||||
|
@ -142,6 +144,7 @@ struct Poll::Answer {
|
||||||
float64 filling = 0.;
|
float64 filling = 0.;
|
||||||
QString votesPercentString;
|
QString votesPercentString;
|
||||||
bool chosen = false;
|
bool chosen = false;
|
||||||
|
bool correct = false;
|
||||||
ClickHandlerPtr handler;
|
ClickHandlerPtr handler;
|
||||||
mutable std::unique_ptr<Ui::RippleAnimation> ripple;
|
mutable std::unique_ptr<Ui::RippleAnimation> ripple;
|
||||||
};
|
};
|
||||||
|
@ -159,7 +162,11 @@ Poll::SendingAnimation::SendingAnimation(
|
||||||
Poll::Answer::Answer() : text(st::msgMinWidth / 2) {
|
Poll::Answer::Answer() : text(st::msgMinWidth / 2) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void Poll::Answer::fillText(const PollAnswer &original) {
|
void Poll::Answer::fillData(
|
||||||
|
not_null<PollData*> poll,
|
||||||
|
const PollAnswer &original) {
|
||||||
|
chosen = original.chosen;
|
||||||
|
correct = poll->quiz() ? original.correct : chosen;
|
||||||
if (!text.isEmpty() && text.toString() == original.text) {
|
if (!text.isEmpty() && text.toString() == original.text) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -218,7 +225,7 @@ QSize Poll::countOptimalSize() {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Poll::showVotes() const {
|
bool Poll::showVotes() const {
|
||||||
return _voted || _closed;
|
return _voted || (_flags & PollData::Flag::Closed);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Poll::canVote() const {
|
bool Poll::canVote() const {
|
||||||
|
@ -309,11 +316,20 @@ void Poll::updateTexts() {
|
||||||
_poll->question,
|
_poll->question,
|
||||||
options);
|
options);
|
||||||
}
|
}
|
||||||
if (_closed != _poll->closed || _subtitle.isEmpty()) {
|
if (_flags != _poll->flags() || _subtitle.isEmpty()) {
|
||||||
_closed = _poll->closed;
|
using Flag = PollData::Flag;
|
||||||
|
_flags = _poll->flags();
|
||||||
_subtitle.setText(
|
_subtitle.setText(
|
||||||
st::msgDateTextStyle,
|
st::msgDateTextStyle,
|
||||||
_closed ? tr::lng_polls_closed(tr::now) : tr::lng_polls_anonymous(tr::now));
|
((_flags & Flag::Closed)
|
||||||
|
? tr::lng_polls_closed(tr::now)
|
||||||
|
: (_flags & Flag::Quiz)
|
||||||
|
? ((_flags & Flag::PublicVotes)
|
||||||
|
? tr::lng_polls_public_quiz(tr::now)
|
||||||
|
: tr::lng_polls_anonymous_quiz(tr::now))
|
||||||
|
: ((_flags & Flag::PublicVotes)
|
||||||
|
? tr::lng_polls_public(tr::now)
|
||||||
|
: tr::lng_polls_anonymous(tr::now))));
|
||||||
}
|
}
|
||||||
|
|
||||||
updateAnswers();
|
updateAnswers();
|
||||||
|
@ -334,16 +350,16 @@ void Poll::updateAnswers() {
|
||||||
if (!changed) {
|
if (!changed) {
|
||||||
auto &&answers = ranges::view::zip(_answers, _poll->answers);
|
auto &&answers = ranges::view::zip(_answers, _poll->answers);
|
||||||
for (auto &&[answer, original] : answers) {
|
for (auto &&[answer, original] : answers) {
|
||||||
answer.fillText(original);
|
answer.fillData(_poll, original);
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_answers = ranges::view::all(
|
_answers = ranges::view::all(
|
||||||
_poll->answers
|
_poll->answers
|
||||||
) | 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.fillText(answer);
|
result.fillData(_poll, answer);
|
||||||
return result;
|
return result;
|
||||||
}) | ranges::to_vector;
|
}) | ranges::to_vector;
|
||||||
|
|
||||||
|
@ -592,6 +608,8 @@ int Poll::paintAnswer(
|
||||||
p.setOpacity(sqrt(opacity));
|
p.setOpacity(sqrt(opacity));
|
||||||
paintFilling(
|
paintFilling(
|
||||||
p,
|
p,
|
||||||
|
animation->chosen,
|
||||||
|
animation->correct,
|
||||||
animation->filling.current(),
|
animation->filling.current(),
|
||||||
left,
|
left,
|
||||||
top,
|
top,
|
||||||
|
@ -613,6 +631,8 @@ int Poll::paintAnswer(
|
||||||
selection);
|
selection);
|
||||||
paintFilling(
|
paintFilling(
|
||||||
p,
|
p,
|
||||||
|
answer.chosen,
|
||||||
|
answer.correct,
|
||||||
answer.filling,
|
answer.filling,
|
||||||
left,
|
left,
|
||||||
top,
|
top,
|
||||||
|
@ -696,6 +716,8 @@ void Poll::paintPercent(
|
||||||
|
|
||||||
void Poll::paintFilling(
|
void Poll::paintFilling(
|
||||||
Painter &p,
|
Painter &p,
|
||||||
|
bool chosen,
|
||||||
|
bool correct,
|
||||||
float64 filling,
|
float64 filling,
|
||||||
int left,
|
int left,
|
||||||
int top,
|
int top,
|
||||||
|
@ -712,15 +734,29 @@ void Poll::paintFilling(
|
||||||
|
|
||||||
top += st::historyPollAnswerPadding.top();
|
top += st::historyPollAnswerPadding.top();
|
||||||
|
|
||||||
const auto bar = outbg ? (selected ? st::msgWaveformOutActiveSelected : st::msgWaveformOutActive) : (selected ? st::msgWaveformInActiveSelected : st::msgWaveformInActive);
|
|
||||||
PainterHighQualityEnabler hq(p);
|
PainterHighQualityEnabler hq(p);
|
||||||
p.setPen(Qt::NoPen);
|
p.setPen(Qt::NoPen);
|
||||||
p.setBrush(bar);
|
const auto thickness = st::historyPollFillingHeight;
|
||||||
const auto max = awidth - st::historyPollFillingRight;
|
const auto max = awidth - st::historyPollFillingRight;
|
||||||
const auto size = anim::interpolate(st::historyPollFillingMin, max, filling);
|
const auto size = anim::interpolate(st::historyPollFillingMin, max, filling);
|
||||||
const auto radius = st::historyPollFillingRadius;
|
const auto radius = st::historyPollFillingRadius;
|
||||||
const auto ftop = bottom - st::historyPollFillingBottom - st::historyPollFillingHeight;
|
const auto ftop = bottom - st::historyPollFillingBottom - thickness;
|
||||||
p.drawRoundedRect(aleft, ftop, size, st::historyPollFillingHeight, radius, radius);
|
|
||||||
|
if (chosen && !correct) {
|
||||||
|
p.setBrush(st::boxTextFgError);
|
||||||
|
} else {
|
||||||
|
const auto bar = outbg ? (selected ? st::msgWaveformOutActiveSelected : st::msgWaveformOutActive) : (selected ? st::msgWaveformInActiveSelected : st::msgWaveformInActive);
|
||||||
|
p.setBrush(bar);
|
||||||
|
}
|
||||||
|
auto barleft = aleft;
|
||||||
|
auto barwidth = size;
|
||||||
|
if (chosen || correct) {
|
||||||
|
p.drawEllipse(aleft, ftop - thickness, thickness * 3, thickness * 3);
|
||||||
|
barleft += thickness * 3 - radius;
|
||||||
|
barwidth -= thickness * 3 - radius;
|
||||||
|
}
|
||||||
|
|
||||||
|
p.drawRoundedRect(barleft, ftop, barwidth, thickness, radius, radius);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Poll::answerVotesChanged() const {
|
bool Poll::answerVotesChanged() const {
|
||||||
|
@ -748,6 +784,8 @@ void Poll::saveStateInAnimation() const {
|
||||||
result.percent = show ? float64(answer.votesPercent) : 0.;
|
result.percent = show ? float64(answer.votesPercent) : 0.;
|
||||||
result.filling = show ? answer.filling : 0.;
|
result.filling = show ? answer.filling : 0.;
|
||||||
result.opacity = show ? 1. : 0.;
|
result.opacity = show ? 1. : 0.;
|
||||||
|
result.chosen = answer.chosen;
|
||||||
|
result.correct = answer.correct;
|
||||||
return result;
|
return result;
|
||||||
};
|
};
|
||||||
ranges::transform(
|
ranges::transform(
|
||||||
|
@ -761,7 +799,7 @@ bool Poll::checkAnimationStart() const {
|
||||||
// Skip initial changes.
|
// Skip initial changes.
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
const auto result = (showVotes() != (_poll->voted() || _poll->closed))
|
const auto result = (showVotes() != (_poll->voted() || _poll->closed()))
|
||||||
|| answerVotesChanged();
|
|| answerVotesChanged();
|
||||||
if (result) {
|
if (result) {
|
||||||
saveStateInAnimation();
|
saveStateInAnimation();
|
||||||
|
@ -780,6 +818,8 @@ void Poll::startAnswersAnimation() const {
|
||||||
data.percent.start(show ? float64(answer.votesPercent) : 0.);
|
data.percent.start(show ? float64(answer.votesPercent) : 0.);
|
||||||
data.filling.start(show ? answer.filling : 0.);
|
data.filling.start(show ? answer.filling : 0.);
|
||||||
data.opacity.start(show ? 1. : 0.);
|
data.opacity.start(show ? 1. : 0.);
|
||||||
|
data.chosen = data.chosen || answer.chosen;
|
||||||
|
data.correct = data.correct || answer.correct;
|
||||||
}
|
}
|
||||||
_answersAnimation->progress.start(
|
_answersAnimation->progress.start(
|
||||||
[=] { history()->owner().requestViewRepaint(_parent); },
|
[=] { history()->owner().requestViewRepaint(_parent); },
|
||||||
|
|
|
@ -8,8 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "history/view/media/history_view_media.h"
|
#include "history/view/media/history_view_media.h"
|
||||||
|
#include "data/data_poll.h"
|
||||||
struct PollAnswer;
|
|
||||||
|
|
||||||
namespace HistoryView {
|
namespace HistoryView {
|
||||||
|
|
||||||
|
@ -99,6 +98,8 @@ private:
|
||||||
TextSelection selection) const;
|
TextSelection selection) const;
|
||||||
void paintFilling(
|
void paintFilling(
|
||||||
Painter &p,
|
Painter &p,
|
||||||
|
bool chosen,
|
||||||
|
bool correct,
|
||||||
float64 filling,
|
float64 filling,
|
||||||
int left,
|
int left,
|
||||||
int top,
|
int top,
|
||||||
|
@ -115,11 +116,11 @@ private:
|
||||||
|
|
||||||
void toggleRipple(Answer &answer, bool pressed);
|
void toggleRipple(Answer &answer, bool pressed);
|
||||||
|
|
||||||
not_null<PollData*> _poll;
|
const not_null<PollData*> _poll;
|
||||||
int _pollVersion = 0;
|
int _pollVersion = 0;
|
||||||
int _totalVotes = 0;
|
int _totalVotes = 0;
|
||||||
bool _voted = false;
|
bool _voted = false;
|
||||||
bool _closed = false;
|
PollData::Flags _flags = PollData::Flags();
|
||||||
|
|
||||||
Ui::Text::String _question;
|
Ui::Text::String _question;
|
||||||
Ui::Text::String _subtitle;
|
Ui::Text::String _subtitle;
|
||||||
|
|
Loading…
Reference in New Issue