LaravelVisitor is a Visitor Design Pattern implementation for Laravel. It allows to easily execute processing of collections of arbitrary elements, without requiring to use repeated conditionals, thus improving code abstraction.
Without Visitor:
public function process()
{
$result = '';
foreach ($this->elements as $element) {
if ($element instanceof FooClass) {
$result .= ((FooClass)$element)->getData();
} elseif ($element instanceof BarClass) {
$result .= ((BarClass)$element)->getData();
} elseif ($element instanceof BazClass) {
$result .= ((BazClass)$element)->getData();
}
}
}
With Visitor:
public function process()
{
$visitor = new MyVisitor([
new FooClass(),
new BarClass(),
new BazClass(),
]);
$visitor->execute();
$result = $visitor->getResult();
}
All of the complexity is hidden in the MyVisitor
class, which must define methods for processing classes. In the previous example, MyVisitor
would be implemented as:
class MyVisitor extends Visitor
{
private $result;
public function getResult()
{
return $this->result;
}
public function visitFooClass(FooClass $fooClass)
{
$this->result .= ... ;
}
public function visitBarClass(BarClass $fooClass)
{
$this->result .= ... ;
}
public function visitBazClass(BazClass $fooClass)
{
$this->result .= ... ;
}
}
Additionally this enforces SRP principle, since Domain Objects don't have to implement representational methods, which are only responsibility of the Visitor classes implementation (especially if several are required).
Install the package via composer:
composer require robertogallea/laravel-visitor
For using the package, you need to define at least one Visitor
and some Visitee
classes.
The only requirement for Visitee
s is to use the Visitable
trait, so you can make any class visitable.
A Visitor
class must impelemnt the CanVisit
interface and subclass the Visitor
abstract class, by defining the getResult()
method.
Additionally, for each defined Visitee
's you have to implement a processing method of your choice. For example, if you have a Book
Visitee, you must define the method:
public function visitBook(Book $book) {
...
}
To generate Visitor
, you can launch the following artisan commands:
php artisan make:visitor MyVisitor
which by default creates classes in the Visitors
folder.
Magazine.php
use robertogallea\LaravelVisitor\Models\Visitable;
class Magazine
{
use Visitable;
private $title;
private $month;
private $year;
public function __construct($title, $month, $year)
{
$this->title = $title;
$this->month = $month;
$this->year = $year;
}
public function getTitle()
{
return $this->title;
}
public function getMonth()
{
return $this->month;
}
public function getYear()
{
return $this->year;
}
}
XMLVisitor.php
use robertogallea\LaravelVisitor\Models\Visitor;
class XMLVisitor extends Visitor
{
private $xml = '';
public function visitMagazine(Magazine $magazine)
{
$this->xml .= '<magazine title="' . $magazine->getTitle() . '" ' .
'issue="' . $magazine->getMonth() . ' ' . $magazine->getYear() . '"></magazine>' . PHP_EOL;
}
public function getResult()
{
return $this->xml;
}
}
$xmlCatalog = new XMLVisitor([
new Magazine('PHP programming', 'July', 2019)
new Magazine('The art of woodworking', 'August', 2019)
]);
$xmlCatalog->execute();
echo($xmlCatalog->getResult());
will produce the following output:
<magazine title="PHP programming" issue="July 2019"></magazine>
<magazine title="The art of woodworking" issue="August 2019"></magazine>
You can report issues and ask questions in the issues section. Please start your issue with ISSUE:
and your question with QUESTION:
If you have a question, check the closed issues first.
To submit a Pull Request, please fork this repository, create a new branch and commit your new/updated code in there. Then open a Pull Request from your new branch. Refer to this guide for more info.