Documentação do Symfony - versão 3.4
Renderizada do repositório symfony-docs-pt-BR no Github

Como estender uma Classe sem usar Herança

Para permitir que várias classes adicionem métodos para uma outra, você pode definir o método mágico __call() na classe que você deseja que seja estendida da seguinte forma:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
class Foo
{
    // ...

    public function __call($method, $arguments)
    {
        // cria um evento chamado 'foo.method_is_not_found'
        $event = new HandleUndefinedMethodEvent($this, $method, $arguments);
        $this->dispatcher->dispatch($this, 'foo.method_is_not_found', $event);

        // nenhum listener foi capaz de processar o evento? O método não existe
        if (!$event->isProcessed()) {
            throw new \Exception(sprintf('Call to undefined method %s::%s.', get_class($this), $method));
        }

        // retorna o valor retornado pelo listener
        return $event->getReturnValue();
    }
}

Ela utiliza um HandleUndefinedMethodEvent especial que também deve ser criado. Esta é uma classe genérica que poderia ser reutilizada cada vez que você precisa utilizar esse padrão de extensão de classe:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
use Symfony\Component\EventDispatcher\Event;

class HandleUndefinedMethodEvent extends Event
{
    protected $subject;
    protected $method;
    protected $arguments;
    protected $returnValue;
    protected $isProcessed = false;

    public function __construct($subject, $method, $arguments)
    {
        $this->subject = $subject;
        $this->method = $method;
        $this->arguments = $arguments;
    }

    public function getSubject()
    {
        return $this->subject;
    }

    public function getMethod()
    {
        return $this->method;
    }

    public function getArguments()
    {
        return $this->arguments;
    }

    /**
     * Define o valor de retorno e pára a notificação para outros listeners
     */
    public function setReturnValue($val)
    {
        $this->returnValue = $val;
        $this->isProcessed = true;
        $this->stopPropagation();
    }

    public function getReturnValue($val)
    {
        return $this->returnValue;
    }

    public function isProcessed()
    {
        return $this->isProcessed;
    }
}

Em seguida, crie uma classe que vai ouvir o evento foo.method_is_not_found e adicionar o método bar():

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
class Bar
{
    public function onFooMethodIsNotFound(HandleUndefinedMethodEvent $event)
    {
        // queremos somente responder as chamadas do método 'bar'
        if ('bar' != $event->getMethod()) {
            // permite que outro listener cuide deste método desconhecido
            return;
        }

        // o objeto (a instância foo)
        $foo = $event->getSubject();

        // os argumentos do método bar
        $arguments = $event->getArguments();

        // faz algo
        // ...

        // define o valor de retorno
        $event->setReturnValue($someValue);
    }
}

Por fim, adicione o novo método bar na classe Foo para registrar uma instância de Bar com o evento foo.method_is_not_found:

1
2
$bar = new Bar();
$dispatcher->addListener('foo.method_is_not_found', $bar);