addPsr4("mindplay\\demo\\", __DIR__);
$auto_loader->register();
## Configure the cache-path. The static `Annotations` class will configure any public
## properties of `AnnotationManager` when it creates it. The `AnnotationManager::$cachePath`
## property is a path to a writable folder, where the `AnnotationManager` caches parsed
## annotations from individual source code files.
Annotations::$config['cache'] = new AnnotationCache(__DIR__ . '/runtime');
## Register demo annotations.
Package::register(Annotations::getManager());
## For this example, we're going to generate a simple form that allows us to edit a `Person`
## object. We'll define a few public properties and annotate them with some useful metadata,
## which will enable us to make decisions (at run-time) about how to display each field,
## how to parse the values posted back from the form, and how to validate the input.
##
## Note the use of standard PHP-DOC annotations, such as `@var string` - this metadata is
## traditionally useful both as documentation to developers, and as hints for an IDE. In
## this example, we're going to use that same information as advice to our components, at
## run-time, to help them establish defaults and make sensible decisions about how to
## handle the value of each property.
class Person
{
/**
* @var string
* @required
* @length(50)
* @text('label' => 'Full Name')
*/
public $name;
/**
* @var string
* @length(50)
* @text('label' => 'Street Address')
*/
public $address;
/**
* @var int
* @range(0, 100)
*/
public $age;
}
## To build a simple form abstraction that can manage the state of an object being edited,
## we start with a simple, abstract base class for input widgets.
abstract class Widget
{
protected $object;
protected $property;
public $value;
## Each widget will maintain a list of error messages.
public $errors = array();
## A widget needs to know which property of what object is being edited.
public function __construct($object, $property)
{
$this->object = $object;
$this->property = $property;
$this->value = $object->$property;
}
## Widget classes will use this method to add an error-message.
public function addError($message)
{
$this->errors[] = $message;
}
## This helper function provides a shortcut to get a named property from a
## particular type of annotation - if no annotation is found, the `$default`
## value is returned instead.
protected function getMetadata($type, $name, $default = null)
{
$a = Annotations::ofProperty($this->object, $this->property, $type);
if (!count($a)) {
return $default;
}
return $a[0]->$name;
}
## Each type of widget will need to implement this interface, which takes a raw
## POST value from the form, and attempts to bind it to the object's property.
abstract public function update($input);
## After a widget successfully updates a property, we may need to perform additional
## validation - this method will perform some basic validations, and if errors are
## found, it will add them to the `$errors` collection.
public function validate()
{
if (empty($this->value)) {
if ($this->isRequired()) {
$this->addError("Please complete this field");
} else {
return;
}
}
if (is_string($this->value)) {
$min = $this->getMetadata('@length', 'min');
$max = $this->getMetadata('@length', 'max');
if ($min !== null && strlen($this->value) < $min) {
$this->addError("Minimum length is {$min} characters");
} else {
if ($max !== null && strlen($this->value) > $max) {
$this->addError("Maximum length is {$max} characters");
}
}
}
if (is_int($this->value)) {
$min = $this->getMetadata('@range', 'min');
$max = $this->getMetadata('@range', 'max');
if (($min !== null && $this->value < $min) || ($max !== null && $this->value > $max)) {
$this->addError("Please enter a value in the range {$min} to {$max}");
}
}
}
## Each type of widget will need to implement this interface, which renders an
## HTML input representing the widget's current value.
abstract public function display();
## This helper function returns a descriptive label for the input.
public function getLabel()
{
return $this->getMetadata('@text', 'label', ucfirst($this->property));
}
## Finally, this little helper function will tell us if the field is required -
## if a property is annotated with `@required`, the field must be filled in.
public function isRequired()
{
return count(Annotations::ofProperty($this->object, $this->property, '@required')) > 0;
}
}
## The first and most basic kind of widget, is this simple string widget.
class StringWidget extends Widget
{
## On update, take into account the min/max string length, and provide error
## messages if the constraints are violated.
public function update($input)
{
$this->value = $input;
$this->validate();
}
## On display, render out a simple `` field, taking into account
## the maximum string-length.
public function display()
{
$length = $this->getMetadata('@length', 'max', 255);
echo '';
}
}
## For the age input, we'll need a specialized `StringWidget` that also checks the input type.
class IntWidget extends StringWidget
{
## On update, take into account the min/max numerical range, and provide error
## messages if the constraints are violated.
public function update($input)
{
if (strval(intval($input)) === $input) {
$this->value = intval($input);
$this->validate();
} else {
$this->value = $input;
if (!empty($input)) {
$this->addError("Please enter a whole number value");
}
}
}
}
## Next, we can build a simple form abstraction - this will hold and object and manage
## the widgets required to edit the object.
class Form
{
private $object;
/**
* Widget list.
*
* @var Widget[]
*/
private $widgets = array();
## The constructor just needs to know which object we're editing.
##
## Using reflection, we enumerate the properties of the object's type, and using the
## `@var` annotation, we decide which type of widget we're going to use.
public function __construct($object)
{
$this->object = $object;
$class = new \ReflectionClass($this->object);
foreach ($class->getProperties() as $property) {
$type = $this->getMetadata($property->name, '@var', 'type', 'string');
$wtype = 'mindplay\\demo\\' . ucfirst($type) . 'Widget';
$this->widgets[$property->name] = new $wtype($this->object, $property->name);
}
}
## This helper-method is similar to the one we defined for the widget base
## class, but fetches annotations for the specified property.
private function getMetadata($property, $type, $name, $default = null)
{
$a = Annotations::ofProperty(get_class($this->object), $property, $type);
if (!count($a)) {
return $default;
}
return $a[0]->$name;
}
## When you post information back to the form, we'll need to update it's state,
## validate each of the fields, and return a value indicating whether the form
## update was successful.
public function update($post)
{
$data = $post[get_class($this->object)];
foreach ($this->widgets as $property => $widget) {
if (array_key_exists($property, $data)) {
$this->widgets[$property]->update($data[$property]);
}
}
$valid = true;
foreach ($this->widgets as $widget) {
$valid = $valid && (count($widget->errors) === 0);
}
if ($valid) {
foreach ($this->widgets as $property => $widget) {
$this->object->$property = $widget->value;
}
}
return $valid;
}
## Finally, this method renders out the form, and each of the widgets inside, with
## a `