address-8.x-1.x-dev/src/Plugin/Field/FieldFormatter/AddressPlainFormatter.php
src/Plugin/Field/FieldFormatter/AddressPlainFormatter.php
<?php
namespace Drupal\address\Plugin\Field\FieldFormatter;
use CommerceGuys\Addressing\AddressFormat\AddressField;
use CommerceGuys\Addressing\AddressFormat\AddressFormat;
use CommerceGuys\Addressing\AddressFormat\AddressFormatRepositoryInterface;
use CommerceGuys\Addressing\Country\CountryRepositoryInterface;
use CommerceGuys\Addressing\Locale;
use CommerceGuys\Addressing\Subdivision\SubdivisionRepositoryInterface;
use Drupal\address\AddressInterface;
use Drupal\Core\Field\FieldDefinitionInterface;
use Drupal\Core\Field\FieldItemListInterface;
use Drupal\Core\Field\FormatterBase;
use Drupal\Core\Language\LanguageInterface;
use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
use Symfony\Component\DependencyInjection\ContainerInterface;
/**
* Plugin implementation of the 'address_plain' formatter.
*
* @FieldFormatter(
* id = "address_plain",
* label = @Translation("Plain"),
* field_types = {
* "address",
* },
* )
*/
class AddressPlainFormatter extends FormatterBase implements ContainerFactoryPluginInterface {
/**
* The address format repository.
*
* @var \CommerceGuys\Addressing\AddressFormat\AddressFormatRepositoryInterface
*/
protected $addressFormatRepository;
/**
* The country repository.
*
* @var \CommerceGuys\Addressing\Country\CountryRepositoryInterface
*/
protected $countryRepository;
/**
* The subdivision repository.
*
* @var \CommerceGuys\Addressing\Subdivision\SubdivisionRepositoryInterface
*/
protected $subdivisionRepository;
/**
* Constructs an AddressPlainFormatter object.
*
* @param string $plugin_id
* The plugin_id for the formatter.
* @param mixed $plugin_definition
* The plugin implementation definition.
* @param \Drupal\Core\Field\FieldDefinitionInterface $field_definition
* The definition of the field to which the formatter is associated.
* @param array $settings
* The formatter settings.
* @param string $label
* The formatter label display setting.
* @param string $view_mode
* The view mode.
* @param array $third_party_settings
* Any third party settings.
* @param \CommerceGuys\Addressing\AddressFormat\AddressFormatRepositoryInterface $address_format_repository
* The address format repository.
* @param \CommerceGuys\Addressing\Country\CountryRepositoryInterface $country_repository
* The country repository.
* @param \CommerceGuys\Addressing\Subdivision\SubdivisionRepositoryInterface $subdivision_repository
* The subdivision repository.
*/
public function __construct($plugin_id, $plugin_definition, FieldDefinitionInterface $field_definition, array $settings, $label, $view_mode, array $third_party_settings, AddressFormatRepositoryInterface $address_format_repository, CountryRepositoryInterface $country_repository, SubdivisionRepositoryInterface $subdivision_repository) {
parent::__construct($plugin_id, $plugin_definition, $field_definition, $settings, $label, $view_mode, $third_party_settings);
$this->addressFormatRepository = $address_format_repository;
$this->countryRepository = $country_repository;
$this->subdivisionRepository = $subdivision_repository;
}
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container, array $configuration, $pluginId, $pluginDefinition) {
// @see \Drupal\Core\Field\FormatterPluginManager::createInstance().
return new static(
$pluginId,
$pluginDefinition,
$configuration['field_definition'],
$configuration['settings'],
$configuration['label'],
$configuration['view_mode'],
$configuration['third_party_settings'],
$container->get('address.address_format_repository'),
$container->get('address.country_repository'),
$container->get('address.subdivision_repository')
);
}
/**
* {@inheritdoc}
*/
public function viewElements(FieldItemListInterface $items, $langcode) {
$elements = [];
foreach ($items as $delta => $item) {
$elements[$delta] = $this->viewElement($item, $langcode);
}
return $elements;
}
/**
* Builds a renderable array for a single address item.
*
* @param \Drupal\address\AddressInterface $address
* The address.
* @param string $langcode
* The language that should be used to render the field.
*
* @return array
* A renderable array.
*/
protected function viewElement(AddressInterface $address, $langcode) {
// If the parent entity is non-translatable, $address->getLocale() contains
// the interface language at the time of address creation, while $langcode
// contains the current interface language. Our goal is to have the country
// name be recognizable by all users, making $langcode a safer bet.
$countries = $this->countryRepository->getList($langcode);
$country_code = $address->getCountryCode();
$address_format = $this->addressFormatRepository->get($country_code);
$values = $this->getValues($address, $address_format);
$element = [
'#theme' => 'address_plain',
'#given_name' => $values['givenName'],
'#additional_name' => $values['additionalName'],
'#family_name' => $values['familyName'],
'#organization' => $values['organization'],
'#address_line1' => $values['addressLine1'],
'#address_line2' => $values['addressLine2'],
'#address_line3' => $values['addressLine3'],
'#postal_code' => $values['postalCode'],
'#sorting_code' => $values['sortingCode'],
'#administrative_area' => $values['administrativeArea'],
'#locality' => $values['locality'],
'#dependent_locality' => $values['dependentLocality'],
'#country' => [
'code' => $country_code,
'name' => $countries[$country_code],
],
'#address' => $address,
'#view_mode' => $this->viewMode,
'#cache' => [
'contexts' => [
'languages:' . LanguageInterface::TYPE_INTERFACE,
'languages:' . LanguageInterface::TYPE_CONTENT,
],
],
];
return $element;
}
/**
* Gets the address values used for rendering.
*
* @param \Drupal\address\AddressInterface $address
* The address.
* @param \CommerceGuys\Addressing\AddressFormat\AddressFormat $address_format
* The address format.
*
* @return array
* The values, keyed by address field.
*/
protected function getValues(AddressInterface $address, AddressFormat $address_format) {
$values = [];
foreach (AddressField::getAll() as $field) {
$getter = 'get' . ucfirst($field);
$values[$field] = $address->$getter();
}
$original_values = [];
$subdivision_fields = $address_format->getUsedSubdivisionFields();
$parents = [];
foreach ($subdivision_fields as $index => $field) {
$value = $values[$field];
// The template needs access to both the subdivision code and name.
$values[$field] = [
'code' => $value,
'name' => '',
];
if (empty($value)) {
// This level is empty, but it might be optional so check sublevels.
continue;
}
$parents[] = ($index > 0 && !empty($original_values[$subdivision_fields[$index - 1]]))
? $original_values[$subdivision_fields[$index - 1]]
: $address->getCountryCode();
$subdivision = $this->subdivisionRepository->get($value, $parents);
if (!$subdivision) {
continue;
}
// Remember the original value so that it can be used for $parents.
$original_values[$field] = $value;
// Replace the value with the expected code.
if (Locale::matchCandidates($address->getLocale(), $subdivision->getLocale())) {
$values[$field] = [
'code' => $subdivision->getLocalCode(),
'name' => $subdivision->getLocalName(),
];
}
else {
$values[$field] = [
'code' => $subdivision->getCode(),
'name' => $subdivision->getName(),
];
}
if (!$subdivision->hasChildren()) {
// The current subdivision has no children, stop.
break;
}
}
return $values;
}
}
