Chris@13: Chris@13: */ Chris@13: class ClassMethodsMatcher extends AbstractMatcher Chris@13: { Chris@13: /** Chris@13: * {@inheritdoc} Chris@13: */ Chris@13: public function getMatches(array $tokens, array $info = []) Chris@13: { Chris@13: $input = $this->getInput($tokens); Chris@13: Chris@17: $firstToken = \array_pop($tokens); Chris@13: if (self::tokenIs($firstToken, self::T_STRING)) { Chris@13: // second token is the nekudotayim operator Chris@17: \array_pop($tokens); Chris@13: } Chris@13: Chris@13: $class = $this->getNamespaceAndClass($tokens); Chris@13: Chris@13: try { Chris@13: $reflection = new \ReflectionClass($class); Chris@13: } catch (\ReflectionException $re) { Chris@13: return []; Chris@13: } Chris@13: Chris@16: if (self::needCompleteClass($tokens[1])) { Chris@16: $methods = $reflection->getMethods(); Chris@16: } else { Chris@16: $methods = $reflection->getMethods(\ReflectionMethod::IS_STATIC); Chris@16: } Chris@16: Chris@17: $methods = \array_map(function (\ReflectionMethod $method) { Chris@13: return $method->getName(); Chris@13: }, $methods); Chris@13: Chris@17: return \array_map( Chris@13: function ($name) use ($class) { Chris@17: $chunks = \explode('\\', $class); Chris@17: $className = \array_pop($chunks); Chris@16: Chris@16: return $className . '::' . $name; Chris@13: }, Chris@17: \array_filter($methods, function ($method) use ($input) { Chris@13: return AbstractMatcher::startsWith($input, $method); Chris@13: }) Chris@13: ); Chris@13: } Chris@13: Chris@13: /** Chris@13: * {@inheritdoc} Chris@13: */ Chris@13: public function hasMatched(array $tokens) Chris@13: { Chris@17: $token = \array_pop($tokens); Chris@17: $prevToken = \array_pop($tokens); Chris@13: Chris@13: switch (true) { Chris@13: case self::tokenIs($prevToken, self::T_DOUBLE_COLON) && self::tokenIs($token, self::T_STRING): Chris@13: case self::tokenIs($token, self::T_DOUBLE_COLON): Chris@13: return true; Chris@13: } Chris@13: Chris@13: return false; Chris@13: } Chris@13: }