Skip to content

Commit f5e2e32

Browse files
committed
Support @mixin above traits
1 parent 0d0de94 commit f5e2e32

7 files changed

+126
-0
lines changed

src/Reflection/Mixin/MixinMethodsClassReflectionExtension.php

+9
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,15 @@ private function findMethod(ClassReflection $classReflection, string $methodName
7474
return new MixinMethodReflection($method, $static);
7575
}
7676

77+
foreach ($classReflection->getTraits() as $traitClass) {
78+
$methodWithDeclaringClass = $this->findMethod($traitClass, $methodName);
79+
if ($methodWithDeclaringClass === null) {
80+
continue;
81+
}
82+
83+
return $methodWithDeclaringClass;
84+
}
85+
7786
$parentClass = $classReflection->getParentClass();
7887
while ($parentClass !== null) {
7988
$method = $this->findMethod($parentClass, $methodName);

src/Reflection/Mixin/MixinPropertiesClassReflectionExtension.php

+9
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,15 @@ private function findProperty(ClassReflection $classReflection, string $property
6565
return $property;
6666
}
6767

68+
foreach ($classReflection->getTraits() as $traitClass) {
69+
$methodWithDeclaringClass = $this->findProperty($traitClass, $propertyName);
70+
if ($methodWithDeclaringClass === null) {
71+
continue;
72+
}
73+
74+
return $methodWithDeclaringClass;
75+
}
76+
6877
$parentClass = $classReflection->getParentClass();
6978
while ($parentClass !== null) {
7079
$property = $this->findProperty($parentClass, $propertyName);

tests/PHPStan/Analyser/NodeScopeResolverTest.php

+2
Original file line numberDiff line numberDiff line change
@@ -114,6 +114,8 @@ public function dataFileAsserts(): iterable
114114
yield from $this->gatherAssertTypes(__DIR__ . '/../Rules/Methods/data/conditional-complex-templates.php');
115115

116116
yield from $this->gatherAssertTypes(__DIR__ . '/../Rules/Methods/data/bug-7511.php');
117+
yield from $this->gatherAssertTypes(__DIR__ . '/../Rules/Properties/data/trait-mixin.php');
118+
yield from $this->gatherAssertTypes(__DIR__ . '/../Rules/Methods/data/trait-mixin.php');
117119
yield from $this->gatherAssertTypes(__DIR__ . '/../Rules/Comparison/data/bug-4708.php');
118120
yield from $this->gatherAssertTypes(__DIR__ . '/../Rules/Functions/data/bug-7156.php');
119121
yield from $this->gatherAssertTypes(__DIR__ . '/../Rules/Arrays/data/bug-6364.php');

tests/PHPStan/Rules/Methods/CallMethodsRuleTest.php

+9
Original file line numberDiff line numberDiff line change
@@ -3345,4 +3345,13 @@ public function testNoNamedArguments(): void
33453345
]);
33463346
}
33473347

3348+
public function testTraitMixin(): void
3349+
{
3350+
$this->checkThisOnly = false;
3351+
$this->checkNullables = true;
3352+
$this->checkUnionTypes = true;
3353+
$this->checkExplicitMixed = true;
3354+
$this->analyse([__DIR__ . '/data/trait-mixin.php'], []);
3355+
}
3356+
33483357
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
<?php
2+
3+
namespace TraitMixinMethods;
4+
5+
use function PHPStan\Testing\assertType;
6+
7+
/**
8+
* @template T
9+
*/
10+
interface Foo
11+
{
12+
13+
/** @return T */
14+
public function get();
15+
16+
}
17+
18+
/**
19+
* @mixin Foo<static>
20+
*/
21+
trait FooTrait
22+
{
23+
24+
}
25+
26+
class Usages
27+
{
28+
29+
use FooTrait;
30+
31+
}
32+
33+
class ChildUsages extends Usages
34+
{
35+
36+
}
37+
38+
function (Usages $u): void {
39+
assertType(Usages::class, $u->get());
40+
};
41+
42+
function (ChildUsages $u): void {
43+
assertType(ChildUsages::class, $u->get());
44+
};

tests/PHPStan/Rules/Properties/AccessPropertiesRuleTest.php

+8
Original file line numberDiff line numberDiff line change
@@ -949,4 +949,12 @@ public function testBug9694(): void
949949
$this->analyse([__DIR__ . '/data/bug-9694.php'], []);
950950
}
951951

952+
public function testTraitMixin(): void
953+
{
954+
$this->checkThisOnly = false;
955+
$this->checkUnionTypes = true;
956+
$this->checkDynamicProperties = true;
957+
$this->analyse([__DIR__ . '/data/trait-mixin.php'], []);
958+
}
959+
952960
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
<?php
2+
3+
namespace TraitMixinProperties;
4+
5+
use function PHPStan\Testing\assertType;
6+
7+
/**
8+
* @template T
9+
*/
10+
class Foo
11+
{
12+
13+
/** @var T */
14+
public $a;
15+
16+
}
17+
18+
/**
19+
* @mixin Foo<static>
20+
*/
21+
trait FooTrait
22+
{
23+
24+
}
25+
26+
#[\AllowDynamicProperties]
27+
class Usages
28+
{
29+
30+
use FooTrait;
31+
32+
}
33+
34+
class ChildUsages extends Usages
35+
{
36+
37+
}
38+
39+
function (Usages $u): void {
40+
assertType(Usages::class, $u->a);
41+
};
42+
43+
function (ChildUsages $u): void {
44+
assertType(ChildUsages::class, $u->a);
45+
};

0 commit comments

Comments
 (0)