 c46b659aa5
			
		
	
	
		c46b659aa5
		
	
	
	
	
		
			
			- View full statistics in your channels and group chats. - Choose which link preview in added to the message. - Choose if link preview is above or below the text. - Choose if link preview has large or small image. - Quote parts of text in replies. - Add quote formatting. - Reply in another chat. - Add nice looking code blocks with syntax highlighting. - Copy full code block by click on its header. - Send a highlighted code block using ```language syntax. - Change your name color in Chat Settings. - Customize quotes, link previews and replies to your messages.
		
			
				
	
	
		
			344 lines
		
	
	
	
		
			9.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			344 lines
		
	
	
	
		
			9.3 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 "main/main_app_config.h"
 | |
| 
 | |
| #include "apiwrap.h"
 | |
| #include "base/call_delayed.h"
 | |
| #include "main/main_account.h"
 | |
| #include "ui/chat/chat_style.h"
 | |
| 
 | |
| namespace Main {
 | |
| namespace {
 | |
| 
 | |
| constexpr auto kRefreshTimeout = 3600 * crl::time(1000);
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| AppConfig::AppConfig(not_null<Account*> account) : _account(account) {
 | |
| 	account->sessionChanges(
 | |
| 	) | rpl::filter([=](Session *session) {
 | |
| 		return (session != nullptr);
 | |
| 	}) | rpl::start_with_next([=] {
 | |
| 		refresh();
 | |
| 	}, _lifetime);
 | |
| }
 | |
| 
 | |
| AppConfig::~AppConfig() = default;
 | |
| 
 | |
| void AppConfig::start() {
 | |
| 	_account->mtpMainSessionValue(
 | |
| 	) | rpl::start_with_next([=](not_null<MTP::Instance*> instance) {
 | |
| 		_api.emplace(instance);
 | |
| 		refresh();
 | |
| 	}, _lifetime);
 | |
| }
 | |
| 
 | |
| void AppConfig::refresh() {
 | |
| 	if (_requestId || !_api) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_requestId = _api->request(MTPhelp_GetAppConfig(
 | |
| 		MTP_int(_hash)
 | |
| 	)).done([=](const MTPhelp_AppConfig &result) {
 | |
| 		_requestId = 0;
 | |
| 		refreshDelayed();
 | |
| 		result.match([&](const MTPDhelp_appConfig &data) {
 | |
| 			_hash = data.vhash().v;
 | |
| 
 | |
| 			const auto &config = data.vconfig();
 | |
| 			if (config.type() != mtpc_jsonObject) {
 | |
| 				LOG(("API Error: Unexpected config type."));
 | |
| 				return;
 | |
| 			}
 | |
| 			_data.clear();
 | |
| 			for (const auto &element : config.c_jsonObject().vvalue().v) {
 | |
| 				element.match([&](const MTPDjsonObjectValue &data) {
 | |
| 					_data.emplace_or_assign(qs(data.vkey()), data.vvalue());
 | |
| 				});
 | |
| 			}
 | |
| 			parseColorIndices();
 | |
| 			DEBUG_LOG(("getAppConfig result handled."));
 | |
| 			_refreshed.fire({});
 | |
| 		}, [](const MTPDhelp_appConfigNotModified &) {});
 | |
| 	}).fail([=] {
 | |
| 		_requestId = 0;
 | |
| 		refreshDelayed();
 | |
| 	}).send();
 | |
| }
 | |
| 
 | |
| void AppConfig::refreshDelayed() {
 | |
| 	base::call_delayed(kRefreshTimeout, _account, [=] {
 | |
| 		refresh();
 | |
| 	});
 | |
| }
 | |
| 
 | |
| rpl::producer<> AppConfig::refreshed() const {
 | |
| 	return _refreshed.events();
 | |
| }
 | |
| 
 | |
| rpl::producer<> AppConfig::value() const {
 | |
| 	return _refreshed.events_starting_with({});
 | |
| }
 | |
| 
 | |
| template <typename Extractor>
 | |
| auto AppConfig::getValue(const QString &key, Extractor &&extractor) const {
 | |
| 	const auto i = _data.find(key);
 | |
| 	return extractor((i != end(_data))
 | |
| 		? i->second
 | |
| 		: MTPJSONValue(MTP_jsonNull()));
 | |
| }
 | |
| 
 | |
| bool AppConfig::getBool(const QString &key, bool fallback) const {
 | |
| 	return getValue(key, [&](const MTPJSONValue &value) {
 | |
| 		return value.match([&](const MTPDjsonBool &data) {
 | |
| 			return mtpIsTrue(data.vvalue());
 | |
| 		}, [&](const auto &data) {
 | |
| 			return fallback;
 | |
| 		});
 | |
| 	});
 | |
| }
 | |
| 
 | |
| double AppConfig::getDouble(const QString &key, double fallback) const {
 | |
| 	return getValue(key, [&](const MTPJSONValue &value) {
 | |
| 		return value.match([&](const MTPDjsonNumber &data) {
 | |
| 			return data.vvalue().v;
 | |
| 		}, [&](const auto &data) {
 | |
| 			return fallback;
 | |
| 		});
 | |
| 	});
 | |
| }
 | |
| 
 | |
| QString AppConfig::getString(
 | |
| 		const QString &key,
 | |
| 		const QString &fallback) const {
 | |
| 	return getValue(key, [&](const MTPJSONValue &value) {
 | |
| 		return value.match([&](const MTPDjsonString &data) {
 | |
| 			return qs(data.vvalue());
 | |
| 		}, [&](const auto &data) {
 | |
| 			return fallback;
 | |
| 		});
 | |
| 	});
 | |
| }
 | |
| 
 | |
| std::vector<QString> AppConfig::getStringArray(
 | |
| 		const QString &key,
 | |
| 		std::vector<QString> &&fallback) const {
 | |
| 	return getValue(key, [&](const MTPJSONValue &value) {
 | |
| 		return value.match([&](const MTPDjsonArray &data) {
 | |
| 			auto result = std::vector<QString>();
 | |
| 			result.reserve(data.vvalue().v.size());
 | |
| 			for (const auto &entry : data.vvalue().v) {
 | |
| 				if (entry.type() != mtpc_jsonString) {
 | |
| 					return std::move(fallback);
 | |
| 				}
 | |
| 				result.push_back(qs(entry.c_jsonString().vvalue()));
 | |
| 			}
 | |
| 			return result;
 | |
| 		}, [&](const auto &data) {
 | |
| 			return std::move(fallback);
 | |
| 		});
 | |
| 	});
 | |
| }
 | |
| 
 | |
| std::vector<std::map<QString, QString>> AppConfig::getStringMapArray(
 | |
| 		const QString &key,
 | |
| 		std::vector<std::map<QString, QString>> &&fallback) const {
 | |
| 	return getValue(key, [&](const MTPJSONValue &value) {
 | |
| 		return value.match([&](const MTPDjsonArray &data) {
 | |
| 			auto result = std::vector<std::map<QString, QString>>();
 | |
| 			result.reserve(data.vvalue().v.size());
 | |
| 			for (const auto &entry : data.vvalue().v) {
 | |
| 				if (entry.type() != mtpc_jsonObject) {
 | |
| 					return std::move(fallback);
 | |
| 				}
 | |
| 				auto element = std::map<QString, QString>();
 | |
| 				for (const auto &field : entry.c_jsonObject().vvalue().v) {
 | |
| 					const auto &data = field.c_jsonObjectValue();
 | |
| 					if (data.vvalue().type() != mtpc_jsonString) {
 | |
| 						return std::move(fallback);
 | |
| 					}
 | |
| 					element.emplace(
 | |
| 						qs(data.vkey()),
 | |
| 						qs(data.vvalue().c_jsonString().vvalue()));
 | |
| 				}
 | |
| 				result.push_back(std::move(element));
 | |
| 			}
 | |
| 			return result;
 | |
| 		}, [&](const auto &data) {
 | |
| 			return std::move(fallback);
 | |
| 		});
 | |
| 	});
 | |
| }
 | |
| 
 | |
| std::vector<int> AppConfig::getIntArray(
 | |
| 		const QString &key,
 | |
| 		std::vector<int> &&fallback) const {
 | |
| 	return getValue(key, [&](const MTPJSONValue &value) {
 | |
| 		return value.match([&](const MTPDjsonArray &data) {
 | |
| 			auto result = std::vector<int>();
 | |
| 			result.reserve(data.vvalue().v.size());
 | |
| 			for (const auto &entry : data.vvalue().v) {
 | |
| 				if (entry.type() != mtpc_jsonNumber) {
 | |
| 					return std::move(fallback);
 | |
| 				}
 | |
| 				result.push_back(
 | |
| 					int(base::SafeRound(entry.c_jsonNumber().vvalue().v)));
 | |
| 			}
 | |
| 			return result;
 | |
| 		}, [&](const auto &data) {
 | |
| 			return std::move(fallback);
 | |
| 		});
 | |
| 	});
 | |
| }
 | |
| 
 | |
| bool AppConfig::suggestionCurrent(const QString &key) const {
 | |
| 	return !_dismissedSuggestions.contains(key)
 | |
| 		&& ranges::contains(
 | |
| 			get<std::vector<QString>>(
 | |
| 				u"pending_suggestions"_q,
 | |
| 				std::vector<QString>()),
 | |
| 			key);
 | |
| }
 | |
| 
 | |
| rpl::producer<> AppConfig::suggestionRequested(const QString &key) const {
 | |
| 	return value(
 | |
| 	) | rpl::filter([=] {
 | |
| 		return suggestionCurrent(key);
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void AppConfig::dismissSuggestion(const QString &key) {
 | |
| 	Expects(_api.has_value());
 | |
| 
 | |
| 	if (!_dismissedSuggestions.emplace(key).second) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_api->request(MTPhelp_DismissSuggestion(
 | |
| 		MTP_inputPeerEmpty(),
 | |
| 		MTP_string(key)
 | |
| 	)).send();
 | |
| }
 | |
| 
 | |
| void AppConfig::parseColorIndices() {
 | |
| 	constexpr auto parseColor = [](const MTPJSONValue &color) {
 | |
| 		if (color.type() != mtpc_jsonString) {
 | |
| 			LOG(("API Error: Bad type for color element."));
 | |
| 			return uint32();
 | |
| 		}
 | |
| 		const auto value = color.c_jsonString().vvalue().v;
 | |
| 		if (value.size() != 6) {
 | |
| 			LOG(("API Error: Bad length for color element: %1"
 | |
| 				).arg(qs(value)));
 | |
| 			return uint32();
 | |
| 		}
 | |
| 		const auto hex = [](char ch) {
 | |
| 			return (ch >= 'a' && ch <= 'f')
 | |
| 				? (ch - 'a' + 10)
 | |
| 				: (ch >= 'A' && ch <= 'F')
 | |
| 				? (ch - 'A' + 10)
 | |
| 				: (ch >= '0' && ch <= '9')
 | |
| 				? (ch - '0')
 | |
| 				: 0;
 | |
| 		};
 | |
| 		auto result = (uint32(1) << 24);
 | |
| 		for (auto i = 0; i != 6; ++i) {
 | |
| 			result |= (uint32(hex(value[i])) << ((5 - i) * 4));
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 
 | |
| 	struct ParsedColor {
 | |
| 		uint8 colorIndex = Ui::kColorIndexCount;
 | |
| 		std::array<uint32, Ui::kColorPatternsCount> colors;
 | |
| 
 | |
| 		explicit operator bool() const {
 | |
| 			return colorIndex < Ui::kColorIndexCount;
 | |
| 		}
 | |
| 	};
 | |
| 	const auto parseColors = [&](const MTPJSONObjectValue &element) {
 | |
| 		const auto &data = element.data();
 | |
| 		if (data.vvalue().type() != mtpc_jsonArray) {
 | |
| 			LOG(("API Error: Bad value for peer_colors element."));
 | |
| 			return ParsedColor();
 | |
| 		}
 | |
| 		const auto &list = data.vvalue().c_jsonArray().vvalue().v;
 | |
| 		if (list.empty() || list.size() > Ui::kColorPatternsCount) {
 | |
| 			LOG(("API Error: Bad count for peer_colors element: %1"
 | |
| 				).arg(list.size()));
 | |
| 			return ParsedColor();
 | |
| 		}
 | |
| 		const auto index = data.vkey().v.toInt();
 | |
| 		if (index < Ui::kSimpleColorIndexCount
 | |
| 			|| index >= Ui::kColorIndexCount) {
 | |
| 			LOG(("API Error: Bad index for peer_colors element: %1"
 | |
| 				).arg(qs(data.vkey().v)));
 | |
| 			return ParsedColor();
 | |
| 		}
 | |
| 		auto result = ParsedColor{ .colorIndex = uint8(index) };
 | |
| 		auto fill = result.colors.data();
 | |
| 		for (const auto &color : list) {
 | |
| 			*fill++ = parseColor(color);
 | |
| 		}
 | |
| 		return result;
 | |
| 	};
 | |
| 	const auto checkColorsObjectType = [&](const MTPJSONValue &value) {
 | |
| 		if (value.type() != mtpc_jsonObject) {
 | |
| 			if (value.type() != mtpc_jsonArray
 | |
| 				|| !value.c_jsonArray().vvalue().v.empty()) {
 | |
| 				LOG(("API Error: Bad value for [dark_]peer_colors."));
 | |
| 			}
 | |
| 			return false;
 | |
| 		}
 | |
| 		return true;
 | |
| 	};
 | |
| 
 | |
| 	auto colors = std::make_shared<
 | |
| 		std::array<Ui::ColorIndexData, Ui::kColorIndexCount>>();
 | |
| 	getValue(u"peer_colors"_q, [&](const MTPJSONValue &value) {
 | |
| 		if (!checkColorsObjectType(value)) {
 | |
| 			return;
 | |
| 		}
 | |
| 		for (const auto &element : value.c_jsonObject().vvalue().v) {
 | |
| 			if (const auto parsed = parseColors(element)) {
 | |
| 				auto &fields = (*colors)[parsed.colorIndex];
 | |
| 				fields.dark = fields.light = parsed.colors;
 | |
| 			}
 | |
| 		}
 | |
| 	});
 | |
| 	getValue(u"dark_peer_colors"_q, [&](const MTPJSONValue &value) {
 | |
| 		if (!checkColorsObjectType(value)) {
 | |
| 			return;
 | |
| 		}
 | |
| 		for (const auto &element : value.c_jsonObject().vvalue().v) {
 | |
| 			if (const auto parsed = parseColors(element)) {
 | |
| 				(*colors)[parsed.colorIndex].dark = parsed.colors;
 | |
| 			}
 | |
| 		}
 | |
| 	});
 | |
| 
 | |
| 	if (!_colorIndicesCurrent) {
 | |
| 		_colorIndicesCurrent = std::make_unique<Ui::ColorIndicesCompressed>(
 | |
| 			Ui::ColorIndicesCompressed{ std::move(colors) });
 | |
| 		_colorIndicesChanged.fire({});
 | |
| 	} else if (*_colorIndicesCurrent->colors != *colors) {
 | |
| 		_colorIndicesCurrent->colors = std::move(colors);
 | |
| 		_colorIndicesChanged.fire({});
 | |
| 	}
 | |
| }
 | |
| 
 | |
| auto AppConfig::colorIndicesValue() const
 | |
| -> rpl::producer<Ui::ColorIndicesCompressed> {
 | |
| 	return rpl::single(_colorIndicesCurrent
 | |
| 		? *_colorIndicesCurrent
 | |
| 		: Ui::ColorIndicesCompressed()
 | |
| 	) | rpl::then(_colorIndicesChanged.events() | rpl::map([=] {
 | |
| 		return *_colorIndicesCurrent;
 | |
| 	}));
 | |
| }
 | |
| 
 | |
| } // namespace Main
 |