626 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			626 lines
		
	
	
	
		
			17 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 "support/support_helper.h"
 | |
| 
 | |
| #include "dialogs/dialogs_key.h"
 | |
| #include "data/data_drafts.h"
 | |
| #include "data/data_forum.h"
 | |
| #include "data/data_forum_topic.h"
 | |
| #include "data/data_user.h"
 | |
| #include "data/data_session.h"
 | |
| #include "data/data_changes.h"
 | |
| #include "api/api_text_entities.h"
 | |
| #include "history/history.h"
 | |
| #include "boxes/abstract_box.h"
 | |
| #include "ui/toast/toast.h"
 | |
| #include "ui/widgets/input_fields.h"
 | |
| #include "ui/chat/attach/attach_prepare.h"
 | |
| #include "ui/text/format_values.h"
 | |
| #include "ui/text/text_entity.h"
 | |
| #include "ui/text/text_options.h"
 | |
| #include "chat_helpers/message_field.h"
 | |
| #include "chat_helpers/emoji_suggestions_widget.h"
 | |
| #include "base/unixtime.h"
 | |
| #include "lang/lang_keys.h"
 | |
| #include "window/window_session_controller.h"
 | |
| #include "storage/storage_media_prepare.h"
 | |
| #include "storage/localimageloader.h"
 | |
| #include "core/launcher.h"
 | |
| #include "core/application.h"
 | |
| #include "core/core_settings.h"
 | |
| #include "main/main_session.h"
 | |
| #include "apiwrap.h"
 | |
| #include "styles/style_layers.h"
 | |
| #include "styles/style_boxes.h"
 | |
| 
 | |
| namespace Main {
 | |
| class Session;
 | |
| } // namespace Main
 | |
| 
 | |
