rector/rules/TypeDeclaration/NodeTypeAnalyzer/PropertyTypeDecorator.php

91 lines
3.1 KiB
PHP

<?php
declare (strict_types=1);
namespace Rector\TypeDeclaration\NodeTypeAnalyzer;
use PhpParser\Node\ComplexType;
use PhpParser\Node\Expr;
use PhpParser\Node\Identifier;
use PhpParser\Node\Name;
use PhpParser\Node\Stmt\Property;
use PHPStan\Type\UnionType;
use PHPStan\Type\VerbosityLevel;
use Rector\BetterPhpDocParser\PhpDocInfo\PhpDocInfo;
use Rector\BetterPhpDocParser\PhpDocManipulator\PhpDocTypeChanger;
use Rector\Php\PhpVersionProvider;
use Rector\PhpParser\Node\NodeFactory;
use Rector\PHPStanStaticTypeMapper\TypeAnalyzer\UnionTypeAnalyzer;
use Rector\ValueObject\PhpVersionFeature;
final class PropertyTypeDecorator
{
/**
* @readonly
* @var \Rector\PHPStanStaticTypeMapper\TypeAnalyzer\UnionTypeAnalyzer
*/
private $unionTypeAnalyzer;
/**
* @readonly
* @var \Rector\BetterPhpDocParser\PhpDocManipulator\PhpDocTypeChanger
*/
private $phpDocTypeChanger;
/**
* @readonly
* @var \Rector\Php\PhpVersionProvider
*/
private $phpVersionProvider;
/**
* @readonly
* @var \Rector\PhpParser\Node\NodeFactory
*/
private $nodeFactory;
public function __construct(UnionTypeAnalyzer $unionTypeAnalyzer, PhpDocTypeChanger $phpDocTypeChanger, PhpVersionProvider $phpVersionProvider, NodeFactory $nodeFactory)
{
$this->unionTypeAnalyzer = $unionTypeAnalyzer;
$this->phpDocTypeChanger = $phpDocTypeChanger;
$this->phpVersionProvider = $phpVersionProvider;
$this->nodeFactory = $nodeFactory;
}
/**
* @param \PhpParser\Node\Name|\PhpParser\Node\ComplexType|\PhpParser\Node\Identifier $typeNode
*/
public function decoratePropertyUnionType(UnionType $unionType, $typeNode, Property $property, PhpDocInfo $phpDocInfo, bool $changeVarTypeFallback = \true) : void
{
if (!$this->unionTypeAnalyzer->isNullable($unionType)) {
if ($this->phpVersionProvider->isAtLeastPhpVersion(PhpVersionFeature::UNION_TYPES)) {
$property->type = $typeNode;
return;
}
if ($changeVarTypeFallback) {
$this->phpDocTypeChanger->changeVarType($property, $phpDocInfo, $unionType);
}
return;
}
$property->type = $typeNode;
$propertyProperty = $property->props[0];
// add null default
if (!$propertyProperty->default instanceof Expr) {
$propertyProperty->default = $this->nodeFactory->createNull();
}
// has array with defined type? add docs
if (!$this->isDocBlockRequired($unionType)) {
return;
}
if (!$changeVarTypeFallback) {
return;
}
$this->phpDocTypeChanger->changeVarType($property, $phpDocInfo, $unionType);
}
private function isDocBlockRequired(UnionType $unionType) : bool
{
foreach ($unionType->getTypes() as $unionedType) {
if ($unionedType->isArray()->yes()) {
$describedArray = $unionedType->describe(VerbosityLevel::value());
if ($describedArray !== 'array') {
return \true;
}
}
}
return \false;
}
}