namespace Illuminate\Filesystem;

use Symfony\Component\Finder\Finder;
// use Finder
class ClassFinder
{
    /**
     * Find all the class and interface names in a given directory.
     *
     * @param  string  $directory
     * @return array
     */
    public function findClasses($directory)
    {// find all class and interface name maybe is too exaggerated
        $classes = [];// class listener

        foreach (Finder::create()->in($directory)->name('*.php') as $file) {
            $classes[] = $this->findClass($file->getRealPath());
        }// check all of this .php files name, if has .class.php ,

        return array_filter($classes);// if only on parameter ,then they will delete this value that it is null.
    }

    /**
     * Extract the class name from the file at the given path.
     *
     * @param  string  $path
     * @return string|null
     */
    public function findClass($path)
    {// extract  means get something from other thing.
       // like get class name by file path,
        $namespace = null;// set a null namespace

        $tokens = token_get_all(file_get_contents($path));// get all string like a php code

        foreach ($tokens as $key => $token) {
            if ($this->tokenIsNamespace($token)) {// if has namespace
                $namespace = $this->getNamespace($key + 2, $tokens);// get it
            } elseif ($this->tokenIsClassOrInterface($token)) {// if this class is a interface
                return ltrim($namespace.'\\'.$this->getClass($key + 2, $tokens), '\\');// use it self to make ltrim
            }
        }
    }

    /**
     * Find the namespace in the tokens starting at a given key.
     *
     * @param  int  $key
     * @param  array  $tokens
     * @return string|null
     */
    protected function getNamespace($key, array $tokens)
    {// Find the namespace in tokens starting at a given key.
        $namespace = null;// a namespace

        $tokenCount = count($tokens);// get this array count

        for ($i = $key; $i < $tokenCount; $i++) {// if has tokens
            if ($this->isPartOfNamespace($tokens[$i])) {// if this is a  namespace ,even it is a array
                $namespace .= $tokens[$i][1];//
            } elseif ($tokens[$i] == ';') {// if
                return $namespace;// null return
            }
        }
    }

    /**
     * Find the class in the tokens starting at a given key.
     *
     * @param  int  $key
     * @param  array  $tokens
     * @return string|null
     */
    protected function getClass($key, array $tokens)
    {
        $class = null;

        $tokenCount = count($tokens);

        for ($i = $key; $i < $tokenCount; $i++) {
            if ($this->isPartOfClass($tokens[$i])) {
                $class .= $tokens[$i][1];
            } elseif ($this->isWhitespace($tokens[$i])) {
                return $class;
            }
        }
    }// same too
    /**
     * Determine if the given token is a namespace keyword.
     *
     * @param  array|string  $token
     * @return bool
     */
    protected function tokenIsNamespace($token)
    {
        return is_array($token) && $token[0] == T_NAMESPACE;// is a array then check this token[0] is a T_NAMESPACE
    }// determine given token is a namespace keyword

    /**
     * Determine if the given token is a class or interface keyword.
     *
     * @param  array|string  $token
     * @return bool
     */
    protected function tokenIsClassOrInterface($token)
    {
        return is_array($token) && ($token[0] == T_CLASS || $token[0] == T_INTERFACE);
    }//Determine if the given token is a class or interface keyword.
   // first this token is array, and the first is a T_CLASS

    /**
     * Determine if the given token is part of the namespace.
     *
     * @param  array|string  $token
     * @return bool
     */
    protected function isPartOfNamespace($token)
    {
        return is_array($token) && ($token[0] == T_STRING || $token[0] == T_NS_SEPARATOR);
    }// check given token is part of the namesapce.

    /**
     * Determine if the given token is part of the class.
     *
     * @param  array|string  $token
     * @return bool
     */
    protected function isPartOfClass($token)
    {
        return is_array($token) && $token[0] == T_STRING;
    }// check class

    /**
     * Determine if the given token is whitespace.
     *
     * @param  array|string  $token
     * @return bool
     */
    protected function isWhitespace($token)
    {
        return is_array($token) && $token[0] == T_WHITESPACE;
    }// check is Whitespace
}