mirror of https://github.com/procxx/kepka.git
Always display nice percent values.
Sum of percent values should never exceed 100%. If any two answers received same amount of votes, they should show same percent values. This way sum could be less than 100% (three answers, one vote each), but this looks better than giving extra vote to some random answer.
This commit is contained in:
parent
6fc4facddf
commit
8708a001c7
|
@ -26,7 +26,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
constexpr auto kQuestionLimit = 255;
|
constexpr auto kQuestionLimit = 255;
|
||||||
constexpr auto kMaxOptionsCount = 10;
|
constexpr auto kMaxOptionsCount = PollData::kMaxOptions;
|
||||||
constexpr auto kOptionLimit = 100;
|
constexpr auto kOptionLimit = 100;
|
||||||
constexpr auto kWarnQuestionLimit = 80;
|
constexpr auto kWarnQuestionLimit = 80;
|
||||||
constexpr auto kWarnOptionLimit = 30;
|
constexpr auto kWarnOptionLimit = 30;
|
||||||
|
|
|
@ -51,7 +51,9 @@ bool PollData::applyChanges(const MTPDpoll &poll) {
|
||||||
result.text = qs(answer.vtext);
|
result.text = qs(answer.vtext);
|
||||||
return result;
|
return result;
|
||||||
});
|
});
|
||||||
}) | ranges::to_vector;
|
}) | ranges::view::take(
|
||||||
|
kMaxOptions
|
||||||
|
) | ranges::to_vector;
|
||||||
|
|
||||||
const auto changed1 = (question != newQuestion)
|
const auto changed1 = (question != newQuestion)
|
||||||
|| (closed != newClosed);
|
|| (closed != newClosed);
|
||||||
|
@ -78,6 +80,8 @@ bool PollData::applyChanges(const MTPDpoll &poll) {
|
||||||
|
|
||||||
bool PollData::applyResults(const MTPPollResults &results) {
|
bool PollData::applyResults(const MTPPollResults &results) {
|
||||||
return results.match([&](const MTPDpollResults &results) {
|
return results.match([&](const MTPDpollResults &results) {
|
||||||
|
lastResultsUpdate = getms();
|
||||||
|
|
||||||
const auto newTotalVoters = results.has_total_voters()
|
const auto newTotalVoters = results.has_total_voters()
|
||||||
? results.vtotal_voters.v
|
? results.vtotal_voters.v
|
||||||
: totalVoters;
|
: totalVoters;
|
||||||
|
@ -89,8 +93,11 @@ bool PollData::applyResults(const MTPPollResults &results) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (!changed) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
totalVoters = newTotalVoters;
|
totalVoters = newTotalVoters;
|
||||||
lastResultsUpdate = getms();
|
++version;
|
||||||
return changed;
|
return changed;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,6 +45,8 @@ struct PollData {
|
||||||
|
|
||||||
int version = 0;
|
int version = 0;
|
||||||
|
|
||||||
|
static constexpr auto kMaxOptions = 10;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool applyResultToAnswers(
|
bool applyResultToAnswers(
|
||||||
const MTPPollAnswerVoters &result,
|
const MTPPollAnswerVoters &result,
|
||||||
|
|
|
@ -1523,7 +1523,7 @@ not_null<PollData*> Session::poll(const MTPDmessageMediaPoll &data) {
|
||||||
const auto result = poll(data.vpoll);
|
const auto result = poll(data.vpoll);
|
||||||
const auto changed = result->applyResults(data.vresults);
|
const auto changed = result->applyResults(data.vresults);
|
||||||
if (changed) {
|
if (changed) {
|
||||||
requestPollViewRepaint(result);
|
notifyPollUpdateDelayed(result);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -1538,7 +1538,7 @@ void Session::applyPollUpdate(const MTPDupdateMessagePoll &update) {
|
||||||
: i->second.get();
|
: i->second.get();
|
||||||
}();
|
}();
|
||||||
if (updated && updated->applyResults(update.vresults)) {
|
if (updated && updated->applyResults(update.vresults)) {
|
||||||
requestPollViewRepaint(updated);
|
notifyPollUpdateDelayed(updated);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -61,6 +61,73 @@ FormattedLargeNumber FormatLargeNumber(int64 number) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct PercentCounterItem {
|
||||||
|
int index = 0;
|
||||||
|
int percent = 0;
|
||||||
|
int remainder = 0;
|
||||||
|
|
||||||
|
inline bool operator<(const PercentCounterItem &other) const {
|
||||||
|
if (remainder > other.remainder) {
|
||||||
|
return true;
|
||||||
|
} else if (remainder < other.remainder) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return percent < other.percent;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void AdjustPercentCount(gsl::span<PercentCounterItem> items, int left) {
|
||||||
|
ranges::sort(items, std::less<>());
|
||||||
|
for (auto i = 0, count = int(items.size()); i != count;) {
|
||||||
|
const auto &item = items[i];
|
||||||
|
auto j = i + 1;
|
||||||
|
for (; j != count; ++j) {
|
||||||
|
if (items[j].percent != item.percent
|
||||||
|
|| items[j].remainder != item.remainder) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const auto equal = j - i;
|
||||||
|
if (equal <= left) {
|
||||||
|
left -= equal;
|
||||||
|
for (; i != j; ++i) {
|
||||||
|
++items[i].percent;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
i = j;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CountNicePercent(
|
||||||
|
gsl::span<const int> votes,
|
||||||
|
int total,
|
||||||
|
gsl::span<int> result) {
|
||||||
|
Expects(result.size() >= votes.size());
|
||||||
|
Expects(votes.size() <= PollData::kMaxOptions);
|
||||||
|
|
||||||
|
const auto count = size_type(votes.size());
|
||||||
|
PercentCounterItem ItemsStorage[PollData::kMaxOptions];
|
||||||
|
const auto items = gsl::make_span(ItemsStorage).subspan(0, count);
|
||||||
|
auto left = 100;
|
||||||
|
auto &&zipped = ranges::view::zip(
|
||||||
|
votes,
|
||||||
|
items,
|
||||||
|
ranges::view::ints(0));
|
||||||
|
for (auto &&[votes, item, index] : zipped) {
|
||||||
|
item.index = index;
|
||||||
|
item.percent = (votes * 100) / total;
|
||||||
|
item.remainder = (votes * 100) - (item.percent * total);
|
||||||
|
left -= item.percent;
|
||||||
|
}
|
||||||
|
if (left > 0 && left <= count) {
|
||||||
|
AdjustPercentCount(items, left);
|
||||||
|
}
|
||||||
|
for (const auto &item : items) {
|
||||||
|
result[item.index] = item.percent;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
struct HistoryPoll::AnswerAnimation {
|
struct HistoryPoll::AnswerAnimation {
|
||||||
|
@ -90,11 +157,12 @@ struct HistoryPoll::Answer {
|
||||||
|
|
||||||
Text text;
|
Text text;
|
||||||
QByteArray option;
|
QByteArray option;
|
||||||
mutable int votes = 0;
|
int votes = 0;
|
||||||
mutable int votesPercentWidth = 0;
|
int votesPercent = 0;
|
||||||
mutable float64 filling = 0.;
|
int votesPercentWidth = 0;
|
||||||
mutable QString votesPercent;
|
float64 filling = 0.;
|
||||||
mutable bool chosen = false;
|
QString votesPercentString;
|
||||||
|
bool chosen = false;
|
||||||
ClickHandlerPtr handler;
|
ClickHandlerPtr handler;
|
||||||
mutable std::unique_ptr<Ui::RippleAnimation> ripple;
|
mutable std::unique_ptr<Ui::RippleAnimation> ripple;
|
||||||
};
|
};
|
||||||
|
@ -247,14 +315,18 @@ void HistoryPoll::updateTexts() {
|
||||||
|
|
||||||
const auto willStartAnimation = checkAnimationStart();
|
const auto willStartAnimation = checkAnimationStart();
|
||||||
|
|
||||||
_closed = _poll->closed;
|
if (_question.originalText() != _poll->question) {
|
||||||
_question.setText(
|
_question.setText(
|
||||||
st::historyPollQuestionStyle,
|
st::historyPollQuestionStyle,
|
||||||
_poll->question,
|
_poll->question,
|
||||||
Ui::WebpageTextTitleOptions());
|
Ui::WebpageTextTitleOptions());
|
||||||
_subtitle.setText(
|
}
|
||||||
st::msgDateTextStyle,
|
if (_closed != _poll->closed || _subtitle.isEmpty()) {
|
||||||
lang(_closed ? lng_polls_closed : lng_polls_anonymous));
|
_closed = _poll->closed;
|
||||||
|
_subtitle.setText(
|
||||||
|
st::msgDateTextStyle,
|
||||||
|
lang(_closed ? lng_polls_closed : lng_polls_anonymous));
|
||||||
|
}
|
||||||
|
|
||||||
updateAnswers();
|
updateAnswers();
|
||||||
updateVotes();
|
updateVotes();
|
||||||
|
@ -303,19 +375,13 @@ ClickHandlerPtr HistoryPoll::createAnswerClickHandler(
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryPoll::updateVotes() const {
|
void HistoryPoll::updateVotes() {
|
||||||
_voted = _poll->voted();
|
_voted = _poll->voted();
|
||||||
updateAnswerVotes();
|
updateAnswerVotes();
|
||||||
updateTotalVotes();
|
updateTotalVotes();
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryPoll::updateVotesCheckAnimations() const {
|
void HistoryPoll::checkSendingAnimation() const {
|
||||||
const auto willStartAnimation = checkAnimationStart();
|
|
||||||
updateVotes();
|
|
||||||
if (willStartAnimation) {
|
|
||||||
startAnswersAnimation();
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto &sending = _poll->sendingVote;
|
const auto &sending = _poll->sendingVote;
|
||||||
if (sending.isEmpty() == !_sendingAnimation) {
|
if (sending.isEmpty() == !_sendingAnimation) {
|
||||||
if (_sendingAnimation) {
|
if (_sendingAnimation) {
|
||||||
|
@ -337,7 +403,7 @@ void HistoryPoll::updateVotesCheckAnimations() const {
|
||||||
_sendingAnimation->animation.start();
|
_sendingAnimation->animation.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryPoll::updateTotalVotes() const {
|
void HistoryPoll::updateTotalVotes() {
|
||||||
if (_totalVotes == _poll->totalVoters && !_totalVotesLabel.isEmpty()) {
|
if (_totalVotes == _poll->totalVoters && !_totalVotesLabel.isEmpty()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -357,26 +423,26 @@ void HistoryPoll::updateTotalVotes() const {
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryPoll::updateAnswerVotesFromOriginal(
|
void HistoryPoll::updateAnswerVotesFromOriginal(
|
||||||
const Answer &answer,
|
Answer &answer,
|
||||||
const PollAnswer &original,
|
const PollAnswer &original,
|
||||||
int totalVotes,
|
int percent,
|
||||||
int maxVotes) const {
|
int maxVotes) {
|
||||||
if (canVote()) {
|
if (canVote()) {
|
||||||
answer.votesPercent.clear();
|
answer.votesPercent = 0;
|
||||||
} else if (answer.votes != original.votes
|
answer.votesPercentString.clear();
|
||||||
|| answer.votesPercent.isEmpty()
|
answer.votesPercentWidth = 0;
|
||||||
|| std::max(_totalVotes, 1) != totalVotes) {
|
} else if (answer.votesPercentString.isEmpty()
|
||||||
const auto percent = int(std::round(
|
|| answer.votesPercent != percent) {
|
||||||
original.votes * 100. / totalVotes));
|
answer.votesPercent = percent;
|
||||||
answer.votesPercent = QString::number(percent) + '%';
|
answer.votesPercentString = QString::number(percent) + '%';
|
||||||
answer.votesPercentWidth = st::historyPollPercentFont->width(
|
answer.votesPercentWidth = st::historyPollPercentFont->width(
|
||||||
answer.votesPercent);
|
answer.votesPercentString);
|
||||||
}
|
}
|
||||||
answer.votes = original.votes;
|
answer.votes = original.votes;
|
||||||
answer.filling = answer.votes / float64(maxVotes);
|
answer.filling = answer.votes / float64(maxVotes);
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryPoll::updateAnswerVotes() const {
|
void HistoryPoll::updateAnswerVotes() {
|
||||||
if (_poll->answers.size() != _answers.size()
|
if (_poll->answers.size() != _answers.size()
|
||||||
|| _poll->answers.empty()) {
|
|| _poll->answers.empty()) {
|
||||||
return;
|
return;
|
||||||
|
@ -386,12 +452,33 @@ void HistoryPoll::updateAnswerVotes() const {
|
||||||
_poll->answers,
|
_poll->answers,
|
||||||
ranges::less(),
|
ranges::less(),
|
||||||
&PollAnswer::votes)->votes);
|
&PollAnswer::votes)->votes);
|
||||||
auto &&answers = ranges::view::zip(_answers, _poll->answers);
|
|
||||||
for (auto &&[answer, original] : answers) {
|
constexpr auto kMaxCount = PollData::kMaxOptions;
|
||||||
|
const auto count = size_type(_poll->answers.size());
|
||||||
|
Assert(count <= kMaxCount);
|
||||||
|
int PercentsStorage[kMaxCount] = { 0 };
|
||||||
|
int VotesStorage[kMaxCount] = { 0 };
|
||||||
|
|
||||||
|
ranges::copy(
|
||||||
|
ranges::view::all(
|
||||||
|
_poll->answers
|
||||||
|
) | ranges::view::transform(&PollAnswer::votes),
|
||||||
|
ranges::begin(VotesStorage));
|
||||||
|
|
||||||
|
CountNicePercent(
|
||||||
|
gsl::make_span(VotesStorage).subspan(0, count),
|
||||||
|
totalVotes,
|
||||||
|
gsl::make_span(PercentsStorage).subspan(0, count));
|
||||||
|
|
||||||
|
auto &&answers = ranges::view::zip(
|
||||||
|
_answers,
|
||||||
|
_poll->answers,
|
||||||
|
PercentsStorage);
|
||||||
|
for (auto &&[answer, original, percent] : answers) {
|
||||||
updateAnswerVotesFromOriginal(
|
updateAnswerVotesFromOriginal(
|
||||||
answer,
|
answer,
|
||||||
original,
|
original,
|
||||||
totalVotes,
|
percent,
|
||||||
maxVotes);
|
maxVotes);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -400,7 +487,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();
|
||||||
|
|
||||||
updateVotesCheckAnimations();
|
checkSendingAnimation();
|
||||||
_poll->checkResultsReload(_parent->data(), ms);
|
_poll->checkResultsReload(_parent->data(), ms);
|
||||||
|
|
||||||
const auto outbg = _parent->hasOutLayout();
|
const auto outbg = _parent->hasOutLayout();
|
||||||
|
@ -542,7 +629,7 @@ int HistoryPoll::paintAnswer(
|
||||||
} else {
|
} else {
|
||||||
paintPercent(
|
paintPercent(
|
||||||
p,
|
p,
|
||||||
answer.votesPercent,
|
answer.votesPercentString,
|
||||||
answer.votesPercentWidth,
|
answer.votesPercentWidth,
|
||||||
left,
|
left,
|
||||||
top,
|
top,
|
||||||
|
@ -682,9 +769,7 @@ void HistoryPoll::saveStateInAnimation() const {
|
||||||
_answersAnimation->data.reserve(_answers.size());
|
_answersAnimation->data.reserve(_answers.size());
|
||||||
const auto convert = [&](const Answer &answer) {
|
const auto convert = [&](const Answer &answer) {
|
||||||
auto result = AnswerAnimation();
|
auto result = AnswerAnimation();
|
||||||
result.percent = can
|
result.percent = can ? 0. : float64(answer.votesPercent);
|
||||||
? 0.
|
|
||||||
: (answer.votes * 100. / std::max(_totalVotes, 1));
|
|
||||||
result.filling = can ? 0. : answer.filling;
|
result.filling = can ? 0. : answer.filling;
|
||||||
result.opacity = can ? 0. : 1.;
|
result.opacity = can ? 0. : 1.;
|
||||||
return result;
|
return result;
|
||||||
|
@ -716,9 +801,7 @@ void HistoryPoll::startAnswersAnimation() const {
|
||||||
const auto can = canVote();
|
const auto can = canVote();
|
||||||
auto &&both = ranges::view::zip(_answers, _answersAnimation->data);
|
auto &&both = ranges::view::zip(_answers, _answersAnimation->data);
|
||||||
for (auto &&[answer, data] : both) {
|
for (auto &&[answer, data] : both) {
|
||||||
data.percent.start(can
|
data.percent.start(can ? 0. : float64(answer.votesPercent));
|
||||||
? 0.
|
|
||||||
: answer.votes * 100. / std::max(_totalVotes, 1));
|
|
||||||
data.filling.start(can ? 0. : answer.filling);
|
data.filling.start(can ? 0. : answer.filling);
|
||||||
data.opacity.start(can ? 0. : 1.);
|
data.opacity.start(can ? 0. : 1.);
|
||||||
}
|
}
|
||||||
|
|
|
@ -61,15 +61,15 @@ private:
|
||||||
const Answer &answer) const;
|
const Answer &answer) const;
|
||||||
void updateTexts();
|
void updateTexts();
|
||||||
void updateAnswers();
|
void updateAnswers();
|
||||||
void updateVotes() const;
|
void updateVotes();
|
||||||
void updateTotalVotes() const;
|
void updateTotalVotes();
|
||||||
void updateAnswerVotes() const;
|
void updateAnswerVotes();
|
||||||
void updateAnswerVotesFromOriginal(
|
void updateAnswerVotesFromOriginal(
|
||||||
const Answer &answer,
|
Answer &answer,
|
||||||
const PollAnswer &original,
|
const PollAnswer &original,
|
||||||
int totalVotes,
|
int percent,
|
||||||
int maxVotes) const;
|
int maxVotes);
|
||||||
void updateVotesCheckAnimations() const;
|
void checkSendingAnimation() const;
|
||||||
|
|
||||||
int paintAnswer(
|
int paintAnswer(
|
||||||
Painter &p,
|
Painter &p,
|
||||||
|
@ -115,14 +115,14 @@ private:
|
||||||
|
|
||||||
not_null<PollData*> _poll;
|
not_null<PollData*> _poll;
|
||||||
int _pollVersion = 0;
|
int _pollVersion = 0;
|
||||||
mutable int _totalVotes = 0;
|
int _totalVotes = 0;
|
||||||
mutable bool _voted = false;
|
bool _voted = false;
|
||||||
bool _closed = false;
|
bool _closed = false;
|
||||||
|
|
||||||
Text _question;
|
Text _question;
|
||||||
Text _subtitle;
|
Text _subtitle;
|
||||||
std::vector<Answer> _answers;
|
std::vector<Answer> _answers;
|
||||||
mutable Text _totalVotesLabel;
|
Text _totalVotesLabel;
|
||||||
|
|
||||||
mutable std::unique_ptr<AnswersAnimation> _answersAnimation;
|
mutable std::unique_ptr<AnswersAnimation> _answersAnimation;
|
||||||
mutable std::unique_ptr<SendingAnimation> _sendingAnimation;
|
mutable std::unique_ptr<SendingAnimation> _sendingAnimation;
|
||||||
|
|
Loading…
Reference in New Issue