Commit 9451f8a8 authored by alexpott's avatar alexpott
Browse files

Issue #2123251 by Jelle_S, marcvangend, attiks, Sutharsan: Improve DX of...

Issue #2123251 by Jelle_S, marcvangend, attiks, Sutharsan: Improve DX of responsive images; convert theme functions to new #type element
parent a420c71c
......@@ -74,11 +74,8 @@ function responsive_image_theme() {
return array(
'responsive_image' => array(
'variables' => array(
'link_path' => NULL,
'uri' => NULL,
'width' => NULL,
'height' => NULL,
'alt' => '',
'title' => NULL,
'attributes' => array(),
'responsive_image_style_id' => array(),
),
......@@ -86,58 +83,51 @@ function responsive_image_theme() {
'responsive_image_formatter' => array(
'variables' => array(
'item' => NULL,
'item_attributes' => NULL,
'url' => NULL,
'responsive_image_style_id' => array(),
'responsive_image_style_id' => NULL,
),
'function' => 'theme_responsive_image_formatter',
),
);
}
/**
* Returns HTML for a responsive image field formatter.
* Prepares variables for responsive image formatter templates.
*
* Default template: responsive-image-formatter.html.twig.
*
* @param array $variables
* An associative array containing:
* - item: An ImageItem object.
* - responsive_image_style_id: The ID of the responsive image style.
* - item_attributes: An optional associative array of HTML attributes to be
* placed in the img tag.
* - responsive_image_style_id: A responsive image style.
* - url: An optional \Drupal\Core\Url object.
*
* @ingroup themeable
*/
function theme_responsive_image_formatter($variables) {
$item = $variables['item'];
if (!isset($variables['responsive_image_style_id']) || empty($variables['responsive_image_style_id'])) {
$image_formatter = array(
'#theme' => 'image_formatter',
'#item' => $item,
'#url' => $variables['url'],
);
return drupal_render($image_formatter);
}
$responsive_image = array(
'#theme' => 'responsive_image',
'#width' => $item->width,
'#height' => $item->height,
function template_preprocess_responsive_image_formatter(&$variables) {
$variables['responsive_image'] = array(
'#type' => 'responsive_image',
'#responsive_image_style_id' => $variables['responsive_image_style_id'],
);
if (isset($item->uri)) {
$responsive_image['#uri'] = $item->uri;
}
elseif ($entity = $item->entity) {
$responsive_image['#uri'] = $entity->getFileUri();
$responsive_image['#entity'] = $entity;
}
$responsive_image['#alt'] = $item->alt;
$item = $variables['item'];
$attributes = array();
// Do not output an empty 'title' attribute.
if (Unicode::strlen($item->title) != 0) {
$responsive_image['#title'] = $item->title;
$attributes['title'] = $item->title;
}
if (isset($variables['url'])) {
return \Drupal::l($responsive_image, $variables['url']);
$attributes['alt'] = $item->alt;
$attributes += $variables['item_attributes'];
if (($entity = $item->entity) && empty($item->uri)) {
$variables['responsive_image']['#uri'] = $entity->getFileUri();
}
else {
$variables['responsive_image']['#uri'] = $item->uri;
}
return drupal_render($responsive_image);
foreach (array('width', 'height') as $key) {
$variables['responsive_image']["#$key"] = $item->$key;
}
$variables['responsive_image']['#attributes'] = $attributes;
}
/**
......@@ -147,13 +137,9 @@ function theme_responsive_image_formatter($variables) {
*
* @param $variables
* An associative array containing:
* - uri: Either the path of the image file (relative to base_path()) or a
* full URL.
* - uri: The URI of the image.
* - width: The width of the image (if known).
* - height: The height of the image (if known).
* - alt: The alternative text for text-based browsers.
* - title: The title text is displayed when the image is hovered in some
* popular browsers.
* - attributes: Associative array of attributes to be placed in the img tag.
* - responsive_image_style_id: The ID of the responsive image style.
*/
......@@ -191,11 +177,16 @@ function template_preprocess_responsive_image(&$variables) {
),
),
);
foreach (array('alt', 'title', 'attributes') as $key) {
if (isset($variables[$key])) {
$variables['img_element']["#$key"] = $variables[$key];
unset($variables[$key]);
if (isset($variables['attributes'])) {
if (isset($variables['attributes']['alt'])) {
$variables['img_element']['#alt'] = $variables['attributes']['alt'];
unset($variables['attributes']['alt']);
}
if (isset($variables['attributes']['title'])) {
$variables['img_element']['#title'] = $variables['attributes']['title'];
unset($variables['attributes']['title']);
}
$variables['img_element']['#attributes'] = $variables['attributes'];
}
}
......
<?php
/**
* @file
* Contains \Drupal\responsive_image\Element\ResponsiveImage.
*/
namespace Drupal\responsive_image\Element;
use Drupal\Core\Render\Element\RenderElement;
/**
* Provides a responsive image element.
*
* @RenderElement("responsive_image")
*/
class ResponsiveImage extends RenderElement {
/**
* {@inheritdoc}
*/
public function getInfo() {
return [
'#theme' => 'responsive_image',
'#attached' => [
'library' => ['core/picturefill'],
],
];
}
}
......@@ -204,14 +204,16 @@ public function viewElements(FieldItemListInterface $items) {
if (isset($link_file)) {
$url = Url::fromUri(file_create_url($file->getFileUri()));
}
// Extract field item attributes for the theme function, and unset them
// from the $item so that the field template does not re-render them.
$item = $file->_referringItem;
$item_attributes = $item->_attributes;
unset($item->_attributes);
$elements[$delta] = array(
'#theme' => 'responsive_image_formatter',
'#attached' => array(
'library' => array(
'core/picturefill',
),
),
'#item' => $file->_referringItem,
'#item' => $item,
'#item_attributes' => $item_attributes,
'#responsive_image_style_id' => $responsive_image_style ? $responsive_image_style->id() : '',
'#url' => $url,
'#cache' => array(
......@@ -219,7 +221,6 @@ public function viewElements(FieldItemListInterface $items) {
),
);
}
return $elements;
}
}
......@@ -193,21 +193,15 @@ protected function doTestResponsiveImageFieldFormatters($scheme, $empty_styles =
$display_options = array(
'type' => 'responsive_image',
'settings' => array(
'image_link' => 'file'
'image_link' => 'file',
'responsive_image_style' => 'style_one',
),
);
$display = entity_get_display('node', 'article', 'default');
$display->setComponent($field_name, $display_options)
->save();
$image = array(
'#theme' => 'image',
'#uri' => $image_uri,
'#width' => 360,
'#height' => 240,
'#alt' => $alt,
);
$default_output = '<a href="' . file_create_url($image_uri) . '">' . drupal_render($image) . '</a>';
$default_output = '<a href="' . file_create_url($image_uri) . '"><picture';
$this->drupalGet('node/' . $nid);
$cache_tags_header = $this->drupalGetHeader('X-Drupal-Cache-Tags');
$this->assertTrue(!preg_match('/ image_style\:/', $cache_tags_header), 'No image style cache tag found.');
......@@ -285,7 +279,10 @@ protected function doTestResponsiveImageFieldFormatters($scheme, $empty_styles =
),
),
);
$default_output = drupal_render($fallback_image);
// The image.html.twig template has a newline after the <img> tag but
// responsive-image.html.twig doesn't have one after the fallback image, so
// we remove it here.
$default_output = trim(drupal_render($fallback_image));
$this->assertRaw($default_output, 'Image style large formatter displaying correctly on full node view.');
if ($scheme == 'private') {
......@@ -354,7 +351,7 @@ public function testResponsiveImageFieldFormattersEmptyMediaQuery() {
$this->drupalGet('node/' . $nid);
// Assert an empty media attribute is not output.
$this->assertNoPattern('@srcset=" 1x".+?media="@');
$this->assertNoPattern('@srcset=" 1x".+?media=".+?/><source@');
// Assert the media attribute is present if it has a value.
$thumbnail_style = ImageStyle::load('thumbnail');
......
{#
/**
* @file
* Default theme implementation to display a formatted responsive image field.
*
* Available variables:
* - responsive_image: A collection of responsive image data.
* - url: An optional URL the image can be linked to.
*
* @see template_preprocess_responsive_image_formatter()
*
* @ingroup themeable
*/
#}
{% if url %}
<a href="{{ url }}">{{ responsive_image }}</a>
{% else %}
{{ responsive_image }}
{% endif %}
......@@ -5,7 +5,7 @@
*
* Available variables:
* - sources: The attributes of the <source> tags for this <picture> tag.
* - fallback_image: The fallback <img> tag to use for this <picture> tag.
* - img_element: The controlling image, with the fallback image in srcset.
*
* @see template_preprocess()
* @see template_preprocess_responsive_image()
......
Markdown is supported
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment