merged branch bschussek/issue3293 (PR #3315)
Commits -------da2447e
[Form] Fixed MergeCollectionListener when used with a custom property pathb56502f0
[Form] Added getParent() to PropertyPath7e5104e
[Form] Fixed MergeCollectionListener for the case that the form's data is updated by the data mapper (as happening in CollectionType) Discussion ---------- [Form] Fixed MergeCollectionListener for the case that the form's data is updated by the data mapper Bug fix: yes Feature addition: yes Backwards compatibility break: no Symfony2 tests pass: yes Fixes the following tickets: #3293, #1499 Todo: - ![Travis Build Status](https://secure.travis-ci.org/bschussek/symfony.png?branch=issue3293) This fixes CollectionType to properly use adders and removers. Apart from that, adders and removers now work with custom property paths. PropertyPath was extended for a method `getParent`. --------------------------------------------------------------------------- by bschussek at 2012-02-10T07:42:13Z @fabpot Ready to merge.
This commit is contained in:
commit
78e6a2f734
@ -13,16 +13,33 @@ namespace Symfony\Component\Form\Extension\Core\EventListener;
|
|||||||
|
|
||||||
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
|
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
|
||||||
use Symfony\Component\Form\FormEvents;
|
use Symfony\Component\Form\FormEvents;
|
||||||
|
use Symfony\Component\Form\Event\DataEvent;
|
||||||
use Symfony\Component\Form\Event\FilterDataEvent;
|
use Symfony\Component\Form\Event\FilterDataEvent;
|
||||||
use Symfony\Component\Form\Exception\UnexpectedTypeException;
|
use Symfony\Component\Form\Exception\UnexpectedTypeException;
|
||||||
use Symfony\Component\Form\Exception\FormException;
|
use Symfony\Component\Form\Exception\FormException;
|
||||||
use Symfony\Component\Form\Util\FormUtil;
|
use Symfony\Component\Form\Util\FormUtil;
|
||||||
|
use Symfony\Component\Form\Util\PropertyPath;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author Bernhard Schussek <bschussek@gmail.com>
|
* @author Bernhard Schussek <bschussek@gmail.com>
|
||||||
*/
|
*/
|
||||||
class MergeCollectionListener implements EventSubscriberInterface
|
class MergeCollectionListener implements EventSubscriberInterface
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
|
* Strategy for merging the new collection into the old collection
|
||||||
|
*
|
||||||
|
* @var integer
|
||||||
|
*/
|
||||||
|
const MERGE_NORMAL = 1;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Strategy for calling add/remove methods on the parent data for all
|
||||||
|
* new/removed elements in the new collection
|
||||||
|
*
|
||||||
|
* @var integer
|
||||||
|
*/
|
||||||
|
const MERGE_INTO_PARENT = 2;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Whether elements may be added to the collection
|
* Whether elements may be added to the collection
|
||||||
* @var Boolean
|
* @var Boolean
|
||||||
@ -39,7 +56,7 @@ class MergeCollectionListener implements EventSubscriberInterface
|
|||||||
* Whether to search for and use adder and remover methods
|
* Whether to search for and use adder and remover methods
|
||||||
* @var Boolean
|
* @var Boolean
|
||||||
*/
|
*/
|
||||||
private $useAccessors;
|
private $mergeStrategy;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The name of the adder method to look for
|
* The name of the adder method to look for
|
||||||
@ -53,35 +70,112 @@ class MergeCollectionListener implements EventSubscriberInterface
|
|||||||
*/
|
*/
|
||||||
private $removeMethod;
|
private $removeMethod;
|
||||||
|
|
||||||
public function __construct($allowAdd = false, $allowDelete = false, $useAccessors = true, $addMethod = null, $removeMethod = null)
|
/**
|
||||||
|
* A copy of the data before starting binding for this form
|
||||||
|
* @var mixed
|
||||||
|
*/
|
||||||
|
private $dataSnapshot;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new listener.
|
||||||
|
*
|
||||||
|
* @param Boolean $allowAdd Whether values might be added to the
|
||||||
|
* collection.
|
||||||
|
* @param Boolean $allowDelete Whether values might be removed from the
|
||||||
|
* collection.
|
||||||
|
* @param integer $mergeStrategy Which strategy to use for merging the
|
||||||
|
* bound collection with the original
|
||||||
|
* collection. Might be any combination of
|
||||||
|
* MERGE_NORMAL and MERGE_INTO_PARENT.
|
||||||
|
* MERGE_INTO_PARENT has precedence over
|
||||||
|
* MERGE_NORMAL if an adder/remover method
|
||||||
|
* is found. The default strategy is to use
|
||||||
|
* both strategies.
|
||||||
|
* @param string $addMethod The name of the adder method to use. If
|
||||||
|
* not given, the listener tries to discover
|
||||||
|
* the method automatically.
|
||||||
|
* @param string $removeMethod The name of the remover method to use. If
|
||||||
|
* not given, the listener tries to discover
|
||||||
|
* the method automatically.
|
||||||
|
*
|
||||||
|
* @throws FormException If the given strategy is invalid.
|
||||||
|
*/
|
||||||
|
public function __construct($allowAdd = false, $allowDelete = false, $mergeStrategy = null, $addMethod = null, $removeMethod = null)
|
||||||
{
|
{
|
||||||
|
if ($mergeStrategy && !($mergeStrategy & (self::MERGE_NORMAL | self::MERGE_INTO_PARENT))) {
|
||||||
|
throw new FormException('The merge strategy needs to be at least MERGE_NORMAL or MERGE_INTO_PARENT');
|
||||||
|
}
|
||||||
|
|
||||||
$this->allowAdd = $allowAdd;
|
$this->allowAdd = $allowAdd;
|
||||||
$this->allowDelete = $allowDelete;
|
$this->allowDelete = $allowDelete;
|
||||||
$this->useAccessors = $useAccessors;
|
$this->mergeStrategy = $mergeStrategy ?: self::MERGE_NORMAL | self::MERGE_INTO_PARENT;
|
||||||
$this->addMethod = $addMethod;
|
$this->addMethod = $addMethod;
|
||||||
$this->removeMethod = $removeMethod;
|
$this->removeMethod = $removeMethod;
|
||||||
}
|
}
|
||||||
|
|
||||||
static public function getSubscribedEvents()
|
static public function getSubscribedEvents()
|
||||||
{
|
{
|
||||||
return array(FormEvents::BIND_NORM_DATA => 'onBindNormData');
|
return array(
|
||||||
|
FormEvents::PRE_BIND => 'preBind',
|
||||||
|
FormEvents::BIND_NORM_DATA => 'onBindNormData',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function preBind(DataEvent $event)
|
||||||
|
{
|
||||||
|
// Get a snapshot of the current state of the normalized data
|
||||||
|
// to compare against later
|
||||||
|
$this->dataSnapshot = $event->getForm()->getNormData();
|
||||||
|
|
||||||
|
if (is_object($this->dataSnapshot)) {
|
||||||
|
// Make sure the snapshot remains stable and doesn't change
|
||||||
|
$this->dataSnapshot = clone $this->dataSnapshot;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (null !== $this->dataSnapshot && !is_array($this->dataSnapshot) && !($this->dataSnapshot instanceof \Traversable && $this->dataSnapshot instanceof \ArrayAccess)) {
|
||||||
|
throw new UnexpectedTypeException($this->dataSnapshot, 'array or (\Traversable and \ArrayAccess)');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function onBindNormData(FilterDataEvent $event)
|
public function onBindNormData(FilterDataEvent $event)
|
||||||
{
|
{
|
||||||
$originalData = $event->getForm()->getData();
|
$originalData = $event->getForm()->getNormData();
|
||||||
|
|
||||||
// If we are not allowed to change anything, return immediately
|
// If we are not allowed to change anything, return immediately
|
||||||
if (!$this->allowAdd && !$this->allowDelete) {
|
if (!$this->allowAdd && !$this->allowDelete) {
|
||||||
|
// Don't set to the snapshot as then we are switching from the
|
||||||
|
// original object to its copy, which might break things
|
||||||
$event->setData($originalData);
|
$event->setData($originalData);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$form = $event->getForm();
|
$form = $event->getForm();
|
||||||
$data = $event->getData();
|
$data = $event->getData();
|
||||||
$parentData = $form->hasParent() ? $form->getParent()->getData() : null;
|
$childPropertyPath = null;
|
||||||
|
$parentData = null;
|
||||||
$addMethod = null;
|
$addMethod = null;
|
||||||
$removeMethod = null;
|
$removeMethod = null;
|
||||||
|
$propertyPath = null;
|
||||||
|
$plural = null;
|
||||||
|
|
||||||
|
if ($form->hasParent() && $form->getAttribute('property_path')) {
|
||||||
|
$propertyPath = new PropertyPath($form->getAttribute('property_path'));
|
||||||
|
$childPropertyPath = $propertyPath;
|
||||||
|
$parentData = $form->getParent()->getClientData();
|
||||||
|
$lastElement = $propertyPath->getElement($propertyPath->getLength() - 1);
|
||||||
|
|
||||||
|
// If the property path contains more than one element, the parent
|
||||||
|
// data is the object at the parent property path
|
||||||
|
if ($propertyPath->getLength() > 1) {
|
||||||
|
$parentData = $propertyPath->getParent()->getValue($parentData);
|
||||||
|
|
||||||
|
// Property path relative to $parentData
|
||||||
|
$childPropertyPath = new PropertyPath($lastElement);
|
||||||
|
}
|
||||||
|
|
||||||
|
// The plural form is the last element of the property path
|
||||||
|
$plural = ucfirst($lastElement);
|
||||||
|
}
|
||||||
|
|
||||||
if (null === $data) {
|
if (null === $data) {
|
||||||
$data = array();
|
$data = array();
|
||||||
@ -96,24 +190,34 @@ class MergeCollectionListener implements EventSubscriberInterface
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Check if the parent has matching methods to add/remove items
|
// Check if the parent has matching methods to add/remove items
|
||||||
if ($this->useAccessors && is_object($parentData)) {
|
if (($this->mergeStrategy & self::MERGE_INTO_PARENT) && is_object($parentData)) {
|
||||||
$reflClass = new \ReflectionClass($parentData);
|
$reflClass = new \ReflectionClass($parentData);
|
||||||
$addMethodNeeded = $this->allowAdd && !$this->addMethod;
|
$addMethodNeeded = $this->allowAdd && !$this->addMethod;
|
||||||
$removeMethodNeeded = $this->allowDelete && !$this->removeMethod;
|
$removeMethodNeeded = $this->allowDelete && !$this->removeMethod;
|
||||||
|
|
||||||
// Any of the two methods is required, but not yet known
|
// Any of the two methods is required, but not yet known
|
||||||
if ($addMethodNeeded || $removeMethodNeeded) {
|
if ($addMethodNeeded || $removeMethodNeeded) {
|
||||||
$singulars = (array) FormUtil::singularify(ucfirst($form->getName()));
|
$singulars = (array) FormUtil::singularify($plural);
|
||||||
|
|
||||||
foreach ($singulars as $singular) {
|
foreach ($singulars as $singular) {
|
||||||
// Try to find adder, but don't override preconfigured one
|
// Try to find adder, but don't override preconfigured one
|
||||||
if ($addMethodNeeded) {
|
if ($addMethodNeeded) {
|
||||||
$addMethod = $this->checkMethod($reflClass, 'add' . $singular);
|
$addMethod = 'add' . $singular;
|
||||||
|
|
||||||
|
// False alert
|
||||||
|
if (!$this->isAccessible($reflClass, $addMethod, 1)) {
|
||||||
|
$addMethod = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try to find remover, but don't override preconfigured one
|
// Try to find remover, but don't override preconfigured one
|
||||||
if ($removeMethodNeeded) {
|
if ($removeMethodNeeded) {
|
||||||
$removeMethod = $this->checkMethod($reflClass, 'remove' . $singular);
|
$removeMethod = 'remove' . $singular;
|
||||||
|
|
||||||
|
// False alert
|
||||||
|
if (!$this->isAccessible($reflClass, $removeMethod, 1)) {
|
||||||
|
$removeMethod = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Found all that we need. Abort search.
|
// Found all that we need. Abort search.
|
||||||
@ -129,12 +233,12 @@ class MergeCollectionListener implements EventSubscriberInterface
|
|||||||
|
|
||||||
// Set preconfigured adder
|
// Set preconfigured adder
|
||||||
if ($this->allowAdd && $this->addMethod) {
|
if ($this->allowAdd && $this->addMethod) {
|
||||||
$addMethod = $this->checkMethod($reflClass, $this->addMethod);
|
$addMethod = $this->addMethod;
|
||||||
|
|
||||||
if (!$addMethod) {
|
if (!$this->isAccessible($reflClass, $addMethod, 1)) {
|
||||||
throw new FormException(sprintf(
|
throw new FormException(sprintf(
|
||||||
'The method "%s" could not be found on class %s',
|
'The public method "%s" could not be found on class %s',
|
||||||
$this->addMethod,
|
$addMethod,
|
||||||
$reflClass->getName()
|
$reflClass->getName()
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
@ -142,25 +246,24 @@ class MergeCollectionListener implements EventSubscriberInterface
|
|||||||
|
|
||||||
// Set preconfigured remover
|
// Set preconfigured remover
|
||||||
if ($this->allowDelete && $this->removeMethod) {
|
if ($this->allowDelete && $this->removeMethod) {
|
||||||
$removeMethod = $this->checkMethod($reflClass, $this->removeMethod);
|
$removeMethod = $this->removeMethod;
|
||||||
|
|
||||||
if (!$removeMethod) {
|
if (!$this->isAccessible($reflClass, $removeMethod, 1)) {
|
||||||
throw new FormException(sprintf(
|
throw new FormException(sprintf(
|
||||||
'The method "%s" could not be found on class %s',
|
'The public method "%s" could not be found on class %s',
|
||||||
$this->removeMethod,
|
$removeMethod,
|
||||||
$reflClass->getName()
|
$reflClass->getName()
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check which items are in $data that are not in $originalData and
|
// Calculate delta between $data and the snapshot created in PRE_BIND
|
||||||
// vice versa
|
|
||||||
$itemsToDelete = array();
|
$itemsToDelete = array();
|
||||||
$itemsToAdd = is_object($data) ? clone $data : $data;
|
$itemsToAdd = is_object($data) ? clone $data : $data;
|
||||||
|
|
||||||
if ($originalData) {
|
if ($this->dataSnapshot) {
|
||||||
foreach ($originalData as $originalKey => $originalItem) {
|
foreach ($this->dataSnapshot as $originalItem) {
|
||||||
foreach ($data as $key => $item) {
|
foreach ($data as $key => $item) {
|
||||||
if ($item === $originalItem) {
|
if ($item === $originalItem) {
|
||||||
// Item found, next original item
|
// Item found, next original item
|
||||||
@ -170,7 +273,12 @@ class MergeCollectionListener implements EventSubscriberInterface
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Item not found, remember for deletion
|
// Item not found, remember for deletion
|
||||||
$itemsToDelete[$originalKey] = $originalItem;
|
foreach ($originalData as $key => $item) {
|
||||||
|
if ($item === $originalItem) {
|
||||||
|
$itemsToDelete[$key] = $item;
|
||||||
|
continue 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -187,43 +295,47 @@ class MergeCollectionListener implements EventSubscriberInterface
|
|||||||
$parentData->$addMethod($item);
|
$parentData->$addMethod($item);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} elseif (!$originalData) {
|
|
||||||
// No original data was set. Set it if allowed
|
|
||||||
if ($this->allowAdd) {
|
|
||||||
$originalData = $data;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Original data is an array-like structure
|
|
||||||
// Add and remove items in the original variable
|
|
||||||
if ($this->allowDelete) {
|
|
||||||
foreach ($itemsToDelete as $key => $item) {
|
|
||||||
unset($originalData[$key]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($this->allowAdd) {
|
$event->setData($childPropertyPath->getValue($parentData));
|
||||||
foreach ($itemsToAdd as $key => $item) {
|
} elseif ($this->mergeStrategy & self::MERGE_NORMAL) {
|
||||||
if (!isset($originalData[$key])) {
|
if (!$originalData) {
|
||||||
$originalData[$key] = $item;
|
// No original data was set. Set it if allowed
|
||||||
} else {
|
if ($this->allowAdd) {
|
||||||
$originalData[] = $item;
|
$originalData = $data;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Original data is an array-like structure
|
||||||
|
// Add and remove items in the original variable
|
||||||
|
if ($this->allowDelete) {
|
||||||
|
foreach ($itemsToDelete as $key => $item) {
|
||||||
|
unset($originalData[$key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->allowAdd) {
|
||||||
|
foreach ($itemsToAdd as $key => $item) {
|
||||||
|
if (!isset($originalData[$key])) {
|
||||||
|
$originalData[$key] = $item;
|
||||||
|
} else {
|
||||||
|
$originalData[] = $item;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
$event->setData($originalData);
|
$event->setData($originalData);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private function checkMethod(\ReflectionClass $reflClass, $methodName) {
|
private function isAccessible(\ReflectionClass $reflClass, $methodName, $numberOfRequiredParameters) {
|
||||||
if ($reflClass->hasMethod($methodName)) {
|
if ($reflClass->hasMethod($methodName)) {
|
||||||
$method = $reflClass->getMethod($methodName);
|
$method = $reflClass->getMethod($methodName);
|
||||||
|
|
||||||
if ($method->isPublic() && $method->getNumberOfRequiredParameters() === 1) {
|
if ($method->isPublic() && $method->getNumberOfRequiredParameters() === $numberOfRequiredParameters) {
|
||||||
return $methodName;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return null;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -81,10 +81,7 @@ class ChoiceType extends AbstractType
|
|||||||
|
|
||||||
if ($options['expanded']) {
|
if ($options['expanded']) {
|
||||||
if ($options['multiple']) {
|
if ($options['multiple']) {
|
||||||
$builder
|
$builder->appendClientTransformer(new ChoicesToBooleanArrayTransformer($options['choice_list']));
|
||||||
->appendClientTransformer(new ChoicesToBooleanArrayTransformer($options['choice_list']))
|
|
||||||
->addEventSubscriber(new MergeCollectionListener(true, true))
|
|
||||||
;
|
|
||||||
} else {
|
} else {
|
||||||
$builder
|
$builder
|
||||||
->appendClientTransformer(new ChoiceToBooleanArrayTransformer($options['choice_list']))
|
->appendClientTransformer(new ChoiceToBooleanArrayTransformer($options['choice_list']))
|
||||||
@ -93,24 +90,31 @@ class ChoiceType extends AbstractType
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if ($options['multiple']) {
|
if ($options['multiple']) {
|
||||||
$builder
|
$builder->appendClientTransformer(new ChoicesToValuesTransformer($options['choice_list']));
|
||||||
->appendClientTransformer(new ChoicesToValuesTransformer($options['choice_list']))
|
|
||||||
->addEventSubscriber(new MergeCollectionListener(
|
|
||||||
true,
|
|
||||||
true,
|
|
||||||
// If "by_reference" is disabled (explicit calling of
|
|
||||||
// the setter is desired), disable support for
|
|
||||||
// adders/removers
|
|
||||||
// Same as in CollectionType
|
|
||||||
$options['by_reference'],
|
|
||||||
$options['add_method'],
|
|
||||||
$options['remove_method']
|
|
||||||
))
|
|
||||||
;
|
|
||||||
} else {
|
} else {
|
||||||
$builder->appendClientTransformer(new ChoiceToValueTransformer($options['choice_list']));
|
$builder->appendClientTransformer(new ChoiceToValueTransformer($options['choice_list']));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($options['multiple']) {
|
||||||
|
// Make sure the collection created during the client->norm
|
||||||
|
// transformation is merged back into the original collection
|
||||||
|
$mergeStrategy = MergeCollectionListener::MERGE_NORMAL;
|
||||||
|
|
||||||
|
// Enable support for adders/removers unless "by_reference" is disabled
|
||||||
|
// (explicit calling of the setter is desired)
|
||||||
|
if ($options['by_reference']) {
|
||||||
|
$mergeStrategy = $mergeStrategy | MergeCollectionListener::MERGE_INTO_PARENT;
|
||||||
|
}
|
||||||
|
|
||||||
|
$builder->addEventSubscriber(new MergeCollectionListener(
|
||||||
|
true,
|
||||||
|
true,
|
||||||
|
$mergeStrategy,
|
||||||
|
$options['add_method'],
|
||||||
|
$options['remove_method']
|
||||||
|
));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -38,23 +38,23 @@ class CollectionType extends AbstractType
|
|||||||
$options['allow_delete']
|
$options['allow_delete']
|
||||||
);
|
);
|
||||||
|
|
||||||
$mergeListener = new MergeCollectionListener(
|
|
||||||
$options['allow_add'],
|
|
||||||
$options['allow_delete'],
|
|
||||||
// If "by_reference" is disabled (explicit calling of the setter
|
|
||||||
// is desired), disable support for adders/removers
|
|
||||||
// Same as in ChoiceType
|
|
||||||
$options['by_reference'],
|
|
||||||
$options['add_method'],
|
|
||||||
$options['remove_method']
|
|
||||||
);
|
|
||||||
|
|
||||||
$builder
|
$builder
|
||||||
->addEventSubscriber($resizeListener)
|
->addEventSubscriber($resizeListener)
|
||||||
->addEventSubscriber($mergeListener)
|
|
||||||
->setAttribute('allow_add', $options['allow_add'])
|
->setAttribute('allow_add', $options['allow_add'])
|
||||||
->setAttribute('allow_delete', $options['allow_delete'])
|
->setAttribute('allow_delete', $options['allow_delete'])
|
||||||
;
|
;
|
||||||
|
|
||||||
|
// Enable support for adders/removers unless "by_reference" is disabled
|
||||||
|
// (explicit calling of the setter is desired)
|
||||||
|
if ($options['by_reference']) {
|
||||||
|
$builder->addEventSubscriber(new MergeCollectionListener(
|
||||||
|
$options['allow_add'],
|
||||||
|
$options['allow_delete'],
|
||||||
|
MergeCollectionListener::MERGE_INTO_PARENT,
|
||||||
|
$options['add_method'],
|
||||||
|
$options['remove_method']
|
||||||
|
));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -27,26 +27,32 @@ class PropertyPath implements \IteratorAggregate
|
|||||||
* The elements of the property path
|
* The elements of the property path
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
protected $elements = array();
|
private $elements = array();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The number of elements in the property path
|
* The number of elements in the property path
|
||||||
* @var integer
|
* @var integer
|
||||||
*/
|
*/
|
||||||
protected $length;
|
private $length;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Contains a Boolean for each property in $elements denoting whether this
|
* Contains a Boolean for each property in $elements denoting whether this
|
||||||
* element is an index. It is a property otherwise.
|
* element is an index. It is a property otherwise.
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
protected $isIndex = array();
|
private $isIndex = array();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* String representation of the path
|
* String representation of the path
|
||||||
* @var string
|
* @var string
|
||||||
*/
|
*/
|
||||||
protected $string;
|
private $string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Positions where the individual elements start in the string representation
|
||||||
|
* @var array
|
||||||
|
*/
|
||||||
|
private $positions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Parses the given property path
|
* Parses the given property path
|
||||||
@ -67,6 +73,8 @@ class PropertyPath implements \IteratorAggregate
|
|||||||
$pattern = '/^(([^\.\[]+)|\[([^\]]+)\])(.*)/';
|
$pattern = '/^(([^\.\[]+)|\[([^\]]+)\])(.*)/';
|
||||||
|
|
||||||
while (preg_match($pattern, $remaining, $matches)) {
|
while (preg_match($pattern, $remaining, $matches)) {
|
||||||
|
$this->positions[] = $position;
|
||||||
|
|
||||||
if ('' !== $matches[2]) {
|
if ('' !== $matches[2]) {
|
||||||
$this->elements[] = $matches[2];
|
$this->elements[] = $matches[2];
|
||||||
$this->isIndex[] = false;
|
$this->isIndex[] = false;
|
||||||
@ -102,6 +110,43 @@ class PropertyPath implements \IteratorAggregate
|
|||||||
return $this->string;
|
return $this->string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the length of the property path.
|
||||||
|
*
|
||||||
|
* @return integer
|
||||||
|
*/
|
||||||
|
public function getLength()
|
||||||
|
{
|
||||||
|
return $this->length;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the parent property path.
|
||||||
|
*
|
||||||
|
* The parent property path is the one that contains the same items as
|
||||||
|
* this one except for the last one.
|
||||||
|
*
|
||||||
|
* If this property path only contains one item, null is returned.
|
||||||
|
*
|
||||||
|
* @return PropertyPath The parent path or null.
|
||||||
|
*/
|
||||||
|
public function getParent()
|
||||||
|
{
|
||||||
|
if ($this->length <= 1) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$parent = clone $this;
|
||||||
|
|
||||||
|
--$parent->length;
|
||||||
|
$parent->string = substr($parent->string, 0, $parent->positions[$parent->length]);
|
||||||
|
array_pop($parent->elements);
|
||||||
|
array_pop($parent->isIndex);
|
||||||
|
array_pop($parent->positions);
|
||||||
|
|
||||||
|
return $parent;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a new iterator for this path
|
* Returns a new iterator for this path
|
||||||
*
|
*
|
||||||
|
@ -22,6 +22,8 @@ class MergeCollectionListenerTest_Car
|
|||||||
public function addAxis($axis) {}
|
public function addAxis($axis) {}
|
||||||
|
|
||||||
public function removeAxis($axis) {}
|
public function removeAxis($axis) {}
|
||||||
|
|
||||||
|
public function getAxes() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
class MergeCollectionListenerTest_CarCustomNames
|
class MergeCollectionListenerTest_CarCustomNames
|
||||||
@ -29,16 +31,36 @@ class MergeCollectionListenerTest_CarCustomNames
|
|||||||
public function foo($axis) {}
|
public function foo($axis) {}
|
||||||
|
|
||||||
public function bar($axis) {}
|
public function bar($axis) {}
|
||||||
|
|
||||||
|
public function getAxes() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
class MergeCollectionListenerTest_CarOnlyAdder
|
class MergeCollectionListenerTest_CarOnlyAdder
|
||||||
{
|
{
|
||||||
public function addAxis($axis) {}
|
public function addAxis($axis) {}
|
||||||
|
|
||||||
|
public function getAxes() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
class MergeCollectionListenerTest_CarOnlyRemover
|
class MergeCollectionListenerTest_CarOnlyRemover
|
||||||
{
|
{
|
||||||
public function removeAxis($axis) {}
|
public function removeAxis($axis) {}
|
||||||
|
|
||||||
|
public function getAxes() {}
|
||||||
|
}
|
||||||
|
|
||||||
|
class MergeCollectionListenerTest_CompositeCar
|
||||||
|
{
|
||||||
|
public function getStructure() {}
|
||||||
|
}
|
||||||
|
|
||||||
|
class MergeCollectionListenerTest_CarStructure
|
||||||
|
{
|
||||||
|
public function addAxis($axis) {}
|
||||||
|
|
||||||
|
public function removeAxis($axis) {}
|
||||||
|
|
||||||
|
public function getAxes() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
||||||
@ -66,9 +88,11 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
return new FormBuilder($name, $this->factory, $this->dispatcher);
|
return new FormBuilder($name, $this->factory, $this->dispatcher);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getForm($name = 'name')
|
protected function getForm($name = 'name', $propertyPath = null)
|
||||||
{
|
{
|
||||||
return $this->getBuilder($name)->getForm();
|
$propertyPath = $propertyPath ?: $name;
|
||||||
|
|
||||||
|
return $this->getBuilder($name)->setAttribute('property_path', $propertyPath)->getForm();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function getMockForm()
|
protected function getMockForm()
|
||||||
@ -76,17 +100,55 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
return $this->getMock('Symfony\Tests\Component\Form\FormInterface');
|
return $this->getMock('Symfony\Tests\Component\Form\FormInterface');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getModesWithNormal()
|
||||||
|
{
|
||||||
|
return array(
|
||||||
|
array(MergeCollectionListener::MERGE_NORMAL),
|
||||||
|
array(MergeCollectionListener::MERGE_NORMAL | MergeCollectionListener::MERGE_INTO_PARENT),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getModesWithMergeIntoParent()
|
||||||
|
{
|
||||||
|
return array(
|
||||||
|
array(MergeCollectionListener::MERGE_INTO_PARENT),
|
||||||
|
array(MergeCollectionListener::MERGE_INTO_PARENT | MergeCollectionListener::MERGE_NORMAL),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getModesWithoutMergeIntoParent()
|
||||||
|
{
|
||||||
|
return array(
|
||||||
|
array(MergeCollectionListener::MERGE_NORMAL),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getInvalidModes()
|
||||||
|
{
|
||||||
|
return array(
|
||||||
|
// 0 is a valid mode, because it is treated as "default" (=3)
|
||||||
|
array(4),
|
||||||
|
array(8),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
abstract protected function getData(array $data);
|
abstract protected function getData(array $data);
|
||||||
|
|
||||||
public function testAddExtraEntriesIfAllowAdd()
|
/**
|
||||||
|
* @dataProvider getModesWithNormal
|
||||||
|
*/
|
||||||
|
public function testAddExtraEntriesIfAllowAdd($mode)
|
||||||
{
|
{
|
||||||
$originalData = $this->getData(array(1 => 'second'));
|
$originalData = $this->getData(array(1 => 'second'));
|
||||||
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(true, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
// The original object was modified
|
// The original object was modified
|
||||||
@ -98,15 +160,21 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$this->assertEquals($newData, $event->getData());
|
$this->assertEquals($newData, $event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testAddExtraEntriesIfAllowAddDontOverwriteExistingIndices()
|
/**
|
||||||
|
* @dataProvider getModesWithNormal
|
||||||
|
*/
|
||||||
|
public function testAddExtraEntriesIfAllowAddDontOverwriteExistingIndices($mode)
|
||||||
{
|
{
|
||||||
$originalData = $this->getData(array(1 => 'first'));
|
$originalData = $this->getData(array(1 => 'first'));
|
||||||
$newData = $this->getData(array(0 => 'first', 1 => 'second'));
|
$newData = $this->getData(array(0 => 'first', 1 => 'second'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(true, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
// The original object was modified
|
// The original object was modified
|
||||||
@ -118,16 +186,22 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$this->assertEquals($this->getData(array(1 => 'first', 2 => 'second')), $event->getData());
|
$this->assertEquals($this->getData(array(1 => 'first', 2 => 'second')), $event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDoNothingIfNotAllowAdd()
|
/**
|
||||||
|
* @dataProvider getModesWithNormal
|
||||||
|
*/
|
||||||
|
public function testDoNothingIfNotAllowAdd($mode)
|
||||||
{
|
{
|
||||||
$originalDataArray = array(1 => 'second');
|
$originalDataArray = array(1 => 'second');
|
||||||
$originalData = $this->getData($originalDataArray);
|
$originalData = $this->getData($originalDataArray);
|
||||||
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
// We still have the original object
|
// We still have the original object
|
||||||
@ -139,15 +213,21 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testRemoveMissingEntriesIfAllowDelete()
|
/**
|
||||||
|
* @dataProvider getModesWithNormal
|
||||||
|
*/
|
||||||
|
public function testRemoveMissingEntriesIfAllowDelete($mode)
|
||||||
{
|
{
|
||||||
$originalData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
$originalData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
$newData = $this->getData(array(1 => 'second'));
|
$newData = $this->getData(array(1 => 'second'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, true, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, true);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
// The original object was modified
|
// The original object was modified
|
||||||
@ -159,16 +239,22 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$this->assertEquals($newData, $event->getData());
|
$this->assertEquals($newData, $event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDoNothingIfNotAllowDelete()
|
/**
|
||||||
|
* @dataProvider getModesWithNormal
|
||||||
|
*/
|
||||||
|
public function testDoNothingIfNotAllowDelete($mode)
|
||||||
{
|
{
|
||||||
$originalDataArray = array(0 => 'first', 1 => 'second', 2 => 'third');
|
$originalDataArray = array(0 => 'first', 1 => 'second', 2 => 'third');
|
||||||
$originalData = $this->getData($originalDataArray);
|
$originalData = $this->getData($originalDataArray);
|
||||||
$newData = $this->getData(array(1 => 'second'));
|
$newData = $this->getData(array(1 => 'second'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
// We still have the original object
|
// We still have the original object
|
||||||
@ -181,59 +267,81 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* @dataProvider getModesWithNormal
|
||||||
* @expectedException Symfony\Component\Form\Exception\UnexpectedTypeException
|
* @expectedException Symfony\Component\Form\Exception\UnexpectedTypeException
|
||||||
*/
|
*/
|
||||||
public function testRequireArrayOrTraversable()
|
public function testRequireArrayOrTraversable($mode)
|
||||||
{
|
{
|
||||||
$newData = 'no array or traversable';
|
$newData = 'no array or traversable';
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(true, false);
|
$listener = new MergeCollectionListener(true, false, $mode);
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDealWithNullData()
|
/**
|
||||||
|
* @dataProvider getModesWithNormal
|
||||||
|
*/
|
||||||
|
public function testDealWithNullData($mode)
|
||||||
{
|
{
|
||||||
$originalData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
$originalData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
$newData = null;
|
$newData = null;
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
$this->assertSame($originalData, $event->getData());
|
$this->assertSame($originalData, $event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDealWithNullOriginalDataIfAllowAdd()
|
/**
|
||||||
|
* @dataProvider getModesWithNormal
|
||||||
|
*/
|
||||||
|
public function testDealWithNullOriginalDataIfAllowAdd($mode)
|
||||||
{
|
{
|
||||||
$originalData = null;
|
$originalData = null;
|
||||||
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(true, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
$this->assertSame($newData, $event->getData());
|
$this->assertSame($newData, $event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDontDealWithNullOriginalDataIfNotAllowAdd()
|
/**
|
||||||
|
* @dataProvider getModesWithNormal
|
||||||
|
*/
|
||||||
|
public function testDontDealWithNullOriginalDataIfNotAllowAdd($mode)
|
||||||
{
|
{
|
||||||
$originalData = null;
|
$originalData = null;
|
||||||
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
$this->assertNull($event->getData());
|
$this->assertNull($event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testCallAdderIfAllowAdd()
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testCallAdderIfAllowAdd($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_CarOnlyAdder');
|
$parentData = $this->getMock(__CLASS__ . '_CarOnlyAdder');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -244,29 +352,122 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData($originalDataArray);
|
$originalData = $this->getData($originalDataArray);
|
||||||
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
|
|
||||||
$parentData->expects($this->at(0))
|
$parentData->expects($this->at(0))
|
||||||
->method('addAxis')
|
->method('addAxis')
|
||||||
->with('first');
|
->with('first');
|
||||||
$parentData->expects($this->at(1))
|
$parentData->expects($this->at(1))
|
||||||
->method('addAxis')
|
->method('addAxis')
|
||||||
->with('third');
|
->with('third');
|
||||||
|
$parentData->expects($this->at(2))
|
||||||
|
->method('getAxes')
|
||||||
|
->will($this->returnValue('RESULT'));
|
||||||
|
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(true, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
if (is_object($originalData)) {
|
$this->assertEquals('RESULT', $event->getData());
|
||||||
$this->assertSame($originalData, $event->getData());
|
|
||||||
}
|
|
||||||
|
|
||||||
// The data was not modified directly
|
|
||||||
// Thus it should not be written back into the parent data!
|
|
||||||
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDontCallAdderIfNotAllowAdd()
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testCallAdderIfCustomPropertyPath($mode)
|
||||||
|
{
|
||||||
|
$this->form = $this->getForm('structure_axes', 'structure.axes');
|
||||||
|
|
||||||
|
$parentData = $this->getMock(__CLASS__ . '_CompositeCar');
|
||||||
|
$parentForm = $this->getForm('car');
|
||||||
|
$parentForm->setData($parentData);
|
||||||
|
$parentForm->add($this->form);
|
||||||
|
|
||||||
|
$modifData = $this->getMock(__CLASS__ . '_CarStructure');
|
||||||
|
|
||||||
|
$originalDataArray = array(1 => 'second');
|
||||||
|
$originalData = $this->getData($originalDataArray);
|
||||||
|
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, false, $mode);
|
||||||
|
|
||||||
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
|
|
||||||
|
$parentData->expects($this->once())
|
||||||
|
->method('getStructure')
|
||||||
|
->will($this->returnValue($modifData));
|
||||||
|
|
||||||
|
$modifData->expects($this->at(0))
|
||||||
|
->method('addAxis')
|
||||||
|
->with('first');
|
||||||
|
$modifData->expects($this->at(1))
|
||||||
|
->method('addAxis')
|
||||||
|
->with('third');
|
||||||
|
$modifData->expects($this->at(2))
|
||||||
|
->method('getAxes')
|
||||||
|
->will($this->returnValue('RESULT'));
|
||||||
|
|
||||||
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
|
$this->assertEquals('RESULT', $event->getData());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testCallAdderIfOriginalDataAlreadyModified($mode)
|
||||||
|
{
|
||||||
|
$parentData = $this->getMock(__CLASS__ . '_CarOnlyAdder');
|
||||||
|
$parentForm = $this->getForm('car');
|
||||||
|
$parentForm->setData($parentData);
|
||||||
|
$parentForm->add($this->form);
|
||||||
|
|
||||||
|
$originalDataArray = array(1 => 'second');
|
||||||
|
$originalData = $this->getData($originalDataArray);
|
||||||
|
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, false, $mode);
|
||||||
|
|
||||||
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
|
|
||||||
|
// The form already contains the new data
|
||||||
|
// This happens if the data mapper maps the data of the child forms
|
||||||
|
// back into the original collection.
|
||||||
|
// The original collection is then both already modified and passed
|
||||||
|
// as event argument.
|
||||||
|
$this->form->setData($newData);
|
||||||
|
|
||||||
|
$parentData->expects($this->at(0))
|
||||||
|
->method('addAxis')
|
||||||
|
->with('first');
|
||||||
|
$parentData->expects($this->at(1))
|
||||||
|
->method('addAxis')
|
||||||
|
->with('third');
|
||||||
|
$parentData->expects($this->at(2))
|
||||||
|
->method('getAxes')
|
||||||
|
->will($this->returnValue('RESULT'));
|
||||||
|
|
||||||
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
|
$this->assertEquals('RESULT', $event->getData());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testDontCallAdderIfNotAllowAdd($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_Car');
|
$parentData = $this->getMock(__CLASS__ . '_Car');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -277,13 +478,17 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData($originalDataArray);
|
$originalData = $this->getData($originalDataArray);
|
||||||
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
|
|
||||||
$parentData->expects($this->never())
|
$parentData->expects($this->never())
|
||||||
->method('addAxis');
|
->method('addAxis');
|
||||||
|
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
if (is_object($originalData)) {
|
if (is_object($originalData)) {
|
||||||
@ -294,7 +499,10 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDontCallAdderIfNotUseAccessors()
|
/**
|
||||||
|
* @dataProvider getModesWithoutMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testDontCallAdderIfNotMergeIntoParent($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_Car');
|
$parentData = $this->getMock(__CLASS__ . '_Car');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -304,13 +512,17 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData(array(1 => 'second'));
|
$originalData = $this->getData(array(1 => 'second'));
|
||||||
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
$newData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
|
|
||||||
$parentData->expects($this->never())
|
$parentData->expects($this->never())
|
||||||
->method('addAxis');
|
->method('addAxis');
|
||||||
|
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(true, false, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
if (is_object($originalData)) {
|
if (is_object($originalData)) {
|
||||||
@ -321,7 +533,10 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$this->assertEquals($newData, $event->getData());
|
$this->assertEquals($newData, $event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testCallRemoverIfAllowDelete()
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testCallRemoverIfAllowDelete($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_CarOnlyRemover');
|
$parentData = $this->getMock(__CLASS__ . '_CarOnlyRemover');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -332,29 +547,33 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData($originalDataArray);
|
$originalData = $this->getData($originalDataArray);
|
||||||
$newData = $this->getData(array(1 => 'second'));
|
$newData = $this->getData(array(1 => 'second'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, true, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
|
|
||||||
$parentData->expects($this->at(0))
|
$parentData->expects($this->at(0))
|
||||||
->method('removeAxis')
|
->method('removeAxis')
|
||||||
->with('first');
|
->with('first');
|
||||||
$parentData->expects($this->at(1))
|
$parentData->expects($this->at(1))
|
||||||
->method('removeAxis')
|
->method('removeAxis')
|
||||||
->with('third');
|
->with('third');
|
||||||
|
$parentData->expects($this->at(2))
|
||||||
|
->method('getAxes')
|
||||||
|
->will($this->returnValue('RESULT'));
|
||||||
|
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, true);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
if (is_object($originalData)) {
|
$this->assertEquals('RESULT', $event->getData());
|
||||||
$this->assertSame($originalData, $event->getData());
|
|
||||||
}
|
|
||||||
|
|
||||||
// The data was not modified directly
|
|
||||||
// Thus it should not be written back into the parent data!
|
|
||||||
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDontCallRemoverIfNotAllowDelete()
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testDontCallRemoverIfNotAllowDelete($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_Car');
|
$parentData = $this->getMock(__CLASS__ . '_Car');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -365,13 +584,17 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData($originalDataArray);
|
$originalData = $this->getData($originalDataArray);
|
||||||
$newData = $this->getData(array(1 => 'second'));
|
$newData = $this->getData(array(1 => 'second'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, false, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
|
|
||||||
$parentData->expects($this->never())
|
$parentData->expects($this->never())
|
||||||
->method('removeAxis');
|
->method('removeAxis');
|
||||||
|
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
if (is_object($originalData)) {
|
if (is_object($originalData)) {
|
||||||
@ -382,7 +605,10 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDontCallRemoverIfNotUseAccessors()
|
/**
|
||||||
|
* @dataProvider getModesWithoutMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testDontCallRemoverIfNotMergeIntoParent($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_Car');
|
$parentData = $this->getMock(__CLASS__ . '_Car');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -392,13 +618,17 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
$originalData = $this->getData(array(0 => 'first', 1 => 'second', 2 => 'third'));
|
||||||
$newData = $this->getData(array(1 => 'second'));
|
$newData = $this->getData(array(1 => 'second'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, true, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
|
|
||||||
$parentData->expects($this->never())
|
$parentData->expects($this->never())
|
||||||
->method('removeAxis');
|
->method('removeAxis');
|
||||||
|
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, true, false);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
if (is_object($originalData)) {
|
if (is_object($originalData)) {
|
||||||
@ -409,7 +639,10 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$this->assertEquals($newData, $event->getData());
|
$this->assertEquals($newData, $event->getData());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testCallAdderAndDeleterIfAllowAll()
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testCallAdderAndDeleterIfAllowAll($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_Car');
|
$parentData = $this->getMock(__CLASS__ . '_Car');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -420,29 +653,33 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData($originalDataArray);
|
$originalData = $this->getData($originalDataArray);
|
||||||
$newData = $this->getData(array(0 => 'first'));
|
$newData = $this->getData(array(0 => 'first'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, true, $mode);
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
$parentData->expects($this->once())
|
$event = new DataEvent($this->form, $newData);
|
||||||
->method('addAxis')
|
$listener->preBind($event);
|
||||||
->with('first');
|
|
||||||
$parentData->expects($this->once())
|
$parentData->expects($this->at(0))
|
||||||
->method('removeAxis')
|
->method('removeAxis')
|
||||||
->with('second');
|
->with('second');
|
||||||
|
$parentData->expects($this->at(1))
|
||||||
|
->method('addAxis')
|
||||||
|
->with('first');
|
||||||
|
$parentData->expects($this->at(2))
|
||||||
|
->method('getAxes')
|
||||||
|
->will($this->returnValue('RESULT'));
|
||||||
|
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(true, true, true);
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
if (is_object($originalData)) {
|
$this->assertEquals('RESULT', $event->getData());
|
||||||
$this->assertSame($originalData, $event->getData());
|
|
||||||
}
|
|
||||||
|
|
||||||
// The data was not modified directly
|
|
||||||
// Thus it should not be written back into the parent data!
|
|
||||||
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testCallAccessorsWithCustomNames()
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testCallAccessorsWithCustomNames($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_CarCustomNames');
|
$parentData = $this->getMock(__CLASS__ . '_CarCustomNames');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -453,29 +690,33 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData($originalDataArray);
|
$originalData = $this->getData($originalDataArray);
|
||||||
$newData = $this->getData(array(0 => 'first'));
|
$newData = $this->getData(array(0 => 'first'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, true, $mode, 'foo', 'bar');
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
$parentData->expects($this->once())
|
$event = new DataEvent($this->form, $newData);
|
||||||
->method('foo')
|
$listener->preBind($event);
|
||||||
->with('first');
|
|
||||||
$parentData->expects($this->once())
|
$parentData->expects($this->at(0))
|
||||||
->method('bar')
|
->method('bar')
|
||||||
->with('second');
|
->with('second');
|
||||||
|
$parentData->expects($this->at(1))
|
||||||
|
->method('foo')
|
||||||
|
->with('first');
|
||||||
|
$parentData->expects($this->at(2))
|
||||||
|
->method('getAxes')
|
||||||
|
->will($this->returnValue('RESULT'));
|
||||||
|
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(true, true, true, 'foo', 'bar');
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
if (is_object($originalData)) {
|
$this->assertEquals('RESULT', $event->getData());
|
||||||
$this->assertSame($originalData, $event->getData());
|
|
||||||
}
|
|
||||||
|
|
||||||
// The data was not modified directly
|
|
||||||
// Thus it should not be written back into the parent data!
|
|
||||||
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDontCallAdderWithCustomNameIfDisallowed()
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testDontCallAdderWithCustomNameIfDisallowed($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_CarCustomNames');
|
$parentData = $this->getMock(__CLASS__ . '_CarCustomNames');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -486,27 +727,32 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData($originalDataArray);
|
$originalData = $this->getData($originalDataArray);
|
||||||
$newData = $this->getData(array(0 => 'first'));
|
$newData = $this->getData(array(0 => 'first'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, true, $mode, 'foo', 'bar');
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
|
|
||||||
$parentData->expects($this->never())
|
$parentData->expects($this->never())
|
||||||
->method('foo');
|
->method('foo');
|
||||||
$parentData->expects($this->once())
|
$parentData->expects($this->at(0))
|
||||||
->method('bar')
|
->method('bar')
|
||||||
->with('second');
|
->with('second');
|
||||||
|
$parentData->expects($this->at(1))
|
||||||
|
->method('getAxes')
|
||||||
|
->will($this->returnValue('RESULT'));
|
||||||
|
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, true, true, 'foo', 'bar');
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
if (is_object($originalData)) {
|
$this->assertEquals('RESULT', $event->getData());
|
||||||
$this->assertSame($originalData, $event->getData());
|
|
||||||
}
|
|
||||||
|
|
||||||
// The data was not modified
|
|
||||||
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testDontCallRemoverWithCustomNameIfDisallowed()
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
|
*/
|
||||||
|
public function testDontCallRemoverWithCustomNameIfDisallowed($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_CarCustomNames');
|
$parentData = $this->getMock(__CLASS__ . '_CarCustomNames');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -517,30 +763,33 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData($originalDataArray);
|
$originalData = $this->getData($originalDataArray);
|
||||||
$newData = $this->getData(array(0 => 'first'));
|
$newData = $this->getData(array(0 => 'first'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, false, $mode, 'foo', 'bar');
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
$parentData->expects($this->once())
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
|
|
||||||
|
$parentData->expects($this->at(0))
|
||||||
->method('foo')
|
->method('foo')
|
||||||
->with('first');
|
->with('first');
|
||||||
$parentData->expects($this->never())
|
$parentData->expects($this->never())
|
||||||
->method('bar');
|
->method('bar');
|
||||||
|
$parentData->expects($this->at(1))
|
||||||
|
->method('getAxes')
|
||||||
|
->will($this->returnValue('RESULT'));
|
||||||
|
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(true, false, true, 'foo', 'bar');
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
|
|
||||||
if (is_object($originalData)) {
|
$this->assertEquals('RESULT', $event->getData());
|
||||||
$this->assertSame($originalData, $event->getData());
|
|
||||||
}
|
|
||||||
|
|
||||||
// The data was not modified
|
|
||||||
$this->assertEquals($this->getData($originalDataArray), $event->getData());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
* @expectedException Symfony\Component\Form\Exception\FormException
|
* @expectedException Symfony\Component\Form\Exception\FormException
|
||||||
*/
|
*/
|
||||||
public function testThrowExceptionIfInvalidAdder()
|
public function testThrowExceptionIfInvalidAdder($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_CarCustomNames');
|
$parentData = $this->getMock(__CLASS__ . '_CarCustomNames');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -550,17 +799,21 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData(array(1 => 'second'));
|
$originalData = $this->getData(array(1 => 'second'));
|
||||||
$newData = $this->getData(array(0 => 'first'));
|
$newData = $this->getData(array(0 => 'first'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(true, false, $mode, 'doesnotexist');
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(true, false, true, 'doesnotexist');
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* @dataProvider getModesWithMergeIntoParent
|
||||||
* @expectedException Symfony\Component\Form\Exception\FormException
|
* @expectedException Symfony\Component\Form\Exception\FormException
|
||||||
*/
|
*/
|
||||||
public function testThrowExceptionIfInvalidRemover()
|
public function testThrowExceptionIfInvalidRemover($mode)
|
||||||
{
|
{
|
||||||
$parentData = $this->getMock(__CLASS__ . '_CarCustomNames');
|
$parentData = $this->getMock(__CLASS__ . '_CarCustomNames');
|
||||||
$parentForm = $this->getForm('car');
|
$parentForm = $this->getForm('car');
|
||||||
@ -570,10 +823,22 @@ abstract class MergeCollectionListenerTest extends \PHPUnit_Framework_TestCase
|
|||||||
$originalData = $this->getData(array(1 => 'second'));
|
$originalData = $this->getData(array(1 => 'second'));
|
||||||
$newData = $this->getData(array(0 => 'first'));
|
$newData = $this->getData(array(0 => 'first'));
|
||||||
|
|
||||||
|
$listener = new MergeCollectionListener(false, true, $mode, null, 'doesnotexist');
|
||||||
|
|
||||||
$this->form->setData($originalData);
|
$this->form->setData($originalData);
|
||||||
|
|
||||||
|
$event = new DataEvent($this->form, $newData);
|
||||||
|
$listener->preBind($event);
|
||||||
$event = new FilterDataEvent($this->form, $newData);
|
$event = new FilterDataEvent($this->form, $newData);
|
||||||
$listener = new MergeCollectionListener(false, true, true, null, 'doesnotexist');
|
|
||||||
$listener->onBindNormData($event);
|
$listener->onBindNormData($event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dataProvider getInvalidModes
|
||||||
|
* @expectedException Symfony\Component\Form\Exception\FormException
|
||||||
|
*/
|
||||||
|
public function testThrowExceptionIfInvalidMode($mode)
|
||||||
|
{
|
||||||
|
new MergeCollectionListener(true, true, $mode);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -391,4 +391,25 @@ class PropertyPathTest extends \PHPUnit_Framework_TestCase
|
|||||||
|
|
||||||
new PropertyPath(null);
|
new PropertyPath(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testGetParent_dot()
|
||||||
|
{
|
||||||
|
$propertyPath = new PropertyPath('grandpa.parent.child');
|
||||||
|
|
||||||
|
$this->assertEquals(new PropertyPath('grandpa.parent'), $propertyPath->getParent());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testGetParent_index()
|
||||||
|
{
|
||||||
|
$propertyPath = new PropertyPath('grandpa.parent[child]');
|
||||||
|
|
||||||
|
$this->assertEquals(new PropertyPath('grandpa.parent'), $propertyPath->getParent());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testGetParent_noParent()
|
||||||
|
{
|
||||||
|
$propertyPath = new PropertyPath('path');
|
||||||
|
|
||||||
|
$this->assertNull($propertyPath->getParent());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user