/* BSD 3-Clause License
 *
 * Copyright © 2008-2022, Jice and the libtcod contributors.
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * 1. Redistributions of source code must retain the above copyright notice,
 *    this list of conditions and the following disclaimer.
 *
 * 2. Redistributions in binary form must reproduce the above copyright notice,
 *    this list of conditions and the following disclaimer in the documentation
 *    and/or other materials provided with the distribution.
 *
 * 3. Neither the name of the copyright holder nor the names of its
 *    contributors may be used to endorse or promote products derived from
 *    this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */
#ifndef LIBTCOD_TILESET_HPP_
#define LIBTCOD_TILESET_HPP_
#include <array>
#include <filesystem>
#include <memory>

#include "error.hpp"
#include "tileset.h"

namespace tcod {
/**
    A character mapping of a Code Page 437 tileset to Unicode.
    \rst
    .. versionadded:: 1.19
    \endrst
 */
static constexpr std::array<int, 256> CHARMAP_CP437 = TCOD_CHARMAP_CP437_;
/**
    A character mapping of a deprecated TCOD tileset to Unicode.
    \rst
    .. versionadded:: 1.19
    \endrst
 */
static constexpr std::array<int, 256> CHARMAP_TCOD = TCOD_CHARMAP_TCOD_;

struct TilesetDeleter {
  void operator()(TCOD_Tileset* tileset) const { TCOD_tileset_delete(tileset); }
};
/**
    @brief A unique pointer to a TCOD_Tileset.

    \rst
    .. versionadded:: 1.19
    \endrst
 */
typedef std::unique_ptr<TCOD_Tileset, TilesetDeleter> TilesetPtr;
/***************************************************************************
    @brief A C++ Tileset container.

    \rst
    .. versionadded:: 1.19
    \endrst
 */
class Tileset {
 public:
  /***************************************************************************
      @brief Construct a new Tileset object.
   */
  Tileset() = default;
  /***************************************************************************
      @brief Construct a new Tileset object with tiles of the given size.  The tileset will be empty.

      @param tile_width The width of the tiles of this object in pixels.
      @param tile_height The width of the tiles of this object in pixels.
   */
  explicit Tileset(int tile_width, int tile_height) : tileset_{TCOD_tileset_new(tile_width, tile_height)} {
    if (!tileset_) throw std::runtime_error(TCOD_get_error());
  }
  /***************************************************************************
      @brief Construct a new Tileset object with tiles of the given size.  The tileset will be empty.

      @param tile_shape The `{width, height}` of the tiles in pixels.
   */
  explicit Tileset(const std::array<int, 2>& tile_shape) : Tileset{tile_shape.at(0), tile_shape.at(1)} {}
  /***************************************************************************
      @brief Pass ownership of a TilesetPtr to a new Tileset.

      @param ptr A `tcod::TilesetPtr`, must not be nullptr.
   */
  explicit Tileset(TilesetPtr ptr) : tileset_{std::move(ptr)} {
    if (!tileset_) throw std::invalid_argument("Pointer must not be nullptr.");
  }
  /***************************************************************************
      @brief Takes ownership of a raw TCOD_Tileset pointer.

      @param ptr A pointer which will now be managed by this object.
   */
  explicit Tileset(TCOD_Tileset* ptr) : tileset_{ptr} {
    if (!tileset_) throw std::invalid_argument("Pointer must not be nullptr.");
  }
  /***************************************************************************
      @brief Get the width of tiles in this Tileset.

      @return int The total width of tiles in pixels.
   */
  [[nodiscard]] auto get_tile_width() const noexcept -> int { return tileset_->tile_width; }
  /***************************************************************************
      @brief Get the height of tiles in this Tileset.

      @return int The total height of tiles in pixels.
   */
  [[nodiscard]] auto get_tile_height() const noexcept -> int { return tileset_->tile_height; }
  /***************************************************************************
      @brief Get the `{width, height}` shape of tiles in this Tileset.

      @return std::array<int, 2> The `{width, height}` of tiles in this Tileset in pixels.
   */
  [[nodiscard]] auto get_tile_shape() const noexcept -> std::array<int, 2> {
    return {tileset_->tile_width, tileset_->tile_height};
  }
  /***************************************************************************
      @brief Return a non-owning pointer to this objects TCOD_Tileset.

      @return TCOD_Tileset
   */
  [[nodiscard]] auto get() noexcept -> TCOD_Tileset* { return tileset_.get(); }
  /***************************************************************************
      @brief Return a non-owning pointer to this objects TCOD_Tileset.

      @return TCOD_Tileset
   */
  [[nodiscard]] auto get() const noexcept -> TCOD_Tileset* { return tileset_.get(); }
  /***************************************************************************
      @brief Release ownership of this Tileset's `TCOD_Tileset*` and return the pointer.

      Using this Tileset afterwards is undefined.
   */
  auto release() noexcept -> TCOD_Tileset* { return tileset_.release(); }
  /***************************************************************************
      @brief Allow implicit conversions to a TCOD_Console reference.
   */
  [[nodiscard]] operator TCOD_Tileset&() { return *tileset_; }
  /***************************************************************************
      @brief Allow implicit conversions to a const TCOD_Console reference.
   */
  [[nodiscard]] operator const TCOD_Tileset&() const { return *tileset_; }

 private:
  TilesetPtr tileset_ = nullptr;
};
#ifndef TCOD_NO_PNG
/**
    @brief Load a tilesheet from a PNG file.

    An exception will be thrown if the file is missing or corrupt.

    Tiles are indexed in row-major order and should be assigned to Unicode codepoints.

    \rst
    .. versionadded:: 1.19
    \endrst

    @tparam ArrayType Must be a `std::vector` or `std::array` like type.  With `size()` and `data()` methods.
    @param path The file path to the PNG tilesheet image.
    @param columns_rows The shape of the grid on the tileset as {columns, rows}.
    @param charmap
        An array of characters where `charmap[tile_index] = codepoint`.
        `tcod::CHARMAP_CP437` or `tcod::CHARMAP_TCOD` are typical values for this argument.

    @return TilesetPtr A unique pointer to a `TCOD_Tileset`.
 */
template <typename ArrayType>
TCOD_NODISCARD inline auto load_tilesheet(
    const std::filesystem::path& path, const std::array<int, 2>& columns_rows, const ArrayType& charmap) -> Tileset {
  tcod::check_path(path);
  TilesetPtr tileset{TCOD_tileset_load(
      path.string().c_str(), columns_rows.at(0), columns_rows.at(1), static_cast<int>(charmap.size()), charmap.data())};
  if (!tileset) throw std::runtime_error(TCOD_get_error());
  return Tileset{std::move(tileset)};
}
#endif  // TCOD_NO_PNG
}  // namespace tcod
#endif  // LIBTCOD_TILESET_HPP_