Skip to content

Commit

Permalink
Prefer property path instead of method name
Browse files Browse the repository at this point in the history
  • Loading branch information
VincentLanglet committed Feb 9, 2021
1 parent 9f91d3b commit 6dee4e4
Show file tree
Hide file tree
Showing 2 changed files with 21 additions and 37 deletions.
56 changes: 20 additions & 36 deletions src/Admin/BaseFieldDescription.php
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
use Sonata\AdminBundle\Exception\NoValueException;
use Symfony\Component\PropertyAccess\Exception\ExceptionInterface;
use Symfony\Component\PropertyAccess\PropertyAccess;
use Symfony\Component\PropertyAccess\PropertyPathInterface;

/**
* A FieldDescription hold the information about a field. A typical
Expand All @@ -36,7 +37,7 @@
* - name (o) : the name used (label in the form, title in the list)
* - link_parameters (o) : add link parameter to the related Admin class when
* the Admin.generateUrl is called
* - accessor : the method or the method name to retrieve the related value
* - accessor : the method or the property path to retrieve the related value
* - associated_tostring : (deprecated, use associated_property option)
* the method to retrieve the "string" representation
* of the collection element.
Expand Down Expand Up @@ -425,47 +426,30 @@ public function getFieldValue($object, $fieldName)
.' Use the "accessor" code instead',
\E_USER_DEPRECATED
);
}

// prefer method name given in the code option
// NEXT_MAJOR: Remove this line and uncomment the following
$getter = $this->getOption('accessor', $this->getOption('code'));
// $getter = $this->getOption('accessor');
if ($getter) {
if (\is_callable($getter)) {
return $getter($object);
}

if (!method_exists($object, $getter)) {
@trigger_error(
'Passing a non-existing method in the "code" option is deprecated'
.' since sonata-project/admin-bundle 3.x and will throw an exception in 4.0.',
\E_USER_DEPRECATED
);

// NEXT_MAJOR: Remove the deprecation and uncomment the next line.
// throw new \LogicException('The method "%s"() does not exist.', $getter);
} elseif (!\is_callable([$object, $getter])) {
$getter = $this->getOption('code');
if ($this->getOption('parameters')) {
@trigger_error(
'Passing a non-callable method in the "code" option is deprecated'
.' since sonata-project/admin-bundle 3.x and will throw an exception in 4.0.',
'The option "parameters" is deprecated since sonata-project/admin-bundle 3.x and will be removed in 4.0.',
\E_USER_DEPRECATED
);

// NEXT_MAJOR: Remove the deprecation and uncomment the next line.
// throw new \LogicException('The method "%s"() does not have public access.', $getter);
} else {
if ($this->getOption('parameters')) {
@trigger_error(
'The option "parameters" is deprecated since sonata-project/admin-bundle 3.x and will be removed in 4.0.',
\E_USER_DEPRECATED
);
return $object->{$getter}(...$this->getOption('parameters'));
}

return $object->{$getter}(...$this->getOption('parameters'));
}
return $object->{$getter}();
}

return $object->{$getter}();
}
// prefer the method or the property path given in the code option
$accessor = $this->getOption('accessor', $fieldName);
if (\is_callable($accessor)) {
return $accessor($object);
} elseif (!\is_string($accessor) && !$accessor instanceof PropertyPathInterface) {
throw new \TypeError(sprintf(
'The option "accessor" must be a string, a callable or a %s, %s given.',
PropertyPathInterface::class,
\is_object($accessor) ? 'instance of '.\get_class($accessor) : \gettype($accessor)
));
}

// NEXT_MAJOR: Remove the condition code and the else part
Expand All @@ -475,7 +459,7 @@ public function getFieldValue($object, $fieldName)
->getPropertyAccessor();

try {
return $propertyAccesor->getValue($object, $fieldName);
return $propertyAccesor->getValue($object, $accessor);
} catch (ExceptionInterface $exception) {
throw new NoValueException(
sprintf('Cannot access property "%s" in class "%s".', $this->getName(), \get_class($object)),
Expand Down
2 changes: 1 addition & 1 deletion tests/Admin/BaseFieldDescriptionTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -188,7 +188,7 @@ public function testGetFieldValueWithNullObject(): void

public function testGetFieldValueWithAccessor(): void
{
$description = new FieldDescription('name', ['accessor' => 'getFoo']);
$description = new FieldDescription('name', ['accessor' => 'foo']);
$mock = $this->getMockBuilder(\stdClass::class)->addMethods(['getFoo'])->getMock();
$mock->expects($this->once())->method('getFoo')->willReturn(42);
$this->assertSame(42, $description->getFieldValue($mock, 'fake'));
Expand Down

0 comments on commit 6dee4e4

Please sign in to comment.