/*
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