321 lines
		
	
	
	
		
			8.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			321 lines
		
	
	
	
		
			8.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop application for the Telegram messaging service.
 | |
| 
 | |
| For license and copyright information please follow this link:
 | |
| https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | |
| */
 | |
| #include "data/data_poll.h"
 | |
| 
 | |
| #include "apiwrap.h"
 | |
| #include "data/data_user.h"
 | |
| #include "data/data_session.h"
 | |
| #include "base/call_delayed.h"
 | |
| #include "main/main_session.h"
 | |
| #include "api/api_text_entities.h"
 | |
| #include "ui/text/text_options.h"
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| constexpr auto kShortPollTimeout = 30 * crl::time(1000);
 | |
| constexpr auto kReloadAfterAutoCloseDelay = crl::time(1000);
 | |
| 
 | |
| const PollAnswer *AnswerByOption(
 | |
| 		const std::vector<PollAnswer> &list,
 | |
| 		const QByteArray &option) {
 | |
| 	const auto i = ranges::find(
 | |
| 		list,
 | |
| 		option,
 | |
| 		[](const PollAnswer &a) { return a.option; });
 | |
| 	return (i != end(list)) ? &*i : nullptr;
 | |
| }
 | |
| 
 | |
| PollAnswer *AnswerByOption(
 | |
| 		std::vector<PollAnswer> &list,
 | |
| 		const QByteArray &option) {
 | |
| 	return const_cast<PollAnswer*>(AnswerByOption(
 | |
| 		std::as_const(list),
 | |
| 		option));
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| PollData::PollData(not_null<Data::Session*> owner, PollId id)
 | |
| : id(id)
 | |
| , _owner(owner) {
 | |
| }
 | |
| 
 | |
| Data::Session &PollData::owner() const {
 | |
| 	return *_owner;
 | |
| }
 | |
| 
 | |
| Main::Session &PollData::session() const {
 | |
| 	return _owner->session();
 | |
| }
 | |
| 
 | |
| bool PollData::closeByTimer() {
 | |
| 	if (closed()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	_flags |= Flag::Closed;
 | |
| 	++version;
 | |
| 	base::call_delayed(kReloadAfterAutoCloseDelay, &_owner->session(), [=] {
 | |
| 		_lastResultsUpdate = -1; // Force reload results.
 | |
| 		++version;
 | |
| 		_owner->notifyPollUpdateDelayed(this);
 | |
| 	});
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool PollData::applyChanges(const MTPDpoll &poll) {
 | |
| 	Expects(poll.vid().v == id);
 | |
| 
 | |
| 	const auto newQuestion = qs(poll.vquestion());
 | |
| 	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));
 | |
| 	const auto newCloseDate = poll.vclose_date().value_or_empty();
 | |
| 	const auto newClosePeriod = poll.vclose_period().value_or_empty();
 | |
| 	auto newAnswers = ranges::views::all(
 | |
| 		poll.vanswers().v
 | |
| 	) | ranges::views::transform([](const MTPPollAnswer &data) {
 | |
| 		return data.match([](const MTPDpollAnswer &answer) {
 | |
| 			auto result = PollAnswer();
 | |
| 			result.option = answer.voption().v;
 | |
| 			result.text = qs(answer.vtext());
 | |
| 			return result;
 | |
| 		});
 | |
| 	}) | ranges::views::take(
 | |
| 		kMaxOptions
 | |
| 	) | ranges::to_vector;
 | |
| 
 | |
| 	const auto changed1 = (question != newQuestion)
 | |
| 		|| (closeDate != newCloseDate)
 | |
| 		|| (closePeriod != newClosePeriod)
 | |
| 		|| (_flags != newFlags);
 | |
| 	const auto changed2 = (answers != newAnswers);
 | |
| 	if (!changed1 && !changed2) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (changed1) {
 | |
| 		question = newQuestion;
 | |
| 		closeDate = newCloseDate;
 | |
| 		closePeriod = newClosePeriod;
 | |
| 		_flags = newFlags;
 | |
| 	}
 | |
| 	if (changed2) {
 | |
| 		std::swap(answers, newAnswers);
 | |
| 		for (const auto &old : newAnswers) {
 | |
| 			if (const auto current = answerByOption(old.option)) {
 | |
| 				current->votes = old.votes;
 | |
| 				current->chosen = old.chosen;
 | |
| 				current->correct = old.correct;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	++version;
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool PollData::applyResults(const MTPPollResults &results) {
 | |
| 	return results.match([&](const MTPDpollResults &results) {
 | |
| 		_lastResultsUpdate = crl::now();
 | |
| 
 | |
| 		const auto newTotalVoters =
 | |
| 			results.vtotal_voters().value_or(totalVoters);
 | |
| 		auto changed = (newTotalVoters != totalVoters);
 | |
| 		if (const auto list = results.vresults()) {
 | |
| 			for (const auto &result : list->v) {
 | |
| 				if (applyResultToAnswers(result, results.is_min())) {
 | |
| 					changed = true;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		if (const auto recent = results.vrecent_voters()) {
 | |
| 			const auto bareProj = [](not_null<UserData*> user) {
 | |
| 				return peerToUser(user->id).bare;
 | |
| 			};
 | |
| 			const auto recentChanged = !ranges::equal(
 | |
| 				recentVoters,
 | |
| 				recent->v,
 | |
| 				ranges::equal_to(),
 | |
| 				bareProj,
 | |
| 				&MTPlong::v);
 | |
| 			if (recentChanged) {
 | |
| 				changed = true;
 | |
| 				recentVoters = ranges::views::all(
 | |
| 					recent->v
 | |
| 				) | ranges::views::transform([&](MTPlong userId) {
 | |
| 					const auto user = _owner->user(userId.v);
 | |
| 					return user->isMinimalLoaded() ? user.get() : nullptr;
 | |
| 				}) | ranges::views::filter([](UserData *user) {
 | |
| 					return user != nullptr;
 | |
| 				}) | ranges::views::transform([](UserData *user) {
 | |
| 					return not_null<UserData*>(user);
 | |
| 				}) | ranges::to_vector;
 | |
| 			}
 | |
| 		}
 | |
| 		if (results.vsolution()) {
 | |
| 			auto newSolution = TextWithEntities{
 | |
| 				results.vsolution().value_or_empty(),
 | |
| 				Api::EntitiesFromMTP(
 | |
| 					&_owner->session(),
 | |
| 					results.vsolution_entities().value_or_empty())
 | |
| 			};
 | |
| 			if (solution != newSolution) {
 | |
| 				solution = std::move(newSolution);
 | |
| 				changed = true;
 | |
| 			}
 | |
| 		}
 | |
| 		if (!changed) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		totalVoters = newTotalVoters;
 | |
| 		++version;
 | |
| 		return changed;
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void PollData::checkResultsReload(
 | |
| 		not_null<HistoryItem*> item,
 | |
| 		crl::time now) {
 | |
| 	if (_lastResultsUpdate > 0
 | |
| 		&& _lastResultsUpdate + kShortPollTimeout > now) {
 | |
| 		return;
 | |
| 	} else if (closed() && _lastResultsUpdate >= 0) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_lastResultsUpdate = now;
 | |
| 	_owner->session().api().reloadPollResults(item);
 | |
| }
 | |
| 
 | |
| PollAnswer *PollData::answerByOption(const QByteArray &option) {
 | |
| 	return AnswerByOption(answers, option);
 | |
| }
 | |
| 
 | |
| const PollAnswer *PollData::answerByOption(const QByteArray &option) const {
 | |
| 	return AnswerByOption(answers, option);
 | |
| }
 | |
| 
 | |
| bool PollData::applyResultToAnswers(
 | |
| 		const MTPPollAnswerVoters &result,
 | |
| 		bool isMinResults) {
 | |
| 	return result.match([&](const MTPDpollAnswerVoters &voters) {
 | |
| 		const auto &option = voters.voption().v;
 | |
| 		const auto answer = answerByOption(option);
 | |
| 		if (!answer) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		auto changed = (answer->votes != voters.vvoters().v);
 | |
| 		if (changed) {
 | |
| 			answer->votes = voters.vvoters().v;
 | |
| 		}
 | |
| 		if (!isMinResults) {
 | |
| 			if (answer->chosen != voters.is_chosen()) {
 | |
| 				answer->chosen = voters.is_chosen();
 | |
| 				changed = true;
 | |
| 			}
 | |
| 		}
 | |
| 		if (voters.is_correct() && !answer->correct) {
 | |
| 			answer->correct = voters.is_correct();
 | |
| 			changed = true;
 | |
| 		}
 | |
| 		return changed;
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void PollData::setFlags(Flags flags) {
 | |
| 	if (_flags != flags) {
 | |
| 		_flags = flags;
 | |
| 		++version;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| PollData::Flags PollData::flags() const {
 | |
| 	return _flags;
 | |
| }
 | |
| 
 | |
| bool PollData::voted() const {
 | |
| 	return ranges::contains(answers, true, &PollAnswer::chosen);
 | |
| }
 | |
| 
 | |
| 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, bool close) {
 | |
| 	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);
 | |
| 	using Flag = MTPDpoll::Flag;
 | |
| 	const auto flags = ((poll->closed() || close) ? Flag::f_closed : Flag(0))
 | |
| 		| (poll->multiChoice() ? Flag::f_multiple_choice : Flag(0))
 | |
| 		| (poll->publicVotes() ? Flag::f_public_voters : Flag(0))
 | |
| 		| (poll->quiz() ? Flag::f_quiz : Flag(0))
 | |
| 		| (poll->closePeriod > 0 ? Flag::f_close_period : Flag(0))
 | |
| 		| (poll->closeDate > 0 ? Flag::f_close_date : Flag(0));
 | |
| 	return MTP_poll(
 | |
| 		MTP_long(poll->id),
 | |
| 		MTP_flags(flags),
 | |
| 		MTP_string(poll->question),
 | |
| 		MTP_vector<MTPPollAnswer>(answers),
 | |
| 		MTP_int(poll->closePeriod),
 | |
| 		MTP_int(poll->closeDate));
 | |
| }
 | |
| 
 | |
| MTPInputMedia PollDataToInputMedia(
 | |
| 		not_null<const PollData*> poll,
 | |
| 		bool close) {
 | |
| 	auto inputFlags = MTPDinputMediaPoll::Flag(0)
 | |
| 		| (poll->quiz()
 | |
| 			? MTPDinputMediaPoll::Flag::f_correct_answers
 | |
| 			: MTPDinputMediaPoll::Flag(0));
 | |
| 	auto correct = QVector<MTPbytes>();
 | |
| 	for (const auto &answer : poll->answers) {
 | |
| 		if (answer.correct) {
 | |
| 			correct.push_back(MTP_bytes(answer.option));
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	auto solution = poll->solution;
 | |
| 	const auto prepareFlags = Ui::ItemTextDefaultOptions().flags;
 | |
| 	TextUtilities::PrepareForSending(solution, prepareFlags);
 | |
| 	TextUtilities::Trim(solution);
 | |
| 	const auto sentEntities = Api::EntitiesToMTP(
 | |
| 		&poll->session(),
 | |
| 		solution.entities,
 | |
| 		Api::ConvertOption::SkipLocal);
 | |
| 	if (!solution.text.isEmpty()) {
 | |
| 		inputFlags |= MTPDinputMediaPoll::Flag::f_solution;
 | |
| 	}
 | |
| 	if (!sentEntities.v.isEmpty()) {
 | |
| 		inputFlags |= MTPDinputMediaPoll::Flag::f_solution_entities;
 | |
| 	}
 | |
| 	return MTP_inputMediaPoll(
 | |
| 		MTP_flags(inputFlags),
 | |
| 		PollDataToMTP(poll, close),
 | |
| 		MTP_vector<MTPbytes>(correct),
 | |
| 		MTP_string(solution.text),
 | |
| 		sentEntities);
 | |
| }
 | 
