/*
 * This file is part of the TrinityCore Project. See AUTHORS file for Copyright information
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the
 * Free Software Foundation; either version 2 of the License, or (at your
 * option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program. If not, see .
 */
#ifndef TRINITY_ADVSTD_H
#define TRINITY_ADVSTD_H
#include 
#include 
#include 
#include 
// this namespace holds implementations of upcoming stdlib features that our c++ version doesn't have yet
namespace advstd
{
    // C++17 std::apply (constrained to only function pointers, not all callable)
    template 
    using apply_tuple_type = std::tuple>...>;
    template 
    R apply_impl(R(*func)(Ts...), apply_tuple_type&& args, std::index_sequence)
    {
        return func(std::get(std::forward>(args))...);
    }
    template 
    R apply(R(*func)(Ts...), apply_tuple_type&& args)
    {
        return apply_impl(func, std::forward>(args), std::index_sequence_for{});
    }
#define forward_1v(stdname, type) template  constexpr type stdname ## _v = std::stdname::value
#define forward_2v(stdname, type) template  constexpr type stdname ## _v = std::stdname::value
    // C++17 std::is_same_v
    forward_2v(is_same, bool);
    // C++17 std::is_integral_v
    forward_1v(is_integral, bool);
    // C++17 std::is_assignable_v
    forward_2v(is_assignable, bool);
    // C++17 std::is_signed_v
    forward_1v(is_signed, bool);
    // C++17 std::is_unsigned_v
    forward_1v(is_unsigned, bool);
    // C++17 std::is_base_of_v
    forward_2v(is_base_of, bool);
    // C++17 std::is_floating_point_v
    forward_1v(is_floating_point, bool);
    // C++17 std::is_pointer_v
    forward_1v(is_pointer, bool);
    // C++17 std::is_reference_v
    forward_1v(is_reference, bool);
    // C++17 std::tuple_size_v
    forward_1v(tuple_size, size_t);
    // C++17 std::is_enum_v
    forward_1v(is_enum, bool);
    // C++17 std::is_arithmetic_v
    forward_1v(is_arithmetic, bool);
    // C++17 std::is_move_assignable_v
    forward_1v(is_move_assignable, bool);
#undef forward_1v
#undef forward_2v
    // C++17 std::size
    template 
    constexpr auto size(const C& c) { return c.size(); }
    template 
    constexpr std::size_t size(const T(&)[N]) noexcept { return N; }
    // C++17 std::data
    template 
    constexpr auto data(C& c) { return c.data(); }
    template 
    constexpr auto data(C const& c) { return c.data(); }
    template 
    constexpr T* data(T(&a)[N]) noexcept { return a; }
    template 
    constexpr T const* data(const T(&a)[N]) noexcept { return a; }
    template 
    constexpr T const* data(std::initializer_list l) noexcept { return l.begin(); }
    // C++17 std::gcd
    template 
    constexpr std::enable_if_t && advstd::is_unsigned_v, std::common_type_t> gcd(T1 _m, T2 _n)
    {
        using T = std::common_type_t;
        T n=_n, m=_m;
        while (n)
        {
            T o = m;
            m = n;
            n = o%n;
        }
        return m;
    }
    // C++17 std::lcm
    template 
    constexpr std::enable_if_t && advstd::is_unsigned_v, std::common_type_t> lcm(T1 m, T2 n)
    {
        return (m/gcd(m, n))*n;
    }
    // C++20 std::remove_cvref_t
    template 
    using remove_cvref_t = std::remove_cv_t>;
    template
    struct negation : std::integral_constant { };
    template 
    struct conjunction : std::true_type { };
    template 
    struct conjunction : B1 { };
    template 
    struct conjunction : std::conditional_t, B1> { };
    template 
    struct disjunction : std::false_type { };
    template 
    struct disjunction : B1 { };
    template 
    struct disjunction : std::conditional_t>  { };
    template 
    constexpr T const& clamp(T const& val, T const& lo, T const& hi)
    {
        if (hi < val)
            return hi;
        if (val < lo)
            return lo;
        return val;
    }
}
#endif