metatag-8.x-1.x-dev/src/Plugin/Field/FieldWidget/MetatagFirehose.php
src/Plugin/Field/FieldWidget/MetatagFirehose.php
<?php
namespace Drupal\metatag\Plugin\Field\FieldWidget;
use Drupal\Component\Serialization\Json;
use Drupal\Core\Config\ConfigFactoryInterface;
use Drupal\Core\Field\Attribute\FieldWidget;
use Drupal\Core\Field\FieldDefinitionInterface;
use Drupal\Core\Field\FieldItemListInterface;
use Drupal\Core\Field\WidgetBase;
use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\Plugin\ContainerFactoryPluginInterface;
use Drupal\Core\StringTranslation\StringTranslationTrait;
use Drupal\Core\StringTranslation\TranslatableMarkup;
use Drupal\metatag\MetatagManagerInterface;
use Drupal\metatag\MetatagTagPluginManager;
use Symfony\Component\DependencyInjection\ContainerInterface;
/**
* Advanced widget for metatag field.
*/
#[FieldWidget(
id: 'metatag_firehose',
label: new TranslatableMarkup('Advanced meta tags form'),
field_types: [
'metatag',
]
)]
class MetatagFirehose extends WidgetBase implements ContainerFactoryPluginInterface {
use StringTranslationTrait;
/**
* Instance of MetatagManager service.
*
* @var \Drupal\metatag\MetatagManagerInterface
*/
protected $metatagManager;
/**
* Instance of MetatagTagPluginManager service.
*
* @var \Drupal\metatag\MetatagTagPluginManager
*/
protected $metatagPluginManager;
/**
* The config factory.
*
* @var \Drupal\Core\Config\ConfigFactoryInterface
*/
protected $configFactory;
/**
* {@inheritdoc}
*/
public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) {
return new static(
$plugin_id,
$plugin_definition,
$configuration['field_definition'],
$configuration['settings'],
$configuration['third_party_settings'],
$container->get('metatag.manager'),
$container->get('plugin.manager.metatag.tag'),
$container->get('config.factory')
);
}
/**
* {@inheritdoc}
*/
public static function defaultSettings() {
return [
'sidebar' => TRUE,
'use_details' => TRUE,
] + parent::defaultSettings();
}
/**
* {@inheritdoc}
*/
public function settingsForm(array $form, FormStateInterface $form_state) {
$element['sidebar'] = [
'#type' => 'checkbox',
'#title' => $this->t('Place field in sidebar'),
'#default_value' => $this->getSetting('sidebar'),
'#description' => $this->t('If checked, the field will be placed in the sidebar on entity forms.'),
];
$element['use_details'] = [
'#type' => 'checkbox',
'#title' => $this->t('Wrap the meta tags in a collapsed details container.'),
'#default_value' => $this->getSetting('use_details'),
'#description' => $this->t('If checked, the contents of the field will be placed inside a collapsed details container.'),
'#states' => [
'visible' => [
'input[name$="[sidebar]"]' => ['checked' => FALSE],
],
],
];
return $element;
}
/**
* {@inheritdoc}
*/
public function settingsSummary() {
if ($this->getSetting('sidebar')) {
$summary['sidebar'] = $this->t('Use sidebar: Yes');
}
else {
$summary['sidebar'] = $this->t('Use sidebar: No');
if ($this->getSetting('use_details')) {
$summary['use_details'] = $this->t('Use details container: Yes');
}
else {
$summary['use_details'] = $this->t('Use details container: No');
}
}
return $summary;
}
/**
* {@inheritdoc}
*/
public function __construct($plugin_id, $plugin_definition, FieldDefinitionInterface $field_definition, array $settings, array $third_party_settings, MetatagManagerInterface $manager, MetatagTagPluginManager $plugin_manager, ConfigFactoryInterface $config_factory) {
parent::__construct($plugin_id, $plugin_definition, $field_definition, $settings, $third_party_settings);
$this->metatagManager = $manager;
$this->metatagPluginManager = $plugin_manager;
$this->configFactory = $config_factory;
}
/**
* {@inheritdoc}
*/
public function formElement(FieldItemListInterface $items, $delta, array $element, array &$form, FormStateInterface $form_state) {
// @todo Does this need to be rewritten to use $items->getValue()?
$item = $items[$delta];
$default_tags = metatag_get_default_tags($items->getEntity());
// Retrieve the values for each metatag from the serialized array.
$values = [];
if (!empty($item->value)) {
$values = metatag_data_decode($item->value);
}
// Make sure that this variable is always an array to avoid problems when
// unserializing didn't work correctly and it as returned as FALSE.
// @see https://www.php.net/unserialize
if (!is_array($values)) {
$values = [];
}
// Populate fields which have not been overridden in the entity.
if (!empty($default_tags)) {
foreach ($default_tags as $tag_id => $tag_value) {
if (!isset($values[$tag_id]) && !empty($tag_value)) {
$values[$tag_id] = $tag_value;
}
}
}
// Retrieve configuration settings.
$settings = $this->configFactory->get('metatag.settings');
$entity_type_groups = $settings->get('entity_type_groups');
// Find the current entity type and bundle.
/** @var \Drupal\Core\Entity\FieldableEntityInterface $get_entity */
$get_entity = $item->getEntity();
$entity_type = $get_entity->getentityTypeId();
$entity_bundle = $get_entity->bundle();
// See if there are requested groups for this entity type and bundle.
$groups = [];
if (!empty($entity_type_groups[$entity_type]) && !empty($entity_type_groups[$entity_type][$entity_bundle])) {
$groups = $entity_type_groups[$entity_type][$entity_bundle];
}
// Limit the form to requested groups, if any.
if (!empty($groups)) {
$element = $this->metatagManager->form($values, $element, [$entity_type], $groups);
}
// Otherwise, display all groups.
else {
$element = $this->metatagManager->form($values, $element, [$entity_type]);
}
// If the "sidebar" option was checked on the field widget, put the form
// element into the form's "advanced" group. Otherwise, let it default to
// the main field area.
$sidebar = $this->getSetting('sidebar');
if ($sidebar) {
$element['#group'] = 'advanced';
}
// If the "use_details" option was not checked on the field widget, put the
// form element into normal container. Otherwise, let it default to a
// detail's container.
$details = $this->getSetting('use_details');
if (!$sidebar && !$details) {
$element['#type'] = 'container';
}
// Scroll height configuration.
$scroll_height = $settings->get('tag_scroll_max_height');
if (!empty($scroll_height)) {
$form['#attached']['drupalSettings']['metatag']['max_height'] = $scroll_height;
$form['#attached']['library'][] = 'metatag/firehose_widget';
$element['#attributes']['class'][] = 'metatags';
$element['#attributes']['style'][] = 'max-height:' . $scroll_height . ';';
}
return $element;
}
/**
* {@inheritdoc}
*/
public function massageFormValues(array $values, array $form, FormStateInterface $form_state) {
// Flatten the values array to remove the groups and then serialize all the
// meta tags into one value for storage.
$tag_manager = $this->metatagPluginManager;
foreach ($values as &$value) {
$flattened_value = [];
foreach ($value as $group) {
// Exclude the "original delta" value.
if (is_array($group)) {
foreach ($group as $tag_id => $tag_value) {
$tag = $tag_manager->createInstance($tag_id);
$tag->setValue($tag_value);
if (!empty($tag->value())) {
$flattened_value[$tag_id] = $tag->value();
}
}
}
}
$value = Json::encode($flattened_value);
}
return $values;
}
}
