89 lines
		
	
	
	
		
			2.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			89 lines
		
	
	
	
		
			2.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
 | |
| */
 | |
| #pragma once
 | |
| 
 | |
| namespace base {
 | |
| namespace details {
 | |
| 
 | |
| // This implementation was taken from range-v3 library.
 | |
| // It was modified so that more than one of passed function objects
 | |
| // could be called with an argument list and the first one that
 | |
| // matches gets used instead of a compile-time ambiguity error.
 | |
| //
 | |
| // This allows to write "default" visitor handlers with [](auto&&) syntax.
 | |
| 
 | |
| template <typename ...Args>
 | |
| constexpr bool is_callable_v = rpl::details::is_callable_plain_v<Args...>;
 | |
| 
 | |
| template <typename ...Args>
 | |
| struct overloaded;
 | |
| 
 | |
| template <>
 | |
| struct overloaded<> {
 | |
| };
 | |
| 
 | |
| template <typename First, typename ...Rest>
 | |
| struct overloaded<First, Rest...>
 | |
| 	: private First
 | |
| 	, private overloaded<Rest...> {
 | |
| 
 | |
| private:
 | |
| 	using Others = overloaded<Rest...>;
 | |
| 
 | |
| public:
 | |
| 	overloaded() = default;
 | |
| 
 | |
| 	constexpr overloaded(First first, Rest... rest)
 | |
| 	: First(std::move(first))
 | |
| 	, Others(std::move(rest)...) {
 | |
| 	}
 | |
| 
 | |
| 	template <typename... Args>
 | |
| 	auto operator()(Args&&... args)
 | |
| 	-> decltype(std::declval<First&>()(std::forward<Args>(args)...)) {
 | |
| 		return static_cast<First&>(*this)(std::forward<Args>(args)...);
 | |
| 	}
 | |
| 
 | |
| 	template <typename... Args>
 | |
| 	auto operator()(Args&&... args) const
 | |
| 	-> decltype(std::declval<const First&>()(std::forward<Args>(args)...)) {
 | |
| 		return static_cast<const First&>(*this)(std::forward<Args>(args)...);
 | |
| 	}
 | |
| 
 | |
| 	template <
 | |
| 		typename... Args,
 | |
| 		typename = std::enable_if_t<!is_callable_v<First&, Args&&...>>>
 | |
| 	auto operator()(Args&&... args)
 | |
| 	-> decltype(std::declval<Others&>()(std::forward<Args>(args)...)) {
 | |
| 		return static_cast<Others&>(*this)(std::forward<Args>(args)...);
 | |
| 	}
 | |
| 
 | |
| 	template <
 | |
| 		typename... Args,
 | |
| 		typename = std::enable_if_t<!is_callable_v<const First&, Args&&...>>>
 | |
| 	auto operator()(Args&&... args) const
 | |
| 	-> decltype(std::declval<const Others&>()(std::forward<Args>(args)...)) {
 | |
| 		return static_cast<const Others&>(*this)(std::forward<Args>(args)...);
 | |
| 	}
 | |
| 
 | |
| };
 | |
| 
 | |
| } // namespace details
 | |
| 
 | |
| template <typename Function>
 | |
| Function overload(Function &&function) {
 | |
| 	return std::forward<Function>(function);
 | |
| }
 | |
| 
 | |
| template <typename ...Functions, typename = std::enable_if_t<(sizeof...(Functions) > 1)>>
 | |
| auto overload(Functions ...functions) {
 | |
| 	return details::overloaded<Functions...>(std::move(functions)...);
 | |
| }
 | |
| 
 | |
| } // namespace base
 | |
| 
 | 
