/*
 * Copyright 2014 Google Inc. All rights reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#ifndef FRUIT_META_METAPROGRAMMING_H
#define FRUIT_META_METAPROGRAMMING_H

#include <fruit/impl/meta/basics.h>
#include <fruit/impl/meta/vector.h>

#include <fruit/impl/fruit_assert.h>
#include <fruit/impl/fruit_internal_forward_decls.h>
#include <fruit/impl/injection_errors.h>
#include <fruit/impl/meta/errors.h>

#include <memory>

namespace fruit {
namespace impl {
namespace meta {

struct IsConstructible {
  template <typename C, typename... Args>
  struct apply;

  template <typename C, typename... Args>
  struct apply<Type<C>, Type<Args>...> {
    using type = Bool<std::is_constructible<C, Args...>::value>;
  };
};

struct IsConstructibleWithVector {
  template <typename C, typename V>
  struct apply;

  template <typename C, typename... Types>
  struct apply<Type<C>, Vector<Type<Types>...>> {
    using type = Bool<std::is_constructible<C, Types...>::value>;
  };
};

struct AddPointer {
  template <typename T>
  struct apply;

  template <typename T>
  struct apply<Type<T>> {
    using type = Type<T*>;
  };
};

struct IsCallable {
  template <typename T>
  struct apply;

  template <typename C>
  struct apply<Type<C>> {
    template <typename C1>
    static Bool<true> test(decltype(&C1::operator()));

    template <typename>
    static Bool<false> test(...);

    using type = decltype(test<C>(nullptr));
  };
};

struct GetCallOperatorSignature {
  template <typename T>
  struct apply;

  template <typename C>
  struct apply<Type<C>> {
    using type = Type<decltype(&C::operator())>;
  };
};

struct AddPointerToVector {
  template <typename V>
  struct apply;

  template <typename... Ts>
  struct apply<Vector<Type<Ts>...>> {
    using type = Vector<Type<Ts*>...>;
  };
};

struct GetNthTypeHelper {
  template <typename N, typename... Ts>
  struct apply;

  template <typename T, typename... Ts>
  struct apply<Int<0>, T, Ts...> {
    using type = T;
  };

  template <int n, typename T, typename... Ts>
  struct apply<Int<n>, T, Ts...> {
    using type = GetNthTypeHelper(Int<n - 1>, Ts...);
  };
};

struct GetNthType {
  template <typename N, typename V>
  struct apply;

  template <typename N, typename... Ts>
  struct apply<N, Vector<Ts...>> {
    using type = GetNthTypeHelper(N, Ts...);
  };
};

struct FunctorResultHelper {
  template <typename MethodSignature>
  struct apply;

  template <typename Result, typename Functor, typename... Args>
  struct apply<Type<Result (Functor::*)(Args...)>> {
    using type = Type<Result>;
  };
};

struct FunctorResult {
  template <typename F>
  struct apply;

  template <typename F>
  struct apply<Type<F>> {
    using type = FunctorResultHelper(Type<decltype(&F::operator())>);
  };
};

struct FunctionSignatureHelper {
  template <typename LambdaMethod>
  struct apply;

  template <typename Result, typename LambdaObject, typename... Args>
  struct apply<Type<Result (LambdaObject::*)(Args...) const>> {
    using type = Type<Result(Args...)>;
  };
};

// Function is either a plain function type of the form T(*)(Args...) or a lambda.
struct FunctionSignature {
  template <typename Function>
  struct apply;

  template <typename Function>
  struct apply<Type<Function>> {
    using CandidateSignature = FunctionSignatureHelper(GetCallOperatorSignature(Type<Function>));
    using type = If(Not(IsCallable(Type<Function>)), ConstructError(NotALambdaErrorTag, Type<Function>),
                    If(Not(IsConstructible(AddPointer(CandidateSignature), Type<Function>)),
                       ConstructError(FunctorUsedAsProviderErrorTag, Type<Function>), CandidateSignature));
  };

  template <typename Result, typename... Args>
  struct apply<Type<Result(Args...)>> {
    using type = Type<Result(Args...)>;
  };

  template <typename Result, typename... Args>
  struct apply<Type<Result (*)(Args...)>> {
    using type = Type<Result(Args...)>;
  };
};

} // namespace meta
} // namespace impl
} // namespace fruit

#endif // FRUIT_META_METAPROGRAMMING_H