| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  | 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
 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "lang/lang_tag.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum lngtag_count : int; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Lang { | 
					
						
							|  |  |  | namespace details { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline constexpr auto kPluralCount = 6; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-19 14:41:00 +02:00
										 |  |  | template <typename Tag> | 
					
						
							|  |  |  | inline constexpr ushort TagValue(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | template <typename P> | 
					
						
							|  |  |  | using S = std::decay_t<decltype(std::declval<P>()(QString()))>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | [[nodiscard]] QString Current(ushort key); | 
					
						
							|  |  |  | [[nodiscard]] rpl::producer<QString> Value(ushort key); | 
					
						
							|  |  |  | [[nodiscard]] bool IsNonDefaultPlural(ushort keyBase); | 
					
						
							| 
									
										
										
										
											2019-06-18 14:16:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | template <int Index, typename Type, typename Tuple> | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | [[nodiscard]] Type ReplaceUnwrapTuple(Type accumulated, const Tuple &tuple) { | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 	return accumulated; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | template < | 
					
						
							|  |  |  | 	int Index, | 
					
						
							|  |  |  | 	typename Type, | 
					
						
							|  |  |  | 	typename Tuple, | 
					
						
							|  |  |  | 	typename Tag, | 
					
						
							|  |  |  | 	typename ...Tags> | 
					
						
							|  |  |  | [[nodiscard]] Type ReplaceUnwrapTuple( | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 		Type accumulated, | 
					
						
							|  |  |  | 		const Tuple &tuple, | 
					
						
							|  |  |  | 		Tag tag, | 
					
						
							|  |  |  | 		Tags ...tags) { | 
					
						
							|  |  |  | 	return ReplaceUnwrapTuple<Index + 1>( | 
					
						
							|  |  |  | 		ReplaceTag<Type>::Call( | 
					
						
							|  |  |  | 			std::move(accumulated), | 
					
						
							|  |  |  | 			tag, | 
					
						
							|  |  |  | 			std::get<Index>(tuple)), | 
					
						
							|  |  |  | 		tuple, | 
					
						
							|  |  |  | 		tags...); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename ...Tags> | 
					
						
							|  |  |  | struct ReplaceUnwrap; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | struct ReplaceUnwrap<> { | 
					
						
							|  |  |  | 	template <typename Type> | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | 	[[nodiscard]] static Type Call(Type accumulated) { | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 		return accumulated; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Tag, typename ...Tags> | 
					
						
							|  |  |  | struct ReplaceUnwrap<Tag, Tags...> { | 
					
						
							|  |  |  | 	template <typename Type, typename Value, typename ...Values> | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | 	[[nodiscard]] static Type Call( | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 			Type accumulated, | 
					
						
							|  |  |  | 			const Value &value, | 
					
						
							|  |  |  | 			const Values &...values) { | 
					
						
							|  |  |  | 		return ReplaceUnwrap<Tags...>::template Call( | 
					
						
							|  |  |  | 			ReplaceTag<Type>::Call( | 
					
						
							|  |  |  | 				std::move(accumulated), | 
					
						
							| 
									
										
										
										
											2019-06-19 14:41:00 +02:00
										 |  |  | 				TagValue<Tag>(), | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 				value), | 
					
						
							|  |  |  | 			values...); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename ...Tags> | 
					
						
							|  |  |  | struct Producer { | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | 	template <typename P, typename ...Values> | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | 	[[nodiscard]] static rpl::producer<S<P>> Combine(ushort base, P p, Values &...values) { | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 		return rpl::combine( | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | 			Value(base), | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 			std::move(values)... | 
					
						
							|  |  |  | 		) | rpl::map([p = std::move(p)](auto tuple) { | 
					
						
							| 
									
										
										
										
											2019-06-19 14:41:00 +02:00
										 |  |  | 			return ReplaceUnwrapTuple<1>(p(std::get<0>(tuple)), tuple, TagValue<Tags>()...); | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | 	template <typename P, typename ...Values> | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | 	[[nodiscard]] static S<P> Current(ushort base, P p, const Values &...values) { | 
					
						
							| 
									
										
										
										
											2019-06-18 14:16:43 +02:00
										 |  |  | 		return ReplaceUnwrap<Tags...>::template Call( | 
					
						
							|  |  |  | 			p(Lang::details::Current(base)), | 
					
						
							|  |  |  | 			values...); | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | struct Producer<> { | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | 	template <typename P> | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | 	[[nodiscard]] static rpl::producer<S<P>> Combine(ushort base, P p) { | 
					
						
							|  |  |  | 		return Value(base) | rpl::map(std::move(p)); | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | 	template <typename P> | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | 	[[nodiscard]] static S<P> Current(ushort base, P p) { | 
					
						
							| 
									
										
										
										
											2019-06-18 14:16:43 +02:00
										 |  |  | 		return p(Lang::details::Current(base)); | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename ...Tags> | 
					
						
							|  |  |  | struct Producer<lngtag_count, Tags...> { | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | 	template <typename P, typename ...Values> | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | 	[[nodiscard]] static rpl::producer<S<P>> Combine( | 
					
						
							| 
									
										
										
										
											2019-06-19 17:42:16 +02:00
										 |  |  | 			ushort base, | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 			P p, | 
					
						
							|  |  |  | 			lngtag_count type, | 
					
						
							|  |  |  | 			rpl::producer<float64> &count, | 
					
						
							|  |  |  | 			Values &...values) { | 
					
						
							|  |  |  | 		return rpl::combine( | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | 			Value(base), | 
					
						
							|  |  |  | 			Value(base + 1), | 
					
						
							|  |  |  | 			Value(base + 2), | 
					
						
							|  |  |  | 			Value(base + 3), | 
					
						
							|  |  |  | 			Value(base + 4), | 
					
						
							|  |  |  | 			Value(base + 5), | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 			std::move(count), | 
					
						
							|  |  |  | 			std::move(values)... | 
					
						
							|  |  |  | 		) | rpl::map([base, type, p = std::move(p)](auto tuple) { | 
					
						
							|  |  |  | 			auto plural = Plural(base, std::get<6>(tuple), type); | 
					
						
							|  |  |  | 			const auto select = [&] { | 
					
						
							|  |  |  | 				switch (plural.keyShift) { | 
					
						
							|  |  |  | 				case 0: return std::get<0>(tuple); | 
					
						
							|  |  |  | 				case 1: return std::get<1>(tuple); | 
					
						
							|  |  |  | 				case 2: return std::get<2>(tuple); | 
					
						
							|  |  |  | 				case 3: return std::get<3>(tuple); | 
					
						
							|  |  |  | 				case 4: return std::get<4>(tuple); | 
					
						
							|  |  |  | 				case 5: return std::get<5>(tuple); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				Unexpected("Lang shift value in Plural result."); | 
					
						
							|  |  |  | 			}; | 
					
						
							|  |  |  | 			return ReplaceUnwrapTuple<7>( | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | 				ReplaceTag<S<P>>::Call( | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 					p(select()), | 
					
						
							| 
									
										
										
										
											2019-06-19 14:41:00 +02:00
										 |  |  | 					TagValue<lngtag_count>(), | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | 					StartReplacements<S<P>>::Call( | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 						std::move(plural.replacement))), | 
					
						
							|  |  |  | 				tuple, | 
					
						
							| 
									
										
										
										
											2019-06-19 14:41:00 +02:00
										 |  |  | 				TagValue<Tags>()...); | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | 	template <typename P, typename ...Values> | 
					
						
							| 
									
										
										
										
											2020-09-30 12:11:44 +03:00
										 |  |  | 	[[nodiscard]] static S<P> Current( | 
					
						
							| 
									
										
										
										
											2019-06-19 17:42:16 +02:00
										 |  |  | 			ushort base, | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 			P p, | 
					
						
							|  |  |  | 			lngtag_count type, | 
					
						
							|  |  |  | 			float64 count, | 
					
						
							|  |  |  | 			const Values &...values) { | 
					
						
							|  |  |  | 		auto plural = Plural(base, count, type); | 
					
						
							|  |  |  | 		return ReplaceUnwrap<Tags...>::template Call( | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | 			ReplaceTag<S<P>>::Call( | 
					
						
							| 
									
										
										
										
											2019-06-19 17:42:16 +02:00
										 |  |  | 				p(Lang::details::Current(base + plural.keyShift)), | 
					
						
							| 
									
										
										
										
											2019-06-19 14:41:00 +02:00
										 |  |  | 				TagValue<lngtag_count>(), | 
					
						
							| 
									
										
										
										
											2019-06-22 22:42:30 +02:00
										 |  |  | 				StartReplacements<S<P>>::Call( | 
					
						
							| 
									
										
										
										
											2019-06-16 15:08:15 +02:00
										 |  |  | 					std::move(plural.replacement))), | 
					
						
							|  |  |  | 			values...); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace details
 | 
					
						
							|  |  |  | } // namespace Lang
 |