/*
 * Tencent is pleased to support the open source community by making ScriptX available.
 * Copyright (C) 2021 THL A29 Limited, a Tencent company.  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.
 */

#pragma once

#include <utility>
#include "../../src/Exception.h"
#include "../../src/NativeConverter.hpp"
#include "../../src/Reference.h"
#include "../../src/Scope.h"
#include "../../src/utils/GlobalWeakBookkeeping.hpp"
#include "JscHelper.h"

namespace script {

namespace jsc_backend {

inline void retain(JSContextRef context, JSValueRef val) {
  if (val) JSValueProtect(context, val);
}

inline void release(JSContextRef context, JSValueRef val) {
  if (val) JSValueUnprotect(context, val);
}

struct JscBookKeepFetcher {
  template <typename T>
  static ::script::internal::GlobalWeakBookkeeping* get(const T* ref) {
    if (!ref) return nullptr;
    auto& val = JscEngine::refVal(const_cast<T*>(ref));
    if (!val.engine_) return nullptr;
    return &val.engine_->globalWeakBookkeeping_;
  }

  template <typename T>
  static ::script::internal::GlobalWeakBookkeeping::HandleType& handle(const T* ref) {
    auto& val = JscEngine::refVal(const_cast<T*>(ref));
    return val.handle_;
  }
};

using BookKeep = ::script::internal::GlobalWeakBookkeeping::Helper<JscBookKeepFetcher>;

}  // namespace jsc_backend

template <typename T>
Global<T>::Global() noexcept : val_() {}

template <typename T>
Global<T>::Global(const script::Local<T>& localReference)
    : val_{localReference.val_, jsc_backend::currentEngine()} {
  if (val_.ref_) jsc_backend::retain(val_.engine_->context_, val_.ref_);
  jsc_backend::BookKeep::keep(this);
}

template <typename T>
Global<T>::Global(const script::Weak<T>& weak)
    : val_{const_cast<typename jsc_backend::RefTypeMap<T>::jscType>(weak.val_.ref_.get()),
           weak.val_.engine_} {
  if (val_.ref_) jsc_backend::retain(val_.engine_->context_, val_.ref_);
  jsc_backend::BookKeep::keep(this);
}

template <typename T>
Global<T>::Global(const script::Global<T>& copy) : val_() {
  *this = copy;
}

template <typename T>
Global<T>::Global(script::Global<T>&& move) noexcept : val_() {
  *this = std::move(move);
}

template <typename T>
Global<T>::~Global() {
  if (!isEmpty()) {
    EngineScope scope(val_.engine_);
    reset();
  }
}

template <typename T>
Global<T>& Global<T>::operator=(const script::Global<T>& assign) {
  bool wasEmtpy = isEmpty();
  if (!wasEmtpy) {
    jsc_backend::release(val_.engine_->context_, val_.ref_);
  }

  val_.ref_ = assign.val_.ref_;
  val_.engine_ = assign.val_.engine_;
  if (val_.ref_) jsc_backend::retain(val_.engine_->context_, val_.ref_);

  jsc_backend::BookKeep::afterCopy(wasEmtpy, this, &assign);
  return *this;
}

template <typename T>
Global<T>& Global<T>::operator=(script::Global<T>&& move) noexcept {
  bool wasEmtpy = isEmpty();
  if (!wasEmtpy) {
    wasEmtpy = false;
    jsc_backend::release(val_.engine_->context_, val_.ref_);
  }

  val_.ref_ = move.val_.ref_;
  val_.engine_ = move.val_.engine_;
  move.val_.ref_ = nullptr;
  move.val_.engine_ = nullptr;
  jsc_backend::BookKeep::afterMove(wasEmtpy, this, &move);
  return *this;
}

template <typename T>
void Global<T>::swap(Global& rhs) noexcept {
  std::swap(val_.ref_, rhs.val_.ref_);
  std::swap(val_.engine_, rhs.val_.engine_);
  jsc_backend::BookKeep::afterSwap(this, &rhs);
}

template <typename T>
Global<T>& Global<T>::operator=(const script::Local<T>& assign) {
  *this = Global<T>(assign);
  return *this;
}

template <typename T>
Local<T> Global<T>::get() const {
  if (isEmpty()) throw Exception("get on empty Global");
  return Local<T>(val_.ref_);
}

template <typename T>
Local<Value> Global<T>::getValue() const {
  return Local<Value>(val_.ref_);
}

template <typename T>
bool Global<T>::isEmpty() const {
  return val_.ref_ == nullptr;
}

template <typename T>
void Global<T>::reset() {
  if (!isEmpty()) {
    jsc_backend::release(val_.engine_->context_, val_.ref_);
    jsc_backend::BookKeep::remove(this);
    val_.ref_ = nullptr;
    val_.engine_ = nullptr;
  }
}

// == Weak ==

template <typename T>
Weak<T>::Weak() noexcept : val_() {}

template <typename T>
Weak<T>::~Weak() {
  if (!isEmpty()) {
    EngineScope scope(val_.engine_);
    reset();
  }
}

template <typename T>
Weak<T>::Weak(const script::Local<T>& localReference)
    : val_(localReference.val_, jsc_backend::currentEngine()) {
  jsc_backend::BookKeep::keep(this);
}

template <typename T>
Weak<T>::Weak(const script::Global<T>& globalReference)
    : val_(globalReference.val_.ref_, jsc_backend::currentEngine()) {
  jsc_backend::BookKeep::keep(this);
}

template <typename T>
Weak<T>::Weak(const script::Weak<T>& copy) : val_() {
  *this = copy;
}

template <typename T>
Weak<T>::Weak(script::Weak<T>&& move) noexcept : val_() {
  *this = std::move(move);
}

template <typename T>
Weak<T>& Weak<T>::operator=(const script::Weak<T>& assign) {
  bool wasEmtpy = isEmpty();
  val_.ref_ = assign.val_.ref_;
  val_.engine_ = assign.val_.engine_;
  jsc_backend::BookKeep::afterCopy(wasEmtpy, this, &assign);
  return *this;
}

template <typename T>
Weak<T>& Weak<T>::operator=(script::Weak<T>&& move) noexcept {
  bool wasEmpty = isEmpty();
  val_.ref_ = std::move(move.val_.ref_);
  val_.engine_ = std::move(move.val_.engine_);
  jsc_backend::BookKeep::afterMove(wasEmpty, this, &move);
  return *this;
}

template <typename T>
void Weak<T>::swap(Weak& rhs) noexcept {
  std::swap(val_.ref_, rhs.val_.ref_);
  std::swap(val_.engine_, rhs.val_.engine_);
  jsc_backend::BookKeep::afterSwap(this, &rhs);
}

template <typename T>
Weak<T>& Weak<T>::operator=(const script::Local<T>& assign) {
  *this = Weak<T>(assign);
  return *this;
}

template <typename T>
Local<T> Weak<T>::get() const {
  auto value = getValue();
  if (value.isNull()) throw Exception("get on empty Weak");
  return converter::Converter<Local<T>>::toCpp(value);
}

template <typename T>
Local<Value> Weak<T>::getValue() const {
  using type = typename jsc_backend::RefTypeMap<Value>::jscType;
  return Local<Value>(const_cast<type>(val_.ref_.get()));
}

template <typename T>
bool Weak<T>::isEmpty() const {
  return val_.ref_.isEmpty();
}

template <typename T>
void Weak<T>::reset() noexcept {
  if (!isEmpty()) {
    jsc_backend::BookKeep::remove(this);
    val_.ref_.reset(val_.engine_);
    val_.engine_ = nullptr;
  }
}

}  // namespace script