// file      : build/path.ixx -*- C++ -*-
// copyright : Copyright (c) 2014-2015 Code Synthesis Tools CC
// license   : MIT; see accompanying LICENSE file

#ifdef _WIN32
#  include <cctype>  // std::tolower
#  include <cwctype> // std::towlower
#endif

namespace build
{
#ifdef _WIN32
  template <>
  inline char path_traits<char>::
  tolower (char c)
  {
    return std::tolower (c);
  }

  template <>
  inline wchar_t path_traits<wchar_t>::
  tolower (wchar_t c)
  {
    return std::towlower (c);
  }
#endif

  template <typename C>
  inline bool basic_path<C>::
  absolute () const
  {
#ifdef _WIN32
    return path_.size () > 1 && path_[1] == ':';
#else
    return !path_.empty () && traits::is_separator (path_[0]);
#endif
  }

  template <typename C>
  inline bool basic_path<C>::
  root () const
  {
#ifdef _WIN32
    return path_.size () == 2 && path_[1] == ':';
#else
    return path_.size () == 1 && traits::is_separator (path_[0]);
#endif
  }

  template <typename C>
  inline bool basic_path<C>::
  sub (const basic_path& p) const
  {
    size_type n (p.path_.size ());

    if (n == 0)
      return true;

    size_type m (path_.size ());

    // The second condition guards against the /foo-bar vs /foo case.
    //
    return m >= n && path_.compare (0, n, p.path_) == 0 &&
      (traits::is_separator (p.path_.back ()) || // p ends with a separator
       m == n                                 || // *this == p
       traits::is_separator (path_[n]));         // next char is a separator
  }

  template <typename C>
  inline bool basic_path<C>::
  sup (const basic_path& p) const
  {
    size_type n (p.path_.size ());

    if (n == 0)
      return true;

    size_type m (path_.size ());

    // The second condition guards against the /foo-bar vs bar case.
    //
    return m >= n && path_.compare (m - n, n, p.path_) == 0 &&
      (m == n                                   || // *this == p
       traits::is_separator (path_[m - n - 1]));   // prev char is a separator
  }

  template <typename C>
  inline basic_path<C>& basic_path<C>::
  complete ()
  {
    if (relative ())
      *this = current () / *this;

    return *this;
  }

  template <typename C>
  inline basic_path<C> basic_path<C>::
  root_directory () const
  {
    return absolute ()
#ifdef _WIN32
      ? path (path_, 2)
#else
      ? path ("/")
#endif
      : path ();
  }

  template <typename C>
  inline basic_path<C> basic_path<C>::
  base () const
  {
    size_type p (traits::find_extension (path_));
    return p != string_type::npos ? basic_path (path_.c_str (), p) : *this;
  }

  template <typename C>
  inline const C* basic_path<C>::
  extension () const
  {
    size_type p (traits::find_extension (path_));
    return p != string_type::npos ? path_.c_str () + p + 1 : nullptr;
  }

#ifndef _WIN32
  template <typename C>
  inline typename basic_path<C>::string_type basic_path<C>::
  posix_string () const
  {
    return string ();
  }
#endif
}