mirror of https://github.com/procxx/kepka.git
Implement polls voting and actions.
This commit is contained in:
parent
4bb5dcf50c
commit
74c1db740d
|
@ -1832,6 +1832,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
"lng_polls_anonymous" = "Anonymous Poll";
|
"lng_polls_anonymous" = "Anonymous Poll";
|
||||||
"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";
|
||||||
|
"lng_polls_retract" = "Retract vote";
|
||||||
|
"lng_polls_stop" = "Stop poll";
|
||||||
|
"lng_polls_stop_warning" = "If you stop this poll now, nobody will be able to vote in it anymore. This action cannot be undone.";
|
||||||
|
"lng_polls_stop_sure" = "Stop";
|
||||||
|
|
||||||
// Wnd specific
|
// Wnd specific
|
||||||
|
|
||||||
|
|
|
@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "data/data_drafts.h"
|
#include "data/data_drafts.h"
|
||||||
#include "data/data_photo.h"
|
#include "data/data_photo.h"
|
||||||
#include "data/data_web_page.h"
|
#include "data/data_web_page.h"
|
||||||
|
#include "data/data_poll.h"
|
||||||
#include "data/data_feed.h"
|
#include "data/data_feed.h"
|
||||||
#include "data/data_media_types.h"
|
#include "data/data_media_types.h"
|
||||||
#include "data/data_sparse_ids.h"
|
#include "data/data_sparse_ids.h"
|
||||||
|
@ -4751,35 +4752,35 @@ void ApiWrap::sendExistingDocument(
|
||||||
};
|
};
|
||||||
performRequest();
|
performRequest();
|
||||||
|
|
||||||
//AssertIsDebug();
|
AssertIsDebug();
|
||||||
//auto answers = QVector<MTPPollAnswer>{
|
auto answers = QVector<MTPPollAnswer>{
|
||||||
// MTP_pollAnswer(MTP_string("first option"), MTP_bytes("a")),
|
MTP_pollAnswer(MTP_string("first option"), MTP_bytes("a")),
|
||||||
// MTP_pollAnswer(MTP_string("second option"), MTP_bytes("b")),
|
MTP_pollAnswer(MTP_string("second option"), MTP_bytes("b")),
|
||||||
// MTP_pollAnswer(MTP_string("third very very very very very very "
|
MTP_pollAnswer(MTP_string("third very very very very very very "
|
||||||
// "very very very very very very option"), MTP_bytes("c")),
|
"very very very very very very option"), MTP_bytes("c")),
|
||||||
// MTP_pollAnswer(MTP_string("fourth option"), MTP_bytes("d")),
|
MTP_pollAnswer(MTP_string("fourth option"), MTP_bytes("d")),
|
||||||
//};
|
};
|
||||||
//history->sendRequestId = request(MTPmessages_SendMedia(
|
history->sendRequestId = request(MTPmessages_SendMedia(
|
||||||
// MTP_flags(sendFlags),
|
MTP_flags(sendFlags),
|
||||||
// peer->input,
|
peer->input,
|
||||||
// MTP_int(replyTo),
|
MTP_int(replyTo),
|
||||||
// MTP_inputMediaPoll(
|
MTP_inputMediaPoll(
|
||||||
// MTP_poll(
|
MTP_poll(
|
||||||
// MTP_long(rand_value<uint64>()),
|
MTP_long(rand_value<uint64>()),
|
||||||
// MTP_flags(0),
|
MTP_flags(0),
|
||||||
// MTP_string("Very very very very very very very very very very "
|
MTP_string("Very very very very very very very very very very "
|
||||||
// "very very very long poll question text"),
|
"very very very long poll question text"),
|
||||||
// MTP_vector<MTPPollAnswer>(answers))),
|
MTP_vector<MTPPollAnswer>(answers))),
|
||||||
// MTP_string(captionText),
|
MTP_string(captionText),
|
||||||
// MTP_long(rand_value<uint64>()),
|
MTP_long(rand_value<uint64>()),
|
||||||
// MTPnullMarkup,
|
MTPnullMarkup,
|
||||||
// sentEntities
|
sentEntities
|
||||||
//)).done([=](const MTPUpdates &result) {
|
)).done([=](const MTPUpdates &result) {
|
||||||
// applyUpdates(result);
|
applyUpdates(result);
|
||||||
//}).fail(
|
}).fail(
|
||||||
// base::duplicate(*failHandler)
|
base::duplicate(*failHandler)
|
||||||
//).afterRequest(history->sendRequestId
|
).afterRequest(history->sendRequestId
|
||||||
//).send();
|
).send();
|
||||||
|
|
||||||
if (const auto main = App::main()) {
|
if (const auto main = App::main()) {
|
||||||
main->finishForwarding(history);
|
main->finishForwarding(history);
|
||||||
|
@ -5343,6 +5344,81 @@ void ApiWrap::setSelfDestructDays(int days) {
|
||||||
_selfDestructChanges.fire_copy(days);
|
_selfDestructChanges.fire_copy(days);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ApiWrap::sendPollVotes(
|
||||||
|
FullMsgId itemId,
|
||||||
|
const std::vector<QByteArray> &options) {
|
||||||
|
if (_pollVotesRequestIds.contains(itemId)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const auto item = App::histItemById(itemId);
|
||||||
|
if (!item) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto prepared = QVector<MTPbytes>();
|
||||||
|
prepared.reserve(options.size());
|
||||||
|
ranges::transform(
|
||||||
|
options,
|
||||||
|
ranges::back_inserter(prepared),
|
||||||
|
[](const QByteArray &option) { return MTP_bytes(option); });
|
||||||
|
const auto requestId = request(MTPmessages_SendVote(
|
||||||
|
item->history()->peer->input,
|
||||||
|
MTP_int(item->id),
|
||||||
|
MTP_vector<MTPbytes>(prepared)
|
||||||
|
)).done([=](const MTPUpdates &result) {
|
||||||
|
_pollVotesRequestIds.erase(itemId);
|
||||||
|
applyUpdates(result);
|
||||||
|
}).fail([=](const RPCError &error) {
|
||||||
|
_pollVotesRequestIds.erase(itemId);
|
||||||
|
}).send();
|
||||||
|
_pollVotesRequestIds.emplace(itemId, requestId);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ApiWrap::closePoll(FullMsgId itemId) {
|
||||||
|
if (_pollCloseRequestIds.contains(itemId)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const auto item = App::histItemById(itemId);
|
||||||
|
const auto media = item ? item->media() : nullptr;
|
||||||
|
const auto poll = media ? media->poll() : nullptr;
|
||||||
|
if (!poll) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto convert = [](const PollAnswer &answer) {
|
||||||
|
return MTP_pollAnswer(
|
||||||
|
MTP_string(answer.text),
|
||||||
|
MTP_bytes(answer.option));
|
||||||
|
};
|
||||||
|
|
||||||
|
auto answers = QVector<MTPPollAnswer>();
|
||||||
|
answers.reserve(poll->answers.size());
|
||||||
|
ranges::transform(
|
||||||
|
poll->answers,
|
||||||
|
ranges::back_inserter(answers),
|
||||||
|
convert);
|
||||||
|
|
||||||
|
const auto requestId = request(MTPmessages_EditMessage(
|
||||||
|
MTP_flags(MTPmessages_EditMessage::Flag::f_media),
|
||||||
|
item->history()->peer->input,
|
||||||
|
MTP_int(item->id),
|
||||||
|
MTPstring(),
|
||||||
|
MTP_inputMediaPoll(MTP_poll(
|
||||||
|
MTP_long(poll->id),
|
||||||
|
MTP_flags(MTPDpoll::Flag::f_closed),
|
||||||
|
MTP_string(poll->question),
|
||||||
|
MTP_vector<MTPPollAnswer>(answers))),
|
||||||
|
MTPReplyMarkup(),
|
||||||
|
MTPVector<MTPMessageEntity>()
|
||||||
|
)).done([=](const MTPUpdates &result) {
|
||||||
|
_pollCloseRequestIds.erase(itemId);
|
||||||
|
applyUpdates(result);
|
||||||
|
}).fail([=](const RPCError &error) {
|
||||||
|
_pollCloseRequestIds.erase(itemId);
|
||||||
|
}).send();
|
||||||
|
_pollCloseRequestIds.emplace(itemId, requestId);
|
||||||
|
}
|
||||||
|
|
||||||
void ApiWrap::readServerHistory(not_null<History*> history) {
|
void ApiWrap::readServerHistory(not_null<History*> history) {
|
||||||
if (history->unreadCount()) {
|
if (history->unreadCount()) {
|
||||||
readServerHistoryForce(history);
|
readServerHistoryForce(history);
|
||||||
|
|
|
@ -377,6 +377,11 @@ public:
|
||||||
rpl::producer<int> selfDestructValue() const;
|
rpl::producer<int> selfDestructValue() const;
|
||||||
void saveSelfDestruct(int days);
|
void saveSelfDestruct(int days);
|
||||||
|
|
||||||
|
void sendPollVotes(
|
||||||
|
FullMsgId itemId,
|
||||||
|
const std::vector<QByteArray> &options);
|
||||||
|
void closePoll(FullMsgId itemId);
|
||||||
|
|
||||||
~ApiWrap();
|
~ApiWrap();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -739,4 +744,7 @@ private:
|
||||||
std::optional<int> _selfDestructDays;
|
std::optional<int> _selfDestructDays;
|
||||||
rpl::event_stream<int> _selfDestructChanges;
|
rpl::event_stream<int> _selfDestructChanges;
|
||||||
|
|
||||||
|
base::flat_map<FullMsgId, mtpRequestId> _pollVotesRequestIds;
|
||||||
|
base::flat_map<FullMsgId, mtpRequestId> _pollCloseRequestIds;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -33,6 +33,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "data/data_document.h"
|
#include "data/data_document.h"
|
||||||
#include "data/data_game.h"
|
#include "data/data_game.h"
|
||||||
#include "data/data_web_page.h"
|
#include "data/data_web_page.h"
|
||||||
|
#include "data/data_poll.h"
|
||||||
#include "lang/lang_keys.h"
|
#include "lang/lang_keys.h"
|
||||||
#include "auth_session.h"
|
#include "auth_session.h"
|
||||||
#include "layout.h"
|
#include "layout.h"
|
||||||
|
@ -1092,7 +1093,7 @@ GameData *MediaGame::game() const {
|
||||||
}
|
}
|
||||||
|
|
||||||
QString MediaGame::pinnedTextSubstring() const {
|
QString MediaGame::pinnedTextSubstring() const {
|
||||||
auto title = _game->title;
|
const auto title = _game->title;
|
||||||
return lng_action_pinned_media_game(lt_game, title);
|
return lng_action_pinned_media_game(lt_game, title);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1224,7 +1225,7 @@ QString MediaPoll::notificationText() const {
|
||||||
}
|
}
|
||||||
|
|
||||||
QString MediaPoll::pinnedTextSubstring() const {
|
QString MediaPoll::pinnedTextSubstring() const {
|
||||||
return QString(); // #TODO polls
|
return QChar(171) + _poll->question + QChar(187);
|
||||||
}
|
}
|
||||||
|
|
||||||
TextWithEntities MediaPoll::clipboardText() const {
|
TextWithEntities MediaPoll::clipboardText() const {
|
||||||
|
|
|
@ -121,3 +121,7 @@ bool PollData::applyResultToAnswers(
|
||||||
return changed;
|
return changed;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool PollData::voted() const {
|
||||||
|
return ranges::find(answers, true, &PollAnswer::chosen) != end(answers);
|
||||||
|
}
|
||||||
|
|
|
@ -32,6 +32,8 @@ struct PollData {
|
||||||
PollAnswer *answerByOption(const QByteArray &option);
|
PollAnswer *answerByOption(const QByteArray &option);
|
||||||
const PollAnswer *answerByOption(const QByteArray &option) const;
|
const PollAnswer *answerByOption(const QByteArray &option) const;
|
||||||
|
|
||||||
|
bool voted() const;
|
||||||
|
|
||||||
PollId id = 0;
|
PollId id = 0;
|
||||||
QString question;
|
QString question;
|
||||||
std::vector<PollAnswer> answers;
|
std::vector<PollAnswer> answers;
|
||||||
|
|
|
@ -492,12 +492,28 @@ webPageDescriptionStyle: defaultTextStyle;
|
||||||
webPagePhotoSize: 100px;
|
webPagePhotoSize: 100px;
|
||||||
webPagePhotoDelta: 8px;
|
webPagePhotoDelta: 8px;
|
||||||
|
|
||||||
historyPollQuestionStyle: semiboldTextStyle;
|
historyPollQuestionFont: font(boxFontSize semibold);
|
||||||
historyPollAnswerStyle: defaultTextStyle;
|
historyPollQuestionStyle: TextStyle(defaultTextStyle) {
|
||||||
|
font: historyPollQuestionFont;
|
||||||
|
linkFont: historyPollQuestionFont;
|
||||||
|
linkFontOver: historyPollQuestionFont;
|
||||||
|
}
|
||||||
|
historyPollAnswerStyle: TextStyle(defaultTextStyle) {
|
||||||
|
font: boxTextFont;
|
||||||
|
linkFont: boxTextFont;
|
||||||
|
linkFontOver: boxTextFont;
|
||||||
|
}
|
||||||
historyPollQuestionTop: 7px;
|
historyPollQuestionTop: 7px;
|
||||||
historyPollSubtitleSkip: 5px;
|
historyPollSubtitleSkip: 5px;
|
||||||
historyPollAnswerPadding: margins(30px, 16px, 0px, 16px);
|
historyPollAnswerPadding: margins(30px, 16px, 0px, 16px);
|
||||||
historyPollAnswersSkip: 5px;
|
historyPollAnswersSkip: 5px;
|
||||||
|
historyPollPercentFont: semiboldFont;
|
||||||
|
historyPollPercentSkip: 5px;
|
||||||
|
historyPollPercentTop: 3px;
|
||||||
|
historyPollFillingMin: 8px;
|
||||||
|
historyPollFillingHeight: 6px;
|
||||||
|
historyPollFillingRadius: 2px;
|
||||||
|
historyPollFillingBottom: 10px;
|
||||||
|
|
||||||
boxAttachEmoji: IconButton(historyAttachEmoji) {
|
boxAttachEmoji: IconButton(historyAttachEmoji) {
|
||||||
width: 30px;
|
width: 30px;
|
||||||
|
|
|
@ -21,6 +21,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "history/view/history_view_message.h"
|
#include "history/view/history_view_message.h"
|
||||||
#include "history/view/history_view_service_message.h"
|
#include "history/view/history_view_service_message.h"
|
||||||
#include "history/view/history_view_cursor_state.h"
|
#include "history/view/history_view_cursor_state.h"
|
||||||
|
#include "history/view/history_view_context_menu.h"
|
||||||
#include "ui/widgets/popup_menu.h"
|
#include "ui/widgets/popup_menu.h"
|
||||||
#include "ui/image/image.h"
|
#include "ui/image/image.h"
|
||||||
#include "ui/text_options.h"
|
#include "ui/text_options.h"
|
||||||
|
@ -42,6 +43,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "data/data_session.h"
|
#include "data/data_session.h"
|
||||||
#include "data/data_media_types.h"
|
#include "data/data_media_types.h"
|
||||||
#include "data/data_document.h"
|
#include "data/data_document.h"
|
||||||
|
#include "data/data_poll.h"
|
||||||
#include "data/data_photo.h"
|
#include "data/data_photo.h"
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
@ -1638,6 +1640,22 @@ void HistoryInner::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (const auto media = item->media()) {
|
||||||
|
if (const auto poll = media->poll()) {
|
||||||
|
if (!poll->closed) {
|
||||||
|
if (poll->voted()) {
|
||||||
|
_menu->addAction(lang(lng_polls_retract), [=] {
|
||||||
|
Auth().api().sendPollVotes(itemId, {});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (item->canStopPoll()) {
|
||||||
|
_menu->addAction(lang(lng_polls_stop), [=] {
|
||||||
|
HistoryView::StopPoll(itemId);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
if (msg && view && !link && (view->hasVisibleText() || mediaHasTextForCopy)) {
|
if (msg && view && !link && (view->hasVisibleText() || mediaHasTextForCopy)) {
|
||||||
_menu->addAction(lang(lng_context_copy_text), [=] {
|
_menu->addAction(lang(lng_context_copy_text), [=] {
|
||||||
copyContextText(itemId);
|
copyContextText(itemId);
|
||||||
|
|
|
@ -414,6 +414,28 @@ bool HistoryItem::allowsEdit(TimeId now) const {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool HistoryItem::canStopPoll() const {
|
||||||
|
if (id < 0
|
||||||
|
|| Has<HistoryMessageVia>()
|
||||||
|
|| Has<HistoryMessageForwarded>()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto peer = _history->peer;
|
||||||
|
if (peer->isSelf()) {
|
||||||
|
return true;
|
||||||
|
} else if (const auto channel = peer->asChannel()) {
|
||||||
|
if (isPost() && channel->canEditMessages()) {
|
||||||
|
return true;
|
||||||
|
} else if (out()) {
|
||||||
|
return isPost() ? channel->canPublish() : channel->canWrite();
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return out();
|
||||||
|
}
|
||||||
|
|
||||||
bool HistoryItem::canDelete() const {
|
bool HistoryItem::canDelete() const {
|
||||||
if (isLogEntry() || (!IsServerMsgId(id) && serviceMsg())) {
|
if (isLogEntry() || (!IsServerMsgId(id) && serviceMsg())) {
|
||||||
return false;
|
return false;
|
||||||
|
|
|
@ -211,6 +211,7 @@ public:
|
||||||
|
|
||||||
bool isPinned() const;
|
bool isPinned() const;
|
||||||
bool canPin() const;
|
bool canPin() const;
|
||||||
|
bool canStopPoll() const;
|
||||||
virtual bool allowsForward() const;
|
virtual bool allowsForward() const;
|
||||||
virtual bool allowsEdit(TimeId now) const;
|
virtual bool allowsEdit(TimeId now) const;
|
||||||
bool canDelete() const;
|
bool canDelete() const;
|
||||||
|
|
|
@ -588,41 +588,22 @@ bool HistoryMessage::allowsForward() const {
|
||||||
return !_media || _media->allowsForward();
|
return !_media || _media->allowsForward();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool HistoryMessage::allowsEdit(TimeId now) const {
|
bool HistoryMessage::isTooOldForEdit(TimeId now) const {
|
||||||
const auto peer = _history->peer;
|
const auto peer = _history->peer;
|
||||||
const auto messageToMyself = peer->isSelf();
|
if (peer->isSelf()) {
|
||||||
const auto canPinInMegagroup = [&] {
|
return false;
|
||||||
if (const auto megagroup = peer->asMegagroup()) {
|
} else if (const auto megagroup = peer->asMegagroup()) {
|
||||||
return megagroup->canPinMessages();
|
if (megagroup->canPinMessages()) {
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
return false;
|
|
||||||
}();
|
|
||||||
const auto messageTooOld = (messageToMyself || canPinInMegagroup)
|
|
||||||
? false
|
|
||||||
: (now - date() >= Global::EditTimeLimit());
|
|
||||||
if (id < 0 || messageTooOld) {
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
return (now - date() >= Global::EditTimeLimit());
|
||||||
|
}
|
||||||
|
|
||||||
if (Has<HistoryMessageVia>() || Has<HistoryMessageForwarded>()) {
|
bool HistoryMessage::allowsEdit(TimeId now) const {
|
||||||
return false;
|
return canStopPoll()
|
||||||
}
|
&& !isTooOldForEdit(now)
|
||||||
|
&& (!_media || _media->allowsEdit());
|
||||||
if (_media && !_media->allowsEdit()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (messageToMyself) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (const auto channel = _history->peer->asChannel()) {
|
|
||||||
if (isPost() && channel->canEditMessages()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (out()) {
|
|
||||||
return isPost() ? channel->canPublish() : channel->canWrite();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return out();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool HistoryMessage::uploading() const {
|
bool HistoryMessage::uploading() const {
|
||||||
|
|
|
@ -148,6 +148,7 @@ private:
|
||||||
bool hasAdminBadge() const {
|
bool hasAdminBadge() const {
|
||||||
return _flags & MTPDmessage_ClientFlag::f_has_admin_badge;
|
return _flags & MTPDmessage_ClientFlag::f_has_admin_badge;
|
||||||
}
|
}
|
||||||
|
bool isTooOldForEdit(TimeId now) const;
|
||||||
|
|
||||||
// For an invoice button we replace the button text with a "Receipt" key.
|
// For an invoice button we replace the button text with a "Receipt" key.
|
||||||
// It should show the receipt for the payed invoice. Still let mobile apps do that.
|
// It should show the receipt for the payed invoice. Still let mobile apps do that.
|
||||||
|
|
|
@ -8,7 +8,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "history/media/history_media_poll.h"
|
#include "history/media/history_media_poll.h"
|
||||||
|
|
||||||
#include "lang/lang_keys.h"
|
#include "lang/lang_keys.h"
|
||||||
#include "layout.h"
|
|
||||||
#include "history/history.h"
|
#include "history/history.h"
|
||||||
#include "history/history_item.h"
|
#include "history/history_item.h"
|
||||||
#include "history/view/history_view_element.h"
|
#include "history/view/history_view_element.h"
|
||||||
|
@ -17,6 +16,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "ui/text_options.h"
|
#include "ui/text_options.h"
|
||||||
#include "data/data_media_types.h"
|
#include "data/data_media_types.h"
|
||||||
#include "data/data_poll.h"
|
#include "data/data_poll.h"
|
||||||
|
#include "data/data_session.h"
|
||||||
|
#include "layout.h"
|
||||||
|
#include "auth_session.h"
|
||||||
|
#include "apiwrap.h"
|
||||||
#include "styles/style_history.h"
|
#include "styles/style_history.h"
|
||||||
#include "styles/style_widgets.h"
|
#include "styles/style_widgets.h"
|
||||||
|
|
||||||
|
@ -67,6 +70,7 @@ HistoryPoll::HistoryPoll(
|
||||||
: HistoryMedia(parent)
|
: HistoryMedia(parent)
|
||||||
, _poll(poll)
|
, _poll(poll)
|
||||||
, _question(st::msgMinWidth / 2) {
|
, _question(st::msgMinWidth / 2) {
|
||||||
|
Auth().data().registerPollView(_poll, _parent);
|
||||||
}
|
}
|
||||||
|
|
||||||
QSize HistoryPoll::countOptimalSize() {
|
QSize HistoryPoll::countOptimalSize() {
|
||||||
|
@ -98,11 +102,11 @@ QSize HistoryPoll::countOptimalSize() {
|
||||||
auto minHeight = st::historyPollQuestionTop
|
auto minHeight = st::historyPollQuestionTop
|
||||||
+ _question.minHeight()
|
+ _question.minHeight()
|
||||||
+ st::historyPollSubtitleSkip
|
+ st::historyPollSubtitleSkip
|
||||||
+ st::msgDateTextStyle.font->height
|
+ st::msgDateFont->height
|
||||||
+ st::historyPollAnswersSkip
|
+ st::historyPollAnswersSkip
|
||||||
+ answersHeight
|
+ answersHeight
|
||||||
+ st::msgPadding.bottom()
|
+ st::msgPadding.bottom()
|
||||||
+ st::msgDateTextStyle.font->height
|
+ st::msgDateFont->height
|
||||||
+ st::msgPadding.bottom();
|
+ st::msgPadding.bottom();
|
||||||
if (!isBubbleTop()) {
|
if (!isBubbleTop()) {
|
||||||
minHeight -= st::msgFileTopMinus;
|
minHeight -= st::msgFileTopMinus;
|
||||||
|
@ -110,6 +114,19 @@ QSize HistoryPoll::countOptimalSize() {
|
||||||
return { maxWidth, minHeight };
|
return { maxWidth, minHeight };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int HistoryPoll::countAnswerHeight(
|
||||||
|
const Answer &answer,
|
||||||
|
int innerWidth) const {
|
||||||
|
const auto answerWidth = innerWidth
|
||||||
|
- st::historyPollAnswerPadding.left()
|
||||||
|
- st::historyPollAnswerPadding.right();
|
||||||
|
return st::historyPollAnswerPadding.top()
|
||||||
|
+ 2 * st::defaultCheckbox.textPosition.y()
|
||||||
|
+ answer.text.countHeight(answerWidth)
|
||||||
|
+ st::historyPollAnswerPadding.bottom()
|
||||||
|
+ st::lineWidth;
|
||||||
|
}
|
||||||
|
|
||||||
QSize HistoryPoll::countCurrentSize(int newWidth) {
|
QSize HistoryPoll::countCurrentSize(int newWidth) {
|
||||||
const auto paddings = st::msgPadding.left() + st::msgPadding.right();
|
const auto paddings = st::msgPadding.left() + st::msgPadding.right();
|
||||||
|
|
||||||
|
@ -118,27 +135,20 @@ QSize HistoryPoll::countCurrentSize(int newWidth) {
|
||||||
- st::msgPadding.left()
|
- st::msgPadding.left()
|
||||||
- st::msgPadding.right();
|
- st::msgPadding.right();
|
||||||
|
|
||||||
const auto answerWidth = innerWidth
|
|
||||||
- st::historyPollAnswerPadding.left()
|
|
||||||
- st::historyPollAnswerPadding.right();
|
|
||||||
const auto answersHeight = ranges::accumulate(ranges::view::all(
|
const auto answersHeight = ranges::accumulate(ranges::view::all(
|
||||||
_answers
|
_answers
|
||||||
) | ranges::view::transform([&](const Answer &answer) {
|
) | ranges::view::transform([&](const Answer &answer) {
|
||||||
return st::historyPollAnswerPadding.top()
|
return countAnswerHeight(answer, innerWidth);
|
||||||
+ 2 * st::defaultCheckbox.textPosition.y()
|
|
||||||
+ answer.text.countHeight(answerWidth)
|
|
||||||
+ st::historyPollAnswerPadding.bottom()
|
|
||||||
+ st::lineWidth;
|
|
||||||
}), 0);
|
}), 0);
|
||||||
|
|
||||||
auto newHeight = st::historyPollQuestionTop
|
auto newHeight = st::historyPollQuestionTop
|
||||||
+ _question.countHeight(innerWidth)
|
+ _question.countHeight(innerWidth)
|
||||||
+ st::historyPollSubtitleSkip
|
+ st::historyPollSubtitleSkip
|
||||||
+ st::msgDateTextStyle.font->height
|
+ st::msgDateFont->height
|
||||||
+ st::historyPollAnswersSkip
|
+ st::historyPollAnswersSkip
|
||||||
+ answersHeight
|
+ answersHeight
|
||||||
+ st::msgPadding.bottom()
|
+ st::msgPadding.bottom()
|
||||||
+ st::msgDateTextStyle.font->height
|
+ st::msgDateFont->height
|
||||||
+ st::msgPadding.bottom();
|
+ st::msgPadding.bottom();
|
||||||
if (!isBubbleTop()) {
|
if (!isBubbleTop()) {
|
||||||
newHeight -= st::msgFileTopMinus;
|
newHeight -= st::msgFileTopMinus;
|
||||||
|
@ -171,15 +181,26 @@ void HistoryPoll::updateTexts() {
|
||||||
return result;
|
return result;
|
||||||
}) | ranges::to_vector;
|
}) | ranges::to_vector;
|
||||||
|
|
||||||
_voted = ranges::find(
|
for (auto &answer : _answers) {
|
||||||
_poll->answers,
|
answer.handler = createAnswerClickHandler(answer);
|
||||||
true,
|
}
|
||||||
[](const PollAnswer &answer) { return answer.chosen; }
|
|
||||||
) != end(_poll->answers);
|
_closed = _poll->closed;
|
||||||
|
}
|
||||||
|
|
||||||
|
ClickHandlerPtr HistoryPoll::createAnswerClickHandler(
|
||||||
|
const Answer &answer) const {
|
||||||
|
const auto option = answer.option;
|
||||||
|
const auto itemId = _parent->data()->fullId();
|
||||||
|
return std::make_shared<LambdaClickHandler>([=] {
|
||||||
|
Auth().api().sendPollVotes(itemId, { option });
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryPoll::updateVotes() const {
|
void HistoryPoll::updateVotes() const {
|
||||||
updateTotalVotes();
|
updateTotalVotes();
|
||||||
|
_voted = _poll->voted();
|
||||||
|
updateAnswerVotes();
|
||||||
}
|
}
|
||||||
|
|
||||||
void HistoryPoll::updateTotalVotes() const {
|
void HistoryPoll::updateTotalVotes() const {
|
||||||
|
@ -199,19 +220,55 @@ void HistoryPoll::updateTotalVotes() const {
|
||||||
_totalVotesLabel.setText(st::msgDateTextStyle, string);
|
_totalVotesLabel.setText(st::msgDateTextStyle, string);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void HistoryPoll::updateAnswerVotesFromOriginal(
|
||||||
|
const Answer &answer,
|
||||||
|
const PollAnswer &original,
|
||||||
|
int totalVotes,
|
||||||
|
int maxVotes) const {
|
||||||
|
if (!_voted && !_closed) {
|
||||||
|
answer.votesPercent.clear();
|
||||||
|
} else if (answer.votes != original.votes
|
||||||
|
|| answer.votesPercent.isEmpty()) {
|
||||||
|
const auto percent = original.votes * 100 / totalVotes;
|
||||||
|
answer.votesPercent = QString::number(percent) + '%';
|
||||||
|
answer.votesPercentWidth = st::historyPollPercentFont->width(
|
||||||
|
answer.votesPercent);
|
||||||
|
}
|
||||||
|
answer.votes = original.votes;
|
||||||
|
answer.filling = answer.votes / float64(maxVotes);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryPoll::updateAnswerVotes() const {
|
||||||
|
if (_poll->answers.size() != _answers.size()
|
||||||
|
|| _poll->answers.empty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const auto totalVotes = std::max(1, _totalVotes);
|
||||||
|
const auto maxVotes = std::max(1, ranges::max_element(
|
||||||
|
_poll->answers,
|
||||||
|
ranges::less(),
|
||||||
|
&PollAnswer::votes)->votes);
|
||||||
|
auto &&answers = ranges::view::zip(_answers, _poll->answers);
|
||||||
|
for (auto &&[answer, original] : answers) {
|
||||||
|
updateAnswerVotesFromOriginal(
|
||||||
|
answer,
|
||||||
|
original,
|
||||||
|
totalVotes,
|
||||||
|
maxVotes);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void HistoryPoll::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const {
|
void HistoryPoll::draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const {
|
||||||
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();
|
updateVotes();
|
||||||
|
|
||||||
auto outbg = _parent->hasOutLayout();
|
const auto outbg = _parent->hasOutLayout();
|
||||||
bool selected = (selection == FullSelection);
|
const auto selected = (selection == FullSelection);
|
||||||
|
const auto ®ular = selected ? (outbg ? st::msgOutDateFgSelected : st::msgInDateFgSelected) : (outbg ? st::msgOutDateFg : st::msgInDateFg);
|
||||||
|
|
||||||
auto &semibold = selected ? (outbg ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (outbg ? st::msgOutServiceFg : st::msgInServiceFg);
|
const auto padding = st::msgPadding;
|
||||||
auto ®ular = selected ? (outbg ? st::msgOutDateFgSelected : st::msgInDateFgSelected) : (outbg ? st::msgOutDateFg : st::msgInDateFg);
|
|
||||||
|
|
||||||
auto padding = st::msgPadding;
|
|
||||||
auto tshift = st::historyPollQuestionTop;
|
auto tshift = st::historyPollQuestionTop;
|
||||||
if (!isBubbleTop()) {
|
if (!isBubbleTop()) {
|
||||||
tshift -= st::msgFileTopMinus;
|
tshift -= st::msgFileTopMinus;
|
||||||
|
@ -224,34 +281,19 @@ void HistoryPoll::draw(Painter &p, const QRect &r, TextSelection selection, Time
|
||||||
|
|
||||||
p.setPen(regular);
|
p.setPen(regular);
|
||||||
_subtitle.drawLeftElided(p, padding.left(), tshift, paintw, width());
|
_subtitle.drawLeftElided(p, padding.left(), tshift, paintw, width());
|
||||||
tshift += st::msgDateTextStyle.font->height + st::historyPollAnswersSkip;
|
tshift += st::msgDateFont->height + st::historyPollAnswersSkip;
|
||||||
|
|
||||||
const auto aleft = padding.left() + st::historyPollAnswerPadding.left();
|
|
||||||
const auto awidth = paintw - st::historyPollAnswerPadding.left() - st::historyPollAnswerPadding.right();
|
|
||||||
for (const auto &answer : _answers) {
|
for (const auto &answer : _answers) {
|
||||||
tshift += st::historyPollAnswerPadding.top();
|
const auto height = paintAnswer(
|
||||||
|
p,
|
||||||
{
|
answer,
|
||||||
PainterHighQualityEnabler hq(p);
|
padding.left(),
|
||||||
const auto &st = st::defaultRadio;
|
tshift,
|
||||||
auto pen = regular->p;
|
paintw,
|
||||||
pen.setWidth(st.thickness);
|
width(),
|
||||||
p.setPen(pen);
|
selection,
|
||||||
p.setBrush(Qt::NoBrush);
|
ms);
|
||||||
p.drawEllipse(QRectF(padding.left(), tshift, st.diameter, st.diameter).marginsRemoved(QMarginsF(st.thickness / 2., st.thickness / 2., st.thickness / 2., st.thickness / 2.)));
|
tshift += height;
|
||||||
}
|
|
||||||
|
|
||||||
tshift += st::defaultCheckbox.textPosition.y();
|
|
||||||
p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg);
|
|
||||||
answer.text.drawLeft(p, aleft, tshift, awidth, width());
|
|
||||||
tshift += answer.text.countHeight(awidth)
|
|
||||||
+ st::defaultCheckbox.textPosition.y()
|
|
||||||
+ st::historyPollAnswerPadding.bottom();
|
|
||||||
|
|
||||||
p.setOpacity(0.5);
|
|
||||||
p.fillRect(aleft, tshift, width() - aleft, st::lineWidth, regular);
|
|
||||||
tshift += st::lineWidth;
|
|
||||||
p.setOpacity(1.);
|
|
||||||
}
|
}
|
||||||
if (!_totalVotesLabel.isEmpty()) {
|
if (!_totalVotesLabel.isEmpty()) {
|
||||||
tshift += st::msgPadding.bottom();
|
tshift += st::msgPadding.bottom();
|
||||||
|
@ -260,11 +302,101 @@ void HistoryPoll::draw(Painter &p, const QRect &r, TextSelection selection, Time
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
TextState HistoryPoll::textState(QPoint point, StateRequest request) const {
|
int HistoryPoll::paintAnswer(
|
||||||
auto result = TextState(_parent);
|
Painter &p,
|
||||||
if (QRect(0, 0, width(), height()).contains(point)) {
|
const Answer &answer,
|
||||||
// result.link = _link;
|
int left,
|
||||||
return result;
|
int top,
|
||||||
|
int width,
|
||||||
|
int outerWidth,
|
||||||
|
TextSelection selection,
|
||||||
|
TimeMs ms) const {
|
||||||
|
const auto result = countAnswerHeight(answer, width);
|
||||||
|
const auto bottom = top + result;
|
||||||
|
const auto outbg = _parent->hasOutLayout();
|
||||||
|
const auto selected = (selection == FullSelection);
|
||||||
|
const auto ®ular = selected ? (outbg ? st::msgOutDateFgSelected : st::msgInDateFgSelected) : (outbg ? st::msgOutDateFg : st::msgInDateFg);
|
||||||
|
|
||||||
|
const auto aleft = left + st::historyPollAnswerPadding.left();
|
||||||
|
const auto awidth = width
|
||||||
|
- st::historyPollAnswerPadding.left()
|
||||||
|
- st::historyPollAnswerPadding.right();
|
||||||
|
|
||||||
|
top += st::historyPollAnswerPadding.top();
|
||||||
|
|
||||||
|
if (_voted || _closed) {
|
||||||
|
p.setFont(st::historyPollPercentFont);
|
||||||
|
p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg);
|
||||||
|
const auto left = aleft
|
||||||
|
- answer.votesPercentWidth
|
||||||
|
- st::historyPollPercentSkip;
|
||||||
|
p.drawTextLeft(left, top + st::historyPollPercentTop, outerWidth, answer.votesPercent, answer.votesPercentWidth);
|
||||||
|
} else {
|
||||||
|
PainterHighQualityEnabler hq(p);
|
||||||
|
const auto &st = st::defaultRadio;
|
||||||
|
auto pen = regular->p;
|
||||||
|
pen.setWidth(st.thickness);
|
||||||
|
p.setPen(pen);
|
||||||
|
p.setBrush(Qt::NoBrush);
|
||||||
|
p.drawEllipse(QRectF(left, top, st.diameter, st.diameter).marginsRemoved(QMarginsF(st.thickness / 2., st.thickness / 2., st.thickness / 2., st.thickness / 2.)));
|
||||||
|
}
|
||||||
|
|
||||||
|
top += st::defaultCheckbox.textPosition.y();
|
||||||
|
p.setPen(outbg ? st::webPageDescriptionOutFg : st::webPageDescriptionInFg);
|
||||||
|
answer.text.drawLeft(p, aleft, top, awidth, outerWidth);
|
||||||
|
|
||||||
|
if (_voted || _closed) {
|
||||||
|
auto &semibold = selected ? (outbg ? st::msgOutServiceFgSelected : st::msgInServiceFgSelected) : (outbg ? st::msgOutServiceFg : st::msgInServiceFg);
|
||||||
|
PainterHighQualityEnabler hq(p);
|
||||||
|
p.setPen(Qt::NoPen);
|
||||||
|
p.setBrush(semibold);
|
||||||
|
const auto size = anim::interpolate(st::historyPollFillingMin, awidth, answer.filling);
|
||||||
|
const auto radius = st::historyPollFillingRadius;
|
||||||
|
const auto top = bottom - st::historyPollFillingBottom - st::historyPollFillingHeight;
|
||||||
|
p.drawRoundedRect(aleft, top, size, st::historyPollFillingHeight, radius, radius);
|
||||||
|
} else {
|
||||||
|
p.setOpacity(0.5);
|
||||||
|
p.fillRect(
|
||||||
|
aleft,
|
||||||
|
bottom - st::lineWidth,
|
||||||
|
outerWidth - aleft,
|
||||||
|
st::lineWidth,
|
||||||
|
regular);
|
||||||
|
p.setOpacity(1.);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TextState HistoryPoll::textState(QPoint point, StateRequest request) const {
|
||||||
|
auto result = TextState(_parent);
|
||||||
|
if (_voted || _closed) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto padding = st::msgPadding;
|
||||||
|
auto paintw = width();
|
||||||
|
auto tshift = st::historyPollQuestionTop;
|
||||||
|
if (!isBubbleTop()) {
|
||||||
|
tshift -= st::msgFileTopMinus;
|
||||||
|
}
|
||||||
|
paintw -= padding.left() + padding.right();
|
||||||
|
|
||||||
|
tshift += _question.countHeight(paintw) + st::historyPollSubtitleSkip;
|
||||||
|
tshift += st::msgDateFont->height + st::historyPollAnswersSkip;
|
||||||
|
const auto awidth = paintw
|
||||||
|
- st::historyPollAnswerPadding.left()
|
||||||
|
- st::historyPollAnswerPadding.right();
|
||||||
|
for (const auto &answer : _answers) {
|
||||||
|
const auto height = countAnswerHeight(answer, paintw);
|
||||||
|
if (point.y() >= tshift && point.y() < tshift + height) {
|
||||||
|
result.link = answer.handler;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
tshift += height;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
HistoryPoll::~HistoryPoll() {
|
||||||
|
Auth().data().unregisterPollView(_poll, _parent);
|
||||||
|
}
|
||||||
|
|
|
@ -9,6 +9,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
|
|
||||||
#include "history/media/history_media.h"
|
#include "history/media/history_media.h"
|
||||||
|
|
||||||
|
struct PollAnswer;
|
||||||
|
|
||||||
class HistoryPoll : public HistoryMedia {
|
class HistoryPoll : public HistoryMedia {
|
||||||
public:
|
public:
|
||||||
HistoryPoll(
|
HistoryPoll(
|
||||||
|
@ -32,6 +34,8 @@ public:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
~HistoryPoll();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
struct Answer {
|
struct Answer {
|
||||||
Answer();
|
Answer();
|
||||||
|
@ -39,19 +43,43 @@ private:
|
||||||
Text text;
|
Text text;
|
||||||
QByteArray option;
|
QByteArray option;
|
||||||
mutable int votes = 0;
|
mutable int votes = 0;
|
||||||
|
mutable int votesPercentWidth = 0;
|
||||||
|
mutable float64 filling = 0.;
|
||||||
|
mutable QString votesPercent;
|
||||||
mutable bool chosen = false;
|
mutable bool chosen = false;
|
||||||
|
ClickHandlerPtr handler;
|
||||||
};
|
};
|
||||||
QSize countOptimalSize() override;
|
QSize countOptimalSize() override;
|
||||||
QSize countCurrentSize(int newWidth) override;
|
QSize countCurrentSize(int newWidth) override;
|
||||||
|
|
||||||
|
int countAnswerHeight(const Answer &answer, int innerWidth) const;
|
||||||
|
[[nodiscard]] ClickHandlerPtr createAnswerClickHandler(
|
||||||
|
const Answer &answer) const;
|
||||||
void updateTexts();
|
void updateTexts();
|
||||||
void updateVotes() const;
|
void updateVotes() const;
|
||||||
void updateTotalVotes() const;
|
void updateTotalVotes() const;
|
||||||
|
void updateAnswerVotes() const;
|
||||||
|
void updateAnswerVotesFromOriginal(
|
||||||
|
const Answer &answer,
|
||||||
|
const PollAnswer &original,
|
||||||
|
int totalVotes,
|
||||||
|
int maxVotes) const;
|
||||||
|
|
||||||
|
int paintAnswer(
|
||||||
|
Painter &p,
|
||||||
|
const Answer &answer,
|
||||||
|
int left,
|
||||||
|
int top,
|
||||||
|
int width,
|
||||||
|
int outerWidth,
|
||||||
|
TextSelection selection,
|
||||||
|
TimeMs ms) const;
|
||||||
|
|
||||||
not_null<PollData*> _poll;
|
not_null<PollData*> _poll;
|
||||||
int _pollVersion = 0;
|
int _pollVersion = 0;
|
||||||
mutable int _totalVotes = 0;
|
mutable int _totalVotes = 0;
|
||||||
mutable bool _voted = false;
|
mutable bool _voted = false;
|
||||||
|
bool _closed = false;
|
||||||
|
|
||||||
Text _question;
|
Text _question;
|
||||||
Text _subtitle;
|
Text _subtitle;
|
||||||
|
|
|
@ -505,4 +505,12 @@ base::unique_qptr<Ui::PopupMenu> FillContextMenu(
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void StopPoll(FullMsgId itemId) {
|
||||||
|
Ui::show(Box<ConfirmBox>(
|
||||||
|
lang(lng_polls_stop_warning),
|
||||||
|
lang(lng_polls_stop_sure),
|
||||||
|
lang(lng_cancel),
|
||||||
|
[=] { Ui::hideLayer(); Auth().api().closePoll(itemId); }));
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace HistoryView
|
} // namespace HistoryView
|
||||||
|
|
|
@ -35,4 +35,6 @@ base::unique_qptr<Ui::PopupMenu> FillContextMenu(
|
||||||
not_null<ListWidget*> list,
|
not_null<ListWidget*> list,
|
||||||
const ContextMenuRequest &request);
|
const ContextMenuRequest &request);
|
||||||
|
|
||||||
|
void StopPoll(FullMsgId itemId);
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
Loading…
Reference in New Issue