| namespace Support {
 | |
| namespace {
 | |
| 
 | |
| constexpr auto kOccupyFor = TimeId(60);
 | |
| constexpr auto kReoccupyEach = 30 * crl::time(1000);
 | |
| constexpr auto kMaxSupportInfoLength = MaxMessageSize * 4;
 | |
| constexpr auto kTopicRootId = MsgId(0);
 | |
| 
 | |
| class EditInfoBox : public Ui::BoxContent {
 | |
| public:
 | |
| 	EditInfoBox(
 | |
| 		QWidget*,
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		const TextWithTags &text,
 | |
| 		Fn<void(TextWithTags, Fn<void(bool success)>)> submit);
 | |
| 
 | |
| protected:
 | |
| 	void prepare() override;
 | |
| 	void setInnerFocus() override;
 | |
| 
 | |
| private:
 | |
| 	const not_null<Window::SessionController*> _controller;
 | |
| 	object_ptr<Ui::InputField> _field = { nullptr };
 | |
| 	Fn<void(TextWithTags, Fn<void(bool success)>)> _submit;
 | |
| 
 | |
| };
 | |
| 
 | |
| EditInfoBox::EditInfoBox(
 | |
| 	QWidget*,
 | |
| 	not_null<Window::SessionController*> controller,
 | |
| 	const TextWithTags &text,
 | |
| 	Fn<void(TextWithTags, Fn<void(bool success)>)> submit)
 | |
| : _controller(controller)
 | |
| , _field(
 | |
| 	this,
 | |
| 	st::supportInfoField,
 | |
| 	Ui::InputField::Mode::MultiLine,
 | |
| 	rpl::single(u"Support information"_q), // #TODO hard_lang
 | |
| 	text)
 | |
| , _submit(std::move(submit)) {
 | |
| 	_field->setMaxLength(kMaxSupportInfoLength);
 | |
| 	_field->setSubmitSettings(
 | |
| 		Core::App().settings().sendSubmitWay());
 | |
| 	_field->setInstantReplaces(Ui::InstantReplaces::Default());
 | |
| 	_field->setInstantReplacesEnabled(
 | |
| 		Core::App().settings().replaceEmojiValue());
 | |
| 	_field->setMarkdownReplacesEnabled(rpl::single(true));
 | |
| 	_field->setEditLinkCallback(
 | |
| 		DefaultEditLinkCallback(controller->uiShow(), _field));
 | |
| }
 | |
| 
 | |
| void EditInfoBox::prepare() {
 | |
| 	setTitle(rpl::single(u"Edit support information"_q)); // #TODO hard_lang
 | |
| 
 | |
| 	const auto save = [=] {
 | |
| 		const auto done = crl::guard(this, [=](bool success) {
 | |
| 			if (success) {
 | |
| 				closeBox();
 | |
| 			} else {
 | |
| 				_field->showError();
 | |
| 			}
 | |
| 		});
 | |
| 		_submit(_field->getTextWithAppliedMarkdown(), done);
 | |
| 	};
 | |
| 	addButton(tr::lng_settings_save(), save);
 | |
| 	addButton(tr::lng_cancel(), [=] { closeBox(); });
 | |
| 
 | |
| 	connect(_field, &Ui::InputField::submitted, save);
 | |
| 	connect(_field, &Ui::InputField::cancelled, [=] { closeBox(); });
 | |
| 	Ui::Emoji::SuggestionsController::Init(
 | |
| 		getDelegate()->outerContainer(),
 | |
| 		_field,
 | |
| 		&_controller->session());
 | |
| 
 | |
| 	auto cursor = _field->textCursor();
 | |
| 	cursor.movePosition(QTextCursor::End);
 | |
| 	_field->setTextCursor(cursor);
 | |
| 
 | |
| 	widthValue(
 | |
| 	) | rpl::start_with_next([=](int width) {
 | |
| 		_field->resizeToWidth(
 | |
| 			width - st::boxPadding.left() - st::boxPadding.right());
 | |
| 		_field->moveToLeft(st::boxPadding.left(), st::boxPadding.bottom());
 | |
| 	}, _field->lifetime());
 | |
| 
 | |
| 	_field->heightValue(
 | |
| 	) | rpl::start_with_next([=](int height) {
 | |
| 		setDimensions(
 | |
| 			st::boxWideWidth,
 | |
| 			st::boxPadding.bottom() + height + st::boxPadding.bottom());
 | |
| 	}, _field->lifetime());
 | |
| }
 | |
| 
 | |
| void EditInfoBox::setInnerFocus() {
 | |
| 	_field->setFocusFast();
 | |
| }
 | |
| 
 | |
| uint32 OccupationTag() {
 | |
| 	return uint32(Core::Launcher::Instance().installationTag() & 0xFFFFFFFF);
 | |
| }
 | |
| 
 | |
| QString NormalizeName(QString name) {
 | |
| 	return name.replace(':', '_').replace(';', '_');
 | |
| }
 | |
| 
 | |
| Data::Draft OccupiedDraft(const QString &normalizedName) {
 | |
| 	const auto now = base::unixtime::now(), till = now + kOccupyFor;
 | |
| 	return {
 | |
| 		TextWithTags{ "t:"
 | |
| 			+ QString::number(till)
 | |
| 			+ ";u:"
 | |
| 			+ QString::number(OccupationTag())
 | |
| 			+ ";n:"
 | |
| 			+ normalizedName },
 | |
| 		MsgId(0), // replyTo
 | |
| 		kTopicRootId,
 | |
| 		MessageCursor(),
 | |
| 		Data::PreviewState::Allowed
 | |
| 	};
 | |
| }
 | |
| 
 | |
| [[nodiscard]] bool TrackHistoryOccupation(History *history) {
 | |
| 	if (!history) {
 | |
| 		return false;
 | |
| 	} else if (const auto user = history->peer->asUser()) {
 | |
| 		return !user->isBot();
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| uint32 ParseOccupationTag(History *history) {
 | |
| 	if (!TrackHistoryOccupation(history)) {
 | |
| 		return 0;
 | |
| 	}
 | |
| 	const auto draft = history->cloudDraft(kTopicRootId);
 | |
| 	if (!draft) {
 | |
| 		return 0;
 | |
| 	}
 | |
| 	const auto &text = draft->textWithTags.text;
 | |
| 	const auto parts = QStringView(text).split(';');
 | |
| 	auto valid = false;
 | |
| 	auto result = uint32();
 | |
| 	for (const auto &part : parts) {
 | |
| 		if (part.startsWith(u"t:"_q)) {
 | |
| 			if (base::StringViewMid(part, 2).toInt() >= base::unixtime::now()) {
 | |
| 				valid = true;
 | |
| 			} else {
 | |
| 				return 0;
 | |
| 			}
 | |
| 		} else if (part.startsWith(u"u:"_q)) {
 | |
| 			result = base::StringViewMid(part, 2).toUInt();
 | |
| 		}
 | |
| 	}
 | |
| 	return valid ? result : 0;
 | |
| }
 | |
| 
 | |
| QString ParseOccupationName(History *history) {
 | |
| 	if (!TrackHistoryOccupation(history)) {
 | |
| 		return QString();
 | |
| 	}
 | |
| 	const auto draft = history->cloudDraft(kTopicRootId);
 | |
| 	if (!draft) {
 | |
| 		return QString();
 | |
| 	}
 | |
| 	const auto &text = draft->textWithTags.text;
 | |
| 	const auto parts = QStringView(text).split(';');
 | |
| 	auto valid = false;
 | |
| 	auto result = QString();
 | |
| 	for (const auto &part : parts) {
 | |
| 		if (part.startsWith(u"t:"_q)) {
 | |
| 			if (base::StringViewMid(part, 2).toInt() >= base::unixtime::now()) {
 | |
| 				valid = true;
 | |
| 			} else {
 | |
| 				return 0;
 | |
| 			}
 | |
| 		} else if (part.startsWith(u"n:"_q)) {
 | |
| 			result = base::StringViewMid(part, 2).toString();
 | |
| 		}
 | |
| 	}
 | |
| 	return valid ? result : QString();
 | |
| }
 | |
| 
 | |
| TimeId OccupiedBySomeoneTill(History *history) {
 | |
| 	if (!TrackHistoryOccupation(history)) {
 | |
| 		return 0;
 | |
| 	}
 | |
| 	const auto draft = history->cloudDraft(kTopicRootId);
 | |
| 	if (!draft) {
 | |
| 		return 0;
 | |
| 	}
 | |
| 	const auto &text = draft->textWithTags.text;
 | |
| 	const auto parts = QStringView(text).split(';');
 | |
| 	auto valid = false;
 | |
| 	auto result = TimeId();
 | |
| 	for (const auto &part : parts) {
 | |
| 		if (part.startsWith(u"t:"_q)) {
 | |
| 			if (base::StringViewMid(part, 2).toInt() >= base::unixtime::now()) {
 | |
| 				result = base::StringViewMid(part, 2).toInt();
 | |
| 			} else {
 | |
| 				return 0;
 | |
| 			}
 | |
| 		} else if (part.startsWith(u"u:"_q)) {
 | |
| 			if (base::StringViewMid(part, 2).toUInt() != OccupationTag()) {
 | |
| 				valid = true;
 | |
| 			} else {
 | |
| 				return 0;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return valid ? result : 0;
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| Helper::Helper(not_null<Main::Session*> session)
 | |
| : _session(session)
 | |
| , _api(&_session->mtp())
 | |
| , _templates(_session)
 | |
| , _reoccupyTimer([=] { reoccupy(); })
 | |
| , _checkOccupiedTimer([=] { checkOccupiedChats(); }) {
 | |
| 	_api.request(MTPhelp_GetSupportName(
 | |
| 	)).done([=](const MTPhelp_SupportName &result) {
 | |
| 		result.match([&](const MTPDhelp_supportName &data) {
 | |
| 			setSupportName(qs(data.vname()));
 | |
| 		});
 | |
| 	}).fail([=] {
 | |
| 		setSupportName(
 | |
| 			u"[rand^"_q
 | |
| 			+ QString::number(Core::Launcher::Instance().installationTag())
 | |
| 			+ ']');
 | |
| 	}).send();
 | |
| }
 | |
| 
 | |
| std::unique_ptr<Helper> Helper::Create(not_null<Main::Session*> session) {
 | |
| 	//return std::make_unique<Helper>(session); AssertIsDebug();
 | |
| 	const auto valid = session->user()->phone().startsWith(u"424"_q);
 | |
| 	return valid ? std::make_unique<Helper>(session) : nullptr;
 | |
| }
 | |
| 
 | |
| void Helper::registerWindow(not_null<Window::SessionController*> controller) {
 | |
| 	controller->activeChatValue(
 | |
| 	) | rpl::map([](Dialogs::Key key) {
 | |
| 		const auto history = key.history();
 | |
| 		return TrackHistoryOccupation(history) ? history : nullptr;
 | |
| 	}) | rpl::distinct_until_changed(
 | |
| 	) | rpl::start_with_next([=](History *history) {
 | |
| 		updateOccupiedHistory(controller, history);
 | |
| 	}, controller->lifetime());
 | |
| }
 | |
| 
 | |
| void Helper::cloudDraftChanged(not_null<History*> history) {
 | |
| 	chatOccupiedUpdated(history);
 | |
| 	if (history != _occupiedHistory) {
 | |
| 		return;
 | |
| 	}
 | |
| 	occupyIfNotYet();
 | |
| }
 | |
| 
 | |
| void Helper::chatOccupiedUpdated(not_null<History*> history) {
 | |
| 	if (const auto till = OccupiedBySomeoneTill(history)) {
 | |
| 		_occupiedChats[history] = till + 2;
 | |
| 		history->session().changes().historyUpdated(
 | |
| 			history,
 | |
| 			Data::HistoryUpdate::Flag::ChatOccupied);
 | |
| 		checkOccupiedChats();
 | |
| 	} else if (_occupiedChats.take(history)) {
 | |
| 		history->session().changes().historyUpdated(
 | |
| 			history,
 | |
| 			Data::HistoryUpdate::Flag::ChatOccupied);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Helper::checkOccupiedChats() {
 | |
| 	const auto now = base::unixtime::now();
 | |
| 	while (!_occupiedChats.empty()) {
 | |
| 		const auto nearest = ranges::min_element(
 | |
| 			_occupiedChats,
 | |
| 			std::less<>(),
 | |
| 			[](const auto &pair) { return pair.second; });
 | |
| 		if (nearest->second <= now) {
 | |
| 			const auto history = nearest->first;
 | |
| 			_occupiedChats.erase(nearest);
 | |
| 			history->session().changes().historyUpdated(
 | |
| 				history,
 | |
| 				Data::HistoryUpdate::Flag::ChatOccupied);
 | |
| 		} else {
 | |
| 			_checkOccupiedTimer.callOnce(
 | |
| 				(nearest->second - now) * crl::time(1000));
 | |
| 			return;
 | |
| 		}
 | |
| 	}
 | |
| 	_checkOccupiedTimer.cancel();
 | |
| }
 | |
| 
 | |
| void Helper::updateOccupiedHistory(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		History *history) {
 | |
| 	if (isOccupiedByMe(_occupiedHistory)) {
 | |
| 		_occupiedHistory->clearCloudDraft(kTopicRootId);
 | |
| 		_session->api().saveDraftToCloudDelayed(_occupiedHistory);
 | |
| 	}
 | |
| 	_occupiedHistory = history;
 | |
| 	occupyInDraft();
 | |
| }
 | |
| 
 | |
| void Helper::setSupportName(const QString &name) {
 | |
| 	_supportName = name;
 | |
| 	_supportNameNormalized = NormalizeName(name);
 | |
| 	occupyIfNotYet();
 | |
| }
 | |
| 
 | |
| void Helper::occupyIfNotYet() {
 | |
| 	if (!isOccupiedByMe(_occupiedHistory)) {
 | |
| 		occupyInDraft();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Helper::occupyInDraft() {
 | |
| 	if (_occupiedHistory
 | |
| 		&& !isOccupiedBySomeone(_occupiedHistory)
 | |
| 		&& !_supportName.isEmpty()) {
 | |
| 		const auto draft = OccupiedDraft(_supportNameNormalized);
 | |
| 		_occupiedHistory->createCloudDraft(kTopicRootId, &draft);
 | |
| 		_session->api().saveDraftToCloudDelayed(_occupiedHistory);
 | |
| 		_reoccupyTimer.callEach(kReoccupyEach);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Helper::reoccupy() {
 | |
| 	if (isOccupiedByMe(_occupiedHistory)) {
 | |
| 		const auto draft = OccupiedDraft(_supportNameNormalized);
 | |
| 		_occupiedHistory->createCloudDraft(kTopicRootId, &draft);
 | |
| 		_session->api().saveDraftToCloudDelayed(_occupiedHistory);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool Helper::isOccupiedByMe(History *history) const {
 | |
| 	if (const auto tag = ParseOccupationTag(history)) {
 | |
| 		return (tag == OccupationTag());
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool Helper::isOccupiedBySomeone(History *history) const {
 | |
| 	if (const auto tag = ParseOccupationTag(history)) {
 | |
| 		return (tag != OccupationTag());
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void Helper::refreshInfo(not_null<UserData*> user) {
 | |
| 	_api.request(MTPhelp_GetUserInfo(
 | |
| 		user->inputUser
 | |
| 	)).done([=](const MTPhelp_UserInfo &result) {
 | |
| 		applyInfo(user, result);
 | |
| 		if (const auto controller = _userInfoEditPending.take(user)) {
 | |
| 			if (const auto strong = controller->get()) {
 | |
| 				showEditInfoBox(strong, user);
 | |
| 			}
 | |
| 		}
 | |
| 	}).send();
 | |
| }
 | |
| 
 | |
| void Helper::applyInfo(
 | |
| 		not_null<UserData*> user,
 | |
| 		const MTPhelp_UserInfo &result) {
 | |
| 	const auto notify = [&] {
 | |
| 		user->session().changes().peerUpdated(
 | |
| 			user,
 | |
| 			Data::PeerUpdate::Flag::SupportInfo);
 | |
| 	};
 | |
| 	const auto remove = [&] {
 | |
| 		if (_userInformation.take(user)) {
 | |
| 			notify();
 | |
| 		}
 | |
| 	};
 | |
| 	result.match([&](const MTPDhelp_userInfo &data) {
 | |
| 		auto info = UserInfo();
 | |
| 		info.author = qs(data.vauthor());
 | |
| 		info.date = data.vdate().v;
 | |
| 		info.text = TextWithEntities{
 | |
| 			qs(data.vmessage()),
 | |
| 			Api::EntitiesFromMTP(&user->session(), data.ventities().v) };
 | |
| 		if (info.text.empty()) {
 | |
| 			remove();
 | |
| 		} else if (_userInformation[user] != info) {
 | |
| 			_userInformation[user] = info;
 | |
| 			notify();
 | |
| 		}
 | |
| 	}, [&](const MTPDhelp_userInfoEmpty &) {
 | |
| 		remove();
 | |
| 	});
 | |
| }
 | |
| 
 | |
| rpl::producer<UserInfo> Helper::infoValue(not_null<UserData*> user) const {
 | |
| 	return user->session().changes().peerFlagsValue(
 | |
| 		user,
 | |
| 		Data::PeerUpdate::Flag::SupportInfo
 | |
| 	) | rpl::map([=] {
 | |
| 		return infoCurrent(user);
 | |
| 	});
 | |
| }
 | |
| 
 | |
| rpl::producer<QString> Helper::infoLabelValue(
 | |
| 		not_null<UserData*> user) const {
 | |
| 	return infoValue(
 | |
| 		user
 | |
| 	) | rpl::map([](const Support::UserInfo &info) {
 | |
| 		const auto time = Ui::FormatDateTime(
 | |
| 			base::unixtime::parse(info.date));
 | |
| 		return info.author + ", " + time;
 | |
| 	});
 | |
| }
 | |
| 
 | |
| rpl::producer<TextWithEntities> Helper::infoTextValue(
 | |
| 		not_null<UserData*> user) const {
 | |
| 	return infoValue(
 | |
| 		user
 | |
| 	) | rpl::map([](const Support::UserInfo &info) {
 | |
| 		return info.text;
 | |
| 	});
 | |
| }
 | |
| 
 | |
| UserInfo Helper::infoCurrent(not_null<UserData*> user) const {
 | |
| 	const auto i = _userInformation.find(user);
 | |
| 	return (i != end(_userInformation)) ? i->second : UserInfo();
 | |
| }
 | |
| 
 | |
| void Helper::editInfo(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<UserData*> user) {
 | |
| 	if (!_userInfoEditPending.contains(user)) {
 | |
| 		_userInfoEditPending.emplace(user, controller.get());
 | |
| 		refreshInfo(user);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Helper::showEditInfoBox(
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		not_null<UserData*> user) {
 | |
| 	const auto info = infoCurrent(user);
 | |
| 	const auto editData = TextWithTags{
 | |
| 		info.text.text,
 | |
| 		TextUtilities::ConvertEntitiesToTextTags(info.text.entities)
 | |
| 	};
 | |
| 
 | |
| 	const auto save = [=](TextWithTags result, Fn<void(bool)> done) {
 | |
| 		saveInfo(user, TextWithEntities{
 | |
| 			result.text,
 | |
| 			TextUtilities::ConvertTextTagsToEntities(result.tags)
 | |
| 		}, done);
 | |
| 	};
 | |
| 	controller->show(Box<EditInfoBox>(controller, editData, save));
 | |
| }
 | |
| 
 | |
| void Helper::saveInfo(
 | |
| 		not_null<UserData*> user,
 | |
| 		TextWithEntities text,
 | |
| 		Fn<void(bool success)> done) {
 | |
| 	const auto i = _userInfoSaving.find(user);
 | |
| 	if (i != end(_userInfoSaving)) {
 | |
| 		if (i->second.data == text) {
 | |
| 			return;
 | |
| 		} else {
 | |
| 			i->second.data = text;
 | |
| 			_api.request(base::take(i->second.requestId)).cancel();
 | |
| 		}
 | |
| 	} else {
 | |
| 		_userInfoSaving.emplace(user, SavingInfo{ text });
 | |
| 	}
 | |
| 
 | |
| 	TextUtilities::PrepareForSending(
 | |
| 		text,
 | |
| 		Ui::ItemTextDefaultOptions().flags);
 | |
| 	TextUtilities::Trim(text);
 | |
| 
 | |
| 	const auto entities = Api::EntitiesToMTP(
 | |
| 		&user->session(),
 | |
| 		text.entities,
 | |
| 		Api::ConvertOption::SkipLocal);
 | |
| 	_userInfoSaving[user].requestId = _api.request(MTPhelp_EditUserInfo(
 | |
| 		user->inputUser,
 | |
| 		MTP_string(text.text),
 | |
| 		entities
 | |
| 	)).done([=](const MTPhelp_UserInfo &result) {
 | |
| 		applyInfo(user, result);
 | |
| 		done(true);
 | |
| 	}).fail([=] {
 | |
| 		done(false);
 | |
| 	}).send();
 | |
| }
 | |
| 
 | |
| Templates &Helper::templates() {
 | |
| 	return _templates;
 | |
| }
 | |
| 
 | |
| QString ChatOccupiedString(not_null<History*> history) {
 | |
| 	const auto hand = QString::fromUtf8("\xe2\x9c\x8b\xef\xb8\x8f");
 | |
| 	const auto name = ParseOccupationName(history);
 | |
| 	return (name.isEmpty() || name.startsWith(u"[rand^"_q))
 | |
| 		? hand + " chat taken"
 | |
| 		: hand + ' ' + name + " is here";
 | |
| }
 | |
| 
 | |
| QString InterpretSendPath(
 | |
| 		not_null<Window::SessionController*> window,
 | |
| 		const QString &path) {
 | |
| 	QFile f(path);
 | |
| 	if (!f.open(QIODevice::ReadOnly)) {
 | |
| 		return "App Error: Could not open interpret file: " + path;
 | |
| 	}
 | |
| 	const auto content = QString::fromUtf8(f.readAll());
 | |
| 	f.close();
 | |
| 	const auto lines = content.split('\n');
 | |
| 	auto toId = PeerId(0);
 | |
| 	auto topicRootId = MsgId(0);
 | |
| 	auto filePath = QString();
 | |
| 	auto caption = QString();
 | |
| 	for (const auto &line : lines) {
 | |
| 		if (line.startsWith(u"from: "_q)) {
 | |
| 			if (window->session().userId().bare
 | |
| 				!= base::StringViewMid(
 | |
| 					line,
 | |
| 					u"from: "_q.size()).toULongLong()) {
 | |
| 				return "App Error: Wrong current user.";
 | |
| 			}
 | |
| 		} else if (line.startsWith(u"channel: "_q)) {
 | |
| 			const auto channelId = base::StringViewMid(
 | |
| 				line,
 | |
| 				u"channel: "_q.size()).toULongLong();
 | |
| 			toId = peerFromChannel(channelId);
 | |
| 		} else if (line.startsWith(u"topic: "_q)) {
 | |
| 			const auto topicId = base::StringViewMid(
 | |
| 				line,
 | |
| 				u"topic: "_q.size()).toULongLong();
 | |
| 			topicRootId = MsgId(topicId);
 | |
| 		} else if (line.startsWith(u"file: "_q)) {
 | |
| 			const auto path = line.mid(u"file: "_q.size());
 | |
| 			if (!QFile(path).exists()) {
 | |
| 				return "App Error: Could not find file with path: " + path;
 | |
| 			}
 | |
| 			filePath = path;
 | |
| 		} else if (line.startsWith(u"caption: "_q)) {
 | |
| 			caption = line.mid(u"caption: "_q.size());
 | |
| 		} else if (!caption.isEmpty()) {
 | |
| 			caption += '\n' + line;
 | |
| 		} else {
 | |
| 			return "App Error: Invalid command: " + line;
 | |
| 		}
 | |
| 	}
 | |
| 	const auto history = window->session().data().historyLoaded(toId);
 | |
| 	const auto sendTo = [=](not_null<Data::Thread*> thread) {
 | |
| 		window->showThread(thread);
 | |
| 		const auto premium = thread->session().user()->isPremium();
 | |
| 		thread->session().api().sendFiles(
 | |
| 			Storage::PrepareMediaList(
 | |
| 				QStringList(filePath),
 | |
| 				st::sendMediaPreviewSize,
 | |
| 				premium),
 | |
| 			SendMediaType::File,
 | |
| 			{ caption },
 | |
| 			nullptr,
 | |
| 			Api::SendAction(thread));
 | |
| 	};
 | |
| 	if (!history) {
 | |
| 		return "App Error: Could not find channel with id: "
 | |
| 			+ QString::number(peerToChannel(toId).bare);
 | |
| 	} else if (const auto forum = history->asForum()) {
 | |
| 		forum->requestTopic(topicRootId, [=] {
 | |
| 			if (const auto forum = history->asForum()) {
 | |
| 				if (const auto topic = forum->topicFor(topicRootId)) {
 | |
| 					sendTo(topic);
 | |
| 				}
 | |
| 			}
 | |
| 		});
 | |
| 	} else if (!topicRootId) {
 | |
| 		sendTo(history);
 | |
| 	}
 | |
| 	return QString();
 | |
| }
 | |
| 
 | |
| } // namespace Support
 | 
