PHPDOC y unión tardía (estática o dinámica)
-
27-10-2019 - |
Pregunta
La mayoría de los IDE PHP dependen de PHPDOC para obtener pistas sobre el tipo de expresión. Sin embargo, uso con frecuencia este patrón, que no parece estar cubierto:
class Control {
private $label = '';
/** @return ??? */
public static function Make(){ return new static(); }
/** @return ??? */
public function WithLabel($value){ $this->label = $value; return $this; }
/** @return void */
public function Render(){ /* ... */ }
}
class Textbox extends Control {
private $text = '';
/** @return ??? */
public function WithText($text){ $this->width = $text; return $this; }
}
Ahora puedo usar las clases como esta:
Textbox::Make() // <-- late static binding, returns Textbox
->WithLabel('foo') // <-- late dynamic binding, returns Textbox
->WithText('bar') // <-- normal binding, returns Textbox
->Render();
¿Hay alguna forma de reemplazar los '???' s con algo para que la información de escritura sea correcta?
Solución
/** @return Control */
Para no estático:
/** @return $this */
Pero no está documentado en el manual de PhpDoc
Otros consejos
RESPUESTA ACTUALIZADA Tomando como referencia el IDE PHP más popular (PhpStorm 8):
Para @return
puedes usar:
self
$this
Para @method
puedes usar:
$this
Ejemplo:
/**
* Class Model
* @method $this parentMethodA
*/
class Model
{
/**
* @return $this
*/
public function parentMethodB()
{
return $this;
}
/**
* @return self
*/
public function parentMethodC()
{
return $this;
}
}
/**
* Class Person
* @method $this childMethodA
*/
class Person extends Model
{
/**
* @return $this
*/
public function childMethodB()
{
return $this;
}
/**
* @return self
*/
public function childMethodC()
{
return $this;
}
}
$p = new Person();
//In the following lines IDE will recognize those variables as:
$t1 = $p->parentMethodA(); //Instance of Person
$t2 = $p->parentMethodB(); //Instance of Person
$t3 = $p->parentMethodC(); //Instance of Model
$t4 = $p->parentMethodA(); //Instance of Person
$t5 = $p->parentMethodB(); //Instance of Person
$t6 = $p->parentMethodC(); //Instance of Person
Actualización de PhpStorm 10 (EAP)
Parece que ahora static
también se puede usar, pero solo para @return
.
Respuesta actualizada de CVSGuimaraes para incluir opciones estáticas:
/**
* Class Bar
* @method $this parentMethodA
*/
class Bar
{
/**
* @return $this
*/
public function parentMethodB()
{
return $this;
}
/**
* @return self
*/
public function parentMethodC()
{
return $this;
}
/**
* @return static
*/
public static function staticMethod()
{
return new static();
}
/**
* @param $id
*
* @return bool
*/
public function load($id)
{
// test
return $id ? true : false;
}
/**
* @param null $id
*
* @return static
*/
public static function get($id = NULL){
$obj = static::staticMethod();
if (is_null($id)) {
return $obj;
}
if ($obj->load($id)) {
return $obj;
}
return false;
}
}
/**
* Class Foo
* @method $this childMethodA
*/
class Foo extends Bar
{
/**
* @return $this
*/
public function childMethodB()
{
return $this;
}
/**
* @return self
*/
public function childMethodC()
{
return $this;
}
}
/**
* Class Bar
*/
class Baz extends Bar
{
}
$p = new Foo();
/** @var Foo $Foo */
$Foo = 'Foo';
$Baz = 'Bar';
// IntelliJ recognizes the following as:
$t1 = $p->parentMethodA(); //Instance of Foo
$t2 = $p->parentMethodB(); //Instance of Foo
$t3 = $p->parentMethodC(); //Instance of Model
$t4 = $p->childMethodA(); //Instance of Foo
$t5 = $p->childMethodB(); //Instance of Foo
$t6 = $p->childMethodC(); //Instance of Foo
$t7 = $Foo::staticMethod(); //Instance of Foo
$t8 = Foo::staticMethod(); //Instance of Foo
$t9 = $p::staticMethod(); //Instance of Foo
$t10 = $Foo::get(); //Instance of Foo
$t12 = Bar::get(); //Instance of Bar
$t11 = $Baz::get(); // Unknown
Licenciado bajo: CC-BY-SA con atribución
No afiliado a StackOverflow