You are here

picture.module in Picture 7.2

Same filename and directory in other branches
  1. 8 picture.module
  2. 7 picture.module

Picture formatter.

File

picture.module
View source
<?php

/**
 * @file
 * Picture formatter.
 */
define('PICTURE_EMPTY_IMAGE', '_empty image_');
define('PICTURE_ORIGINAL_IMAGE', '_original image_');
if (!variable_get('picture_updated_to_file_entity_2', FALSE)) {
  module_load_include('file_entity_1.inc', 'picture');
}

/**
 * Implements hook_help().
 */
function picture_help($path, $arg) {
  $output = '';
  switch ($path) {
    case 'admin/help#picture':
      $output .= '<h3>' . t('About') . '</h3>';
      $output .= '<p>' . t('The Picture module provides an image formatter and breakpoint mappings to output responsive images using the HTML5 picture tag. For more information, see the <a href="!picture">online documentation for the Picture module</a>.', array(
        '!picture' => 'https://drupal.org/documentation/modules/picture',
      )) . '</p>';
      $output .= '<h3>' . t('Uses') . '</h3>';
      $output .= '<dl>';
      $output .= '<dt>' . t('Defining picture mappings') . '</dt>';
      $output .= '<dd>' . t('By creating picture mappings you define the image styles that are being used to output images at certain breakpoints. On the <a href="!picture_mapping">Picture mappings</a> page, click <em>Add picture mapping</em> to create a new mapping. First chose a label and a breakpoint group and click Save. After that you can choose the image styles that will be used for each breakpoint. Image styles can be defined on the <a href="!image_styles">Image styles page</a> that is provided by the <a href="!image_help">Image module</a>. Breakpoints are defined in the info files of the theme or you can create custom breakpoints. See the <a href="!breakpoint_help">help page of the Breakpoint module</a> for more information.', array(
        '!picture_mapping' => url('admin/config/media/picture'),
        '!image_styles' => url('admin/config/media/image-styles'),
        '!image_help' => url('admin/help/image'),
        '!breakpoint_help' => url('admin/help/breakpoints'),
      )) . '</dd>';
      $output .= '<dt>' . t('Using picture mappings in Image fields') . '</dt>';
      $output .= '<dd>' . t('After defining responsive image mappings, you can use them in the display settings for your Image fields, so that the site displays responsive images using the HTML5 picture tag. Open the Manage display page for the entity type (content type, taxonomy vocabulary, etc.) that the Image field is attached to. Choose the format <em>Picture</em>, click the Edit icon, and select one of the picture mappings that you have created. For general information on how to manage fields and their display see the <a href="!field_ui">help page of the Field UI module</a>.', array(
        '!field_ui' => url('admin/help/field_ui'),
      )) . '</dd>';
      $output .= '</dl>';
      break;
    case 'admin/config/media/picture':
      $output .= '<p>' . t('A picture mapping associates an image style with each breakpoint defined by your theme.') . '</p>';
      break;
  }
  return $output;
}

/**
 * Implements hook_flush_caches().
 */
function picture_flush_caches() {

  // After update.php, caches are flushed. Check if file_entity has updated to
  // version 2.
  module_load_include('admin.inc', 'picture');
  variable_set('picture_updated_to_file_entity_2', _picture_update_to_file_entity_2());
}

/**
 * Implements hook_modules_enabled().
 */
function picture_modules_enabled($modules) {
  if (in_array('file_entity', $modules)) {
    module_load_include('admin.inc', 'picture');
    module_load_include('module', 'file_entity');
    variable_set('picture_updated_to_file_entity_2', _picture_update_to_file_entity_2());
  }
}

/**
 * Implements hook_permission().
 */
function picture_permission() {
  return array(
    'administer pictures' => array(
      'title' => t('Administer Pictures'),
      'description' => t('Administer Pictures'),
    ),
  );
}

/**
 * Implements hook_menu().
 */
function picture_menu() {
  $items = array();
  $items['admin/config/media/picture/settings'] = array(
    'title' => 'Settings',
    'type' => MENU_LOCAL_TASK,
    'description' => 'Picture settings (CKEditor, File Entity, ...)',
    'page callback' => 'drupal_get_form',
    'page arguments' => array(
      'picture_admin_settings',
    ),
    'access arguments' => array(
      'administer pictures',
    ),
    'file' => 'picture.admin.inc',
    'weight' => 0,
  );
  return $items;
}

/**
 * Load mappings.
 */
function picture_mapping_load($name) {
  ctools_include('export');
  if ($name) {
    $mappings = ctools_export_load_object('picture_mapping', 'names', array(
      $name,
    ));
    $mapping = isset($mappings[$name]) ? $mappings[$name] : FALSE;
    return $mapping;
  }
  return FALSE;
}

/**
 * Load all mappings.
 */
function picture_mapping_load_all() {
  ctools_include('export');
  return ctools_export_load_object('picture_mapping');
}

/**
 * Save mappings.
 */
function picture_mapping_save(PictureMapping $mapping) {
  return $mapping
    ->save();
}

/**
 * Implements hook_library().
 */
function picture_library() {
  $libraries = array();
  switch (variable_get('picture_polyfill_version', 'min')) {
    case 'min':
      $libraries['picturefill_head'] = array(
        'title' => t('Picturefill head fix'),
        'version' => '3.0.1',
        'js' => array(
          'document.createElement( "picture" );' => array(
            'type' => 'inline',
            'weight' => -20,
            'group' => JS_LIBRARY,
            'scope' => 'header',
            'need_jquery' => FALSE,
          ),
        ),
      );
      $libraries['picturefill'] = array(
        'title' => t('Picturefill'),
        'website' => 'https://github.com/scottjehl/picturefill',
        'version' => '3.0.1',
        'js' => array(
          drupal_get_path('module', 'picture') . '/picturefill/picturefill.min.js' => array(
            'type' => 'file',
            'weight' => -10,
            'group' => JS_DEFAULT,
            'scope' => variable_get('picture_js_scope', 'footer'),
            'need_jquery' => FALSE,
          ),
        ),
      );
      $libraries['picture.ajax'] = array(
        'title' => t('Ajax support for picture'),
        'version' => VERSION,
        'js' => array(
          drupal_get_path('module', 'picture') . '/picture.min.js' => array(
            'type' => 'file',
            'weight' => -10,
            'group' => JS_DEFAULT,
            'scope' => variable_get('picture_js_scope', 'footer'),
            'need_jquery' => FALSE,
          ),
        ),
      );
      $libraries['lazysizes'] = array(
        'title' => t('Lazyload for picture element'),
        'version' => '1.0.1',
        'js' => array(
          drupal_get_path('module', 'picture') . '/lazysizes/lazysizes.min.js' => array(
            'type' => 'file',
            'weight' => -20,
            'group' => JS_LIBRARY,
            'need_jquery' => FALSE,
          ),
        ),
      );
      $libraries['lazysizes_aspect_ratio'] = array(
        'title' => t('Aspect ratio plugin for lazysizes'),
        'version' => '1.0.1',
        'js' => array(
          drupal_get_path('module', 'picture') . '/lazysizes/plugins/aspectratio/ls.aspectratio.min.js' => array(
            'type' => 'file',
            'weight' => -10,
            'group' => JS_LIBRARY,
            'need_jquery' => FALSE,
          ),
        ),
        'css' => array(
          drupal_get_path('module', 'picture') . '/lazysizes/plugins/aspectratio/ls.aspectratio.css' => array(
            'type' => 'file',
            'media' => 'screen',
          ),
        ),
      );
      break;
    case 'dev':
      $libraries['picturefill_head'] = array(
        'title' => t('Picturefill head fix'),
        'version' => '3.0.1',
        'js' => array(
          'document.createElement( "picture" );' => array(
            'type' => 'inline',
            'weight' => -10,
            'group' => JS_DEFAULT,
            'scope' => 'header',
            'need_jquery' => FALSE,
          ),
        ),
      );
      $libraries['picturefill'] = array(
        'title' => t('Picturefill'),
        'website' => 'https://github.com/scottjehl/picturefill',
        'version' => '3.0.1',
        'js' => array(
          drupal_get_path('module', 'picture') . '/picturefill/picturefill.js' => array(
            'type' => 'file',
            'weight' => -10,
            'group' => JS_DEFAULT,
            'scope' => variable_get('picture_js_scope', 'footer'),
            'need_jquery' => FALSE,
          ),
        ),
      );
      $libraries['picture.ajax'] = array(
        'title' => t('Ajax support for picture'),
        'version' => VERSION,
        'js' => array(
          drupal_get_path('module', 'picture') . '/picture.js' => array(
            'type' => 'file',
            'weight' => -10,
            'group' => JS_DEFAULT,
            'scope' => variable_get('picture_js_scope', 'footer'),
            'need_jquery' => FALSE,
          ),
        ),
      );
      $libraries['lazysizes'] = array(
        'title' => t('Lazyload for picture element'),
        'version' => '1.0.1',
        'js' => array(
          drupal_get_path('module', 'picture') . '/lazysizes/lazysizes.js' => array(
            'type' => 'file',
            // file has async, put before all files to prevent JS concatenation breaking
            'weight' => -20,
            'group' => JS_LIBRARY,
            'need_jquery' => FALSE,
          ),
        ),
      );
      $libraries['lazysizes_aspect_ratio'] = array(
        'title' => t('Aspect ratio plugin for lazysizes'),
        'version' => '1.0.1',
        'js' => array(
          drupal_get_path('module', 'picture') . '/lazysizes/plugins/aspectratio/ls.aspectratio.js' => array(
            'type' => 'file',
            'weight' => -10,
            'group' => JS_LIBRARY,
            'need_jquery' => FALSE,
          ),
        ),
        'css' => array(
          drupal_get_path('module', 'picture') . '/lazysizes/plugins/aspectratio/ls.aspectratio.css' => array(
            'type' => 'file',
            'media' => 'screen',
          ),
        ),
      );
      break;
  }
  return $libraries;
}

/**
 * Implements hook_ctools_plugin_directory().
 *
 * Lets CTools know which plugin APIs are implemented by picture module.
 */
function picture_ctools_plugin_directory($owner, $plugin_type) {

  // Load the export_ui plugin.
  if ($owner == 'ctools' && $plugin_type == 'export_ui') {
    return 'ctools/plugins/export_ui';
  }
}

/**
 * Implements hook_theme().
 */
function picture_theme() {
  return array(
    'picture' => array(
      'variables' => array(
        'style_name' => NULL,
        'path' => NULL,
        'uri' => NULL,
        'width' => NULL,
        'height' => NULL,
        'alt' => '',
        'title' => NULL,
        'attributes' => array(),
        'breakpoints' => array(),
        'timestamp' => NULL,
        'lazyload' => NULL,
        'lazyload_aspect_ratio' => NULL,
      ),
    ),
    'picture_formatter' => array(
      'variables' => array(
        'item' => NULL,
        'path' => NULL,
        'image_style' => NULL,
        'breakpoints' => array(),
        'lazyload' => NULL,
        'lazyload_aspect_ratio' => NULL,
      ),
    ),
    'picture_formatter_colorbox' => array(
      'variables' => array(
        'item' => NULL,
        'path' => NULL,
        'image_style' => NULL,
        'breakpoints' => array(),
        'colorbox_group' => array(),
        'colorbox_image_style' => NULL,
        'colorbox_group_id' => NULL,
        'colorbox_caption' => NULL,
      ),
    ),
    'picture_source' => array(
      'variables' => array(
        'srcset' => NULL,
        'media' => NULL,
        'mime_type' => NULL,
        'sizes' => NULL,
        'lazyload' => NULL,
        'lazyload_aspect_ratio' => NULL,
      ),
    ),
    'image_srcset' => array(
      'variables' => array(
        'uri' => NULL,
        'path' => NULL,
        'width' => NULL,
        'height' => NULL,
        'alt' => '',
        'title' => NULL,
        'attributes' => array(),
        'srcset' => array(),
        'sizes' => NULL,
        'lazyload' => NULL,
      ),
    ),
    'picture_sizes_formatter' => array(
      'variables' => array(
        'item' => NULL,
        'path' => NULL,
        'image_styles' => array(),
        'fallback_image_style' => NULL,
        'sizes' => NULL,
        'attributes' => array(),
        'lazyload_data_attributes' => NULL,
        'lazyload_class' => NULL,
      ),
    ),
  );
}

/**
 * Implements hook_field_formatter_info().
 */
function picture_field_formatter_info() {
  $formatters = array();
  $mappings = array_keys(picture_get_mapping_options());
  if ($mappings) {
    $formatters['picture'] = array(
      'label' => t('Picture'),
      'field types' => array(
        'image',
      ),
      'settings' => array(
        'picture_mapping' => reset($mappings),
        'fallback_image_style' => '',
        'lazyload' => '',
        'lazyload_aspect_ratio' => '',
        'image_link' => '',
        'colorbox_settings' => array(
          'colorbox_group' => '',
          'colorbox_gallery' => 'post',
          'colorbox_gallery_custom' => '',
          'colorbox_caption' => 'auto',
          'colorbox_caption_custom' => '',
          'colorbox_multivalue_index' => NULL,
        ),
      ),
    );
  }
  $formatters['picture_sizes_formatter'] = array(
    'label' => t('Image with sizes'),
    'field types' => array(
      'image',
    ),
    'settings' => array(
      'sizes' => '',
      'image_styles' => array(),
      'fallback_image_style' => PICTURE_EMPTY_IMAGE,
      'lazyload' => '',
      'lazyload_aspect_ratio' => '',
      'lazyload_data_attributes' => 0,
      'lazyload_class' => '',
      'image_link' => '',
      'colorbox_settings' => array(
        'colorbox_group' => '',
        'colorbox_gallery' => 'post',
        'colorbox_gallery_custom' => '',
        'colorbox_caption' => 'auto',
        'colorbox_caption_custom' => '',
        'colorbox_multivalue_index' => NULL,
      ),
    ),
  );
  return $formatters;
}

/**
 * Implements hook_field_formatter_settings_form().
 */
function picture_field_formatter_settings_form($field, $instance, $view_mode, $form, &$form_state) {
  $display = $instance['display'][$view_mode];
  $function = "picture_field_formatter_settings_{$display['type']}_form";
  $settings = $display['settings'];
  return $function($field, $instance, $settings);
}

/**
 * Helper function.
 *
 * @see picture_field_formatter_settings_form()
 */
function picture_field_formatter_settings_picture_form($field, $instance, $settings) {
  $options = picture_get_mapping_options();
  if ($options) {
    $element['picture_mapping'] = array(
      '#title' => t('Picture mapping'),
      '#type' => 'select',
      '#default_value' => isset($settings['picture_mapping']) ? $settings['picture_mapping'] : '',
      '#required' => TRUE,
      '#options' => picture_get_mapping_options(),
    );
  }
  else {
    $element['picture_mapping'] = array(
      '#title' => t('Picture mapping'),
      '#type' => 'item',
      '#markup' => t('There are no picture groups defined. !create_link.', array(
        '!create_link' => l(t('Create a picture mapping'), 'admin/config/media/picture/add'),
      )),
    );
  }
  $image_styles = image_style_options(FALSE);
  $element['fallback_image_style'] = array(
    '#title' => t('Fallback image style'),
    '#type' => 'select',
    '#default_value' => $settings['fallback_image_style'],
    '#empty_option' => t('Automatic'),
    '#options' => $image_styles + array(
      PICTURE_EMPTY_IMAGE => t('Empty image'),
      PICTURE_ORIGINAL_IMAGE => t('Original image'),
    ),
  );
  $element['lazyload'] = array(
    '#title' => t('Picture lazyload'),
    '#description' => t('Image will be rendered when it appears in viewport, helps to optimize page load speed.'),
    '#type' => 'checkbox',
    '#default_value' => $settings['lazyload'] ? $settings['lazyload'] : FALSE,
  );
  $element['lazyload_aspect_ratio'] = array(
    '#title' => t('Keep aspect ratio'),
    '#type' => 'checkbox',
    '#description' => t('Preserve the space for the image being lazyloaded to avoid layout reflows. <br /> Image ratio is defined per breakpoint, make sure all images from srcset have the same ratio. <br />Output example: !example', array(
      '!example' => htmlentities('<source media="(...)" data-srcset="image_400x200.jpg x1, image_800x400.jpg x2, image_1200x600.jpg x3" data-aspectratio="2" />'),
    )),
    '#default_value' => !empty($settings['lazyload_aspect_ratio']),
    '#states' => array(
      'visible' => array(
        ':input[name="fields[field_image][settings_edit_form][settings][lazyload]"]' => array(
          'checked' => TRUE,
        ),
        ':input[name="fields[field_image][settings_edit_form][settings][fallback_image_style]"]' => array(
          'value' => PICTURE_EMPTY_IMAGE,
        ),
      ),
    ),
  );
  $link_types = picture_link_types($instance);
  $element['image_link'] = array(
    '#title' => t('Link image to'),
    '#type' => 'select',
    '#default_value' => $settings['image_link'],
    '#empty_option' => t('Nothing'),
    '#options' => $link_types,
    '#attributes' => array(
      'class' => array(
        'picture-image-link',
      ),
    ),
  );
  if (module_exists('colorbox')) {

    // Settings for the colorbox option.
    $element['colorbox_settings'] = array(
      '#type' => 'fieldset',
      '#tree' => TRUE,
      '#title' => t('Colorbox settings'),
      '#collapsed' => FALSE,
      '#collapsible' => TRUE,
      '#states' => array(
        'visible' => array(
          ':input[name$="' . $field['field_name'] . '][settings_edit_form][settings][image_link]"].picture-image-link' => array(
            'value' => 'colorbox',
          ),
        ),
      ),
    );
    $element['colorbox_settings']['colorbox_group'] = array(
      '#title' => t('Colorbox picture mapping'),
      '#type' => 'select',
      '#default_value' => $settings['colorbox_settings']['colorbox_group'],
      '#required' => FALSE,
      '#options' => picture_get_mapping_options(),
    );
    $gallery = array(
      'post' => t('Per post gallery'),
      'page' => t('Per page gallery'),
      'field_post' => t('Per field in post gallery'),
      'field_page' => t('Per field in page gallery'),
      'custom' => t('Custom'),
      'none' => t('No gallery'),
    );
    $element['colorbox_settings']['colorbox_gallery'] = array(
      '#title' => t('Gallery (image grouping)'),
      '#type' => 'select',
      '#default_value' => $settings['colorbox_settings']['colorbox_gallery'],
      '#options' => $gallery,
      '#description' => t('How Colorbox should group the image galleries.'),
      '#attributes' => array(
        'class' => array(
          'picture-colorbox-gallery',
        ),
      ),
    );
    $element['colorbox_settings']['colorbox_gallery_custom'] = array(
      '#title' => t('Custom gallery'),
      '#type' => 'textfield',
      '#maxlength' => 32,
      '#default_value' => $settings['colorbox_settings']['colorbox_gallery_custom'],
      '#description' => t('All images on a page with the same gallery value (rel attribute) will be grouped together. It must only contain lowercase letters, numbers, hyphen and underscores.'),
      '#element_validate' => array(
        'colorbox_gallery_custom_validate',
      ),
      '#required' => FALSE,
      '#states' => array(
        'visible' => array(
          ':input[name$="[settings][colorbox_settings][colorbox_gallery]"].picture-colorbox-gallery' => array(
            'value' => 'custom',
          ),
        ),
      ),
    );
    $caption = array(
      'auto' => t('Automatic'),
      'title' => t('Title text'),
      'alt' => t('Alt text'),
      'node_title' => t('Content title'),
      'custom' => t('Custom (with tokens)'),
      'none' => t('None'),
    );
    $element['colorbox_settings']['colorbox_caption'] = array(
      '#title' => t('Caption'),
      '#type' => 'select',
      '#default_value' => $settings['colorbox_settings']['colorbox_caption'],
      '#options' => $caption,
      '#description' => t('Automatic will use the first none empty value of the title, the alt text and the content title.'),
      '#attributes' => array(
        'class' => array(
          'picture-colorbox-caption',
        ),
      ),
    );
    $element['colorbox_settings']['colorbox_caption_custom'] = array(
      '#title' => t('Custom caption'),
      '#type' => 'textfield',
      '#default_value' => $settings['colorbox_settings']['colorbox_caption_custom'],
      '#states' => array(
        'visible' => array(
          ':input[name$="[settings][colorbox_settings][colorbox_caption]"].picture-colorbox-caption' => array(
            'value' => 'custom',
          ),
        ),
      ),
    );

    // Allow users to hide or set a custom recursion limit.
    // The module token_tweaks sets a global recursion limit that can not be
    // bypassed.
    if (module_exists('token') && ($recursion_limit = min(variable_get('token_tree_recursion_limit', 3), variable_get('colorbox_token_recursion_limit', 3)))) {
      $element['colorbox_settings']['colorbox_token'] = array(
        '#type' => 'fieldset',
        '#title' => t('Replacement patterns'),
        '#theme' => 'token_tree',
        '#token_types' => array(
          $instance['entity_type'],
          'file',
        ),
        '#recursion_limit' => $recursion_limit,
        '#dialog' => TRUE,
        '#states' => array(
          'visible' => array(
            ':input[name$="[settings][colorbox_settings][colorbox_caption]"].picture-colorbox-caption' => array(
              'value' => 'custom',
            ),
          ),
        ),
      );
    }
    else {
      $element['colorbox_settings']['colorbox_token'] = array(
        '#type' => 'fieldset',
        '#title' => t('Replacement patterns'),
        '#description' => '<strong class="error">' . t('For token support the <a href="@token_url">token module</a> must be installed.', array(
          '@token_url' => 'http://drupal.org/project/token',
        )) . '</strong>',
        '#states' => array(
          'visible' => array(
            ':input[name$="[settings][colorbox_settings][colorbox_caption]"].picture-colorbox-caption' => array(
              'value' => 'custom',
            ),
          ),
        ),
      );
    }
  }
  return $element;
}

/**
 * Helper function.
 *
 * @see picture_field_formatter_settings_form()
 */
function picture_field_formatter_settings_picture_sizes_formatter_form($field, $instance, $settings) {
  $element = array();
  if ($instance['entity_type'] == 'file') {
    if (empty($settings['sizes'])) {
      $settings['sizes'] = '(min-width: 0px)';
    }
    $styles = array_filter($settings['image_styles']);
    if (empty($styles)) {
      $settings['image_styles'][PICTURE_EMPTY_IMAGE] = PICTURE_EMPTY_IMAGE;
    }
  }
  $element['sizes'] = array(
    '#title' => t('Sizes'),
    '#type' => 'textfield',
    '#maxlength' => 1024,
    '#description' => t('The value of the sizes attribute. See !link for more information.', array(
      '!link' => l(t('the spec'), 'http://www.whatwg.org/specs/web-apps/current-work/multipage/embedded-content.html#introduction-3:viewport-based-selection-2'),
    )),
    '#default_value' => $settings['sizes'],
    '#required' => TRUE,
  );
  $image_styles = image_style_options(FALSE);
  $image_styles[PICTURE_EMPTY_IMAGE] = t('Empty image');
  $image_styles[PICTURE_ORIGINAL_IMAGE] = t('Original image');
  $element['image_styles'] = array(
    '#title' => t('Image styles'),
    '#type' => 'checkboxes',
    '#default_value' => $settings['image_styles'],
    '#options' => $image_styles,
    '#required' => TRUE,
  );
  $element['fallback_image_style'] = array(
    '#title' => t('Fallback image style'),
    '#type' => 'select',
    '#default_value' => $settings['fallback_image_style'] ? $settings['fallback_image_style'] : PICTURE_EMPTY_IMAGE,
    '#options' => $image_styles + array(
      PICTURE_EMPTY_IMAGE => t('Empty image'),
      PICTURE_ORIGINAL_IMAGE => t('Original image'),
    ),
    '#required' => TRUE,
  );
  $link_types = picture_link_types($instance);
  unset($link_types['colorbox']);
  $element['image_link'] = array(
    '#title' => t('Link image to'),
    '#type' => 'select',
    '#default_value' => $settings['image_link'],
    '#empty_option' => t('Nothing'),
    '#options' => $link_types,
    '#attributes' => array(
      'class' => array(
        'picture-image-link',
      ),
    ),
  );
  $element['lazyload_data_attributes'] = array(
    '#title' => t('Lazyload using data attributes'),
    '#type' => 'select',
    '#default_value' => $settings['lazyload_data_attributes'],
    '#options' => array(
      0 => t('No'),
      1 => t('Yes'),
    ),
  );
  $element['lazyload_class'] = array(
    '#title' => t('Lazyload class'),
    '#type' => 'textfield',
    '#default_value' => $settings['lazyload_class'],
  );
  return $element;
}

/**
 * Returns a list of picture mappings for use in a select list.
 */
function picture_get_mapping_options() {
  $picture_mapping_options = array();
  $picture_mappings = picture_mapping_load_all();
  if ($picture_mappings && !empty($picture_mappings)) {
    foreach ($picture_mappings as $picture_mapping) {

      // Exclude old mappings.
      if ($picture_mapping instanceof PictureMapping && (!isset($picture_mapping->disabled) || !$picture_mapping->disabled) && $picture_mapping
        ->hasMappings()) {
        $picture_mapping_options[$picture_mapping
          ->getMachineName()] = $picture_mapping
          ->label();
      }
    }
  }
  return $picture_mapping_options;
}

/**
 * Implements hook_field_formatter_settings_summary().
 */
function picture_field_formatter_settings_summary($field, $instance, $view_mode) {
  $display = $instance['display'][$view_mode];
  $function = "picture_field_formatter_settings_{$display['type']}_summary";
  $settings = $display['settings'];
  return $function($field, $instance, $settings);
}

/**
 * Helper function.
 *
 * @see picture_field_formatter_settings_summary()
 */
function picture_field_formatter_settings_picture_summary($field, $instance, $settings) {
  $summary = array();
  $picture_mapping = picture_mapping_load($settings['picture_mapping']);
  if ($picture_mapping) {
    $summary[] = t('Picture mapping: @picture_mapping', array(
      '@picture_mapping' => $picture_mapping
        ->label(),
    ));
    $image_styles = image_style_options(FALSE);
    unset($image_styles['']);
    if (isset($image_styles[$settings['fallback_image_style']])) {
      $summary[] = t('Fallback Image style: @style', array(
        '@style' => $image_styles[$settings['fallback_image_style']],
      ));
    }
    else {
      $summary[] = t('Automatic fallback');
    }
    $link_types = picture_link_types($instance);

    // Display this setting only if image is linked.
    if (isset($link_types[$settings['image_link']])) {
      $summary[] = filter_xss_admin($link_types[$settings['image_link']]);
    }
  }
  else {
    $summary[] = t('Select a responsive image mapping.');
  }

  // Display the settings for the colorbox if chosen.
  if ($settings['image_link'] == 'colorbox') {
    $summary[] = t('Colorbox Group: @setting', array(
      '@setting' => $settings['colorbox_settings']['colorbox_group'],
    ));
    $gallery = array(
      'post' => t('Per post gallery'),
      'page' => t('Per page gallery'),
      'field_post' => t('Per field in post gallery'),
      'field_page' => t('Per field in page gallery'),
      'custom' => t('Custom'),
      'none' => t('No gallery'),
    );
    $summary[] = t('Colorbox Gallery: @setting', array(
      '@setting' => $gallery[$settings['colorbox_settings']['colorbox_gallery']],
    ));
    if ($settings['colorbox_settings']['colorbox_gallery'] == 'custom') {
      $summary[] = '&#8594; ' . t('Colorbox Gallery Custom: @setting', array(
        '@setting' => $settings['colorbox_settings']['colorbox_gallery_custom'],
      ));
    }
    $caption = array(
      'auto' => t('Automatic'),
      'title' => t('Title text'),
      'alt' => t('Alt text'),
      'node_title' => t('Content title'),
      'custom' => t('Custom (with tokens)'),
      'none' => t('None'),
    );
    $summary[] = t('Colorbox Caption: @setting', array(
      '@setting' => $caption[$settings['colorbox_settings']['colorbox_caption']],
    ));
    if ($settings['colorbox_settings']['colorbox_caption'] == 'custom') {
      $summary[] = '&#8594; ' . t('Colorbox Caption Custom: @setting', array(
        '@setting' => $settings['colorbox_settings']['colorbox_caption_custom'],
      ));
    }
  }
  return implode('<br />', $summary);
}

/**
 * Helper function.
 *
 * @see picture_field_formatter_settings_form()
 */
function picture_field_formatter_settings_picture_sizes_formatter_summary($field, $instance, $settings) {
  $summary = array();
  $summary[] = t('Sizes: @sizes', array(
    '@sizes' => $settings['sizes'],
  ));
  $image_styles = image_style_options(FALSE);
  unset($image_styles['']);
  $image_styles[PICTURE_EMPTY_IMAGE] = t('Empty image');
  $image_styles[PICTURE_ORIGINAL_IMAGE] = t('Original image');
  $selected_styles = array_filter($settings['image_styles']);
  $summary[] = t('Image styles: @styles', array(
    '@styles' => implode(', ', array_intersect_key($image_styles, $selected_styles)),
  ));
  $summary[] = t('Fallback image style: @style', array(
    '@style' => isset($image_styles[$settings['fallback_image_style']]) ? $image_styles[$settings['fallback_image_style']] : $image_styles[PICTURE_EMPTY_IMAGE],
  ));
  $link_types = picture_link_types($instance);

  // Display this setting only if image is linked.
  if (isset($link_types[$settings['image_link']])) {
    $summary[] = filter_xss_admin($link_types[$settings['image_link']]);
  }
  return implode('<br />', $summary);
}

/**
 * Implements hook_field_formatter_view().
 */
function picture_field_formatter_view($entity_type, $entity, $field, $instance, $langcode, $items, $display) {
  $function = "picture_field_formatter_{$display['type']}_view";
  return $function($entity_type, $entity, $field, $instance, $langcode, $items, $display);
}

/**
 * Helper function.
 *
 * @see picture_field_formatter_view()
 */
function picture_field_formatter_picture_view($entity_type, $entity, $field, $instance, $langcode, $items, $display) {
  $element = array();
  $settings = $display['settings'];

  // Check if the formatter involves a link.
  $image_link = $display['settings']['image_link'];
  if ($image_link == 'content') {
    if (isset($entity->referencing_entity) && isset($entity->referencing_entity_type)) {
      $uri = entity_uri($entity->referencing_entity_type, $entity->referencing_entity);
    }
    else {
      $uri = entity_uri($entity_type, $entity);
    }
  }
  elseif ($image_link == 'file') {
    $link_file = TRUE;
  }
  elseif ($image_link) {
    if (isset($entity->{$image_link})) {

      // Support for field translations.
      $language = field_language($entity_type, $entity, $image_link, $langcode);
      $link_field = $entity->{$image_link};
      if (isset($link_field[$language])) {
        $link_values = $link_field[$language];
      }
    }
    elseif (image_style_load($image_link)) {
      $link_file = $image_link;
    }
  }
  $fallback_image_style = $settings['fallback_image_style'];

  // Support old display settings from 7.x-1.x.
  $mapping_name = isset($display['settings']['picture_mapping']) && !empty($display['settings']['picture_mapping']) ? $display['settings']['picture_mapping'] : $display['settings']['picture_group'];

  // If we haven't saved a picture mapping for this field previously and
  // this isn't a field formatter screen (i.e. panels),
  // load a default mapping.
  $picture_mapping = FALSE;
  if (isset($mapping_name) && !empty($mapping_name)) {
    $picture_mapping = picture_mapping_load($mapping_name);
    if (!$picture_mapping) {
      trigger_error(check_plain("Unable to load picture mapping {$mapping_name}."), E_USER_ERROR);
      return $element;
    }
  }
  else {
    $all_mappings = picture_mapping_load_all();
    $picture_mapping = reset($all_mappings);
    if (!$picture_mapping) {
      trigger_error("No picture mappings have been defined yet.", E_USER_ERROR);
      return $element;
    }
  }
  $breakpoint_styles = picture_get_mapping_breakpoints($picture_mapping, $fallback_image_style);

  // Assume regular display.
  $formatter = 'picture_formatter';
  $colorbox_breakpoints = array();
  $colorbox_fallback_image_style = '';

  // Check for colorbox link.
  if (module_exists('colorbox') && $display['settings']['image_link'] == 'colorbox') {
    $formatter = 'picture_formatter_colorbox';
    $mappings = picture_mapping_load($display['settings']['colorbox_settings']['colorbox_group']);
    if (!$mappings) {
      trigger_error(check_plain("Unable to load picture mapping {$display['settings']['colorbox_settings']['colorbox_group']}."), E_USER_ERROR);
      return $element;
    }
    $colorbox_breakpoints = picture_get_mapping_breakpoints($mappings, $colorbox_fallback_image_style);
  }
  foreach ($items as $delta => $item) {
    if (isset($link_file)) {
      $uri = array(
        'path' => $link_file === TRUE ? file_create_url($item['uri']) : image_style_url($link_file, $item['uri']),
        'options' => array(),
      );
    }

    // Handle multiple link with image values.
    if (isset($link_values)) {
      if (isset($link_values[$delta]['url'])) {
        $uri = array(
          'path' => $link_values[$delta]['url'],
          'options' => array(
            'attributes' => $link_values[$delta]['attributes'],
          ),
        );

        // Handle query fragment if there is any.
        if (!empty($link_values[$delta]['query'])) {
          $uri['options']['query'] = $link_values[$delta]['query'];
        }
      }
      else {
        unset($uri);
      }
    }
    $libraries = array(
      array(
        'picture',
        'picturefill_head',
      ),
      array(
        'picture',
        'picturefill',
      ),
      array(
        'picture',
        'picture.ajax',
      ),
    );
    if (!empty($settings['lazyload'])) {
      $libraries[] = array(
        'picture',
        'lazysizes',
      );
      if (!empty($display['settings']['lazyload_aspect_ratio'])) {
        $libraries[] = array(
          'picture',
          'lazysizes_aspect_ratio',
        );
      }
    }
    $element[$delta] = array(
      '#theme' => $formatter,
      '#attached' => array(
        'library' => $libraries,
      ),
      '#item' => $item,
      '#image_style' => $fallback_image_style,
      '#breakpoints' => $breakpoint_styles,
      '#path' => isset($uri) ? $uri : '',
      '#colorbox_group' => $colorbox_breakpoints,
      '#colorbox_image_style' => $colorbox_fallback_image_style,
      '#lazyload' => !empty($settings['lazyload']),
      '#lazyload_aspect_ratio' => !empty($settings['lazyload_aspect_ratio']),
    );

    // Add css and js for colorbox.
    if ($formatter == 'picture_formatter_colorbox') {
      $element[$delta]['#attached']['css'][drupal_get_path('module', 'picture') . '/picture_colorbox.css'] = array(
        'type' => 'file',
      );
      $element[$delta]['#attached']['js'][drupal_get_path('module', 'picture') . '/picture_colorbox.js'] = array(
        'type' => 'file',
      );
      if (!variable_get('colorbox_inline', 0)) {
        $element[$delta]['#attached']['js'][drupal_get_path('module', 'colorbox') . '/js/colorbox_inline.js'] = array(
          'type' => 'file',
        );
      }
      $colorbox_settings = $display['settings']['colorbox_settings'];

      // Add the group ID.
      list($id) = entity_extract_ids($entity_type, $entity);
      $entity_id = !empty($id) ? $entity_type . '-' . $id : 'entity-id';

      // If this is a file entity field check for the referencing entity to do
      // the proper grouping.
      if (isset($entity->referencing_entity) && isset($entity->referencing_field)) {

        // Because we don't have the entity type we use some "magic" to get a
        // unique entity identifier.
        $entity_id = spl_object_hash($entity->referencing_entity);
        $colorbox_group_field = $entity->referencing_field;
      }
      else {
        $colorbox_group_field = $field['field_name'];
      }
      switch ($colorbox_settings['colorbox_gallery']) {
        case 'post':
          $gallery_id = 'gallery-' . $entity_id;
          break;
        case 'page':
          $gallery_id = 'gallery-all';
          break;
        case 'field_post':
          $gallery_id = 'gallery-' . $entity_id . '-' . $colorbox_group_field;
          break;
        case 'field_page':
          $gallery_id = 'gallery-' . $colorbox_group_field;
          break;
        case 'custom':
          $gallery_id = $colorbox_settings['colorbox_gallery_custom'];
          break;
        default:
          $gallery_id = '';
      }
      $element[$delta]['#colorbox_group_id'] = $gallery_id;

      // Add the caption.
      $entity_title = entity_label($entity_type, $entity);
      switch ($colorbox_settings['colorbox_caption']) {
        case 'auto':

          // If the title is empty use alt or the entity title in that order.
          if (!empty($item['title'])) {
            $caption = $item['title'];
          }
          elseif (!empty($item['alt'])) {
            $caption = $item['alt'];
          }
          elseif (!empty($entity_title)) {
            $caption = $entity_title;
          }
          else {
            $caption = '';
          }
          break;
        case 'title':
          $caption = $item['title'];
          break;
        case 'alt':
          $caption = $item['alt'];
          break;
        case 'node_title':
          $caption = $entity_title;
          break;
        case 'custom':
          $caption = token_replace($colorbox_settings['colorbox_caption_custom'], array(
            $entity_type => $entity,
            'file' => (object) $item,
          ), array(
            'clear' => TRUE,
          ));
          break;
        default:
          $caption = '';
      }

      // Shorten the caption for the example styles or when caption shortening
      // is active.
      $colorbox_style = variable_get('colorbox_style', 'default');
      $trim_length = variable_get('colorbox_caption_trim_length', 75);
      if ((strpos($colorbox_style, 'colorbox/example') !== FALSE || variable_get('colorbox_caption_trim', 0)) && drupal_strlen($caption) > $trim_length) {
        $caption = drupal_substr($caption, 0, $trim_length - 5) . '...';
      }
      $element[$delta]['#colorbox_caption'] = $caption;
    }
  }
  return $element;
}

/**
 * Helper function.
 *
 * @see picture_field_formatter_view()
 */
function picture_field_formatter_picture_sizes_formatter_view($entity_type, $entity, $field, $instance, $langcode, $items, $display) {
  $elements = array();
  $settings = $display['settings'];

  // Check if the formatter involves a link.
  $image_link = $display['settings']['image_link'];
  if ($image_link == 'content') {
    $uri = entity_uri($entity_type, $entity);
  }
  elseif ($image_link == 'file') {
    $link_file = TRUE;
  }
  elseif ($image_link) {
    if (isset($entity->{$image_link})) {

      // Support for field translations.
      $language = field_language($entity_type, $entity, $image_link, $langcode);
      $link_field = $entity->{$image_link};
      if (isset($link_field[$language])) {
        $link_values = $link_field[$language];
      }
    }
    elseif (image_style_load($image_link)) {
      $link_file = $image_link;
    }
  }
  foreach ($items as $delta => $item) {
    if (isset($link_file)) {
      $uri = array(
        'path' => $link_file === TRUE ? file_create_url($item['uri']) : image_style_url($link_file, $item['uri']),
        'options' => array(),
      );
    }

    // Handle multiple link with image values.
    if (isset($link_values)) {
      if (isset($link_values[$delta]['url'])) {
        $uri = array(
          'path' => $link_values[$delta]['url'],
          'options' => array(
            'attributes' => $link_values[$delta]['attributes'],
          ),
        );

        // Handle query fragment if there is any.
        if (!empty($link_values[$delta]['query'])) {
          $uri['options']['query'] = $link_values[$delta]['query'];
        }
      }
      else {
        unset($uri);
      }
    }
    $elements[$delta] = array(
      '#theme' => 'picture_sizes_formatter',
      '#item' => $item,
      '#image_styles' => array_filter($settings['image_styles']),
      '#fallback_image_style' => $settings['fallback_image_style'] ? $settings['fallback_image_style'] : PICTURE_EMPTY_IMAGE,
      '#sizes' => $settings['sizes'],
      '#path' => isset($uri) ? $uri : '',
      '#lazyload_data_attributes' => $settings['lazyload_data_attributes'],
      '#lazyload_class' => $settings['lazyload_class'],
    );
  }
  return $elements;
}

/**
 * Theme picture formatter.
 */
function theme_picture_formatter($variables) {
  if (!isset($variables['breakpoints']) || empty($variables['breakpoints'])) {
    $image_formatter = array(
      '#theme' => 'image_formatter',
      '#item' => $variables['item'],
      '#image_style' => $variables['image_style'],
      '#path' => $variables['path'],
    );
    return drupal_render($image_formatter);
  }
  $item = $variables['item'];
  $responsive_image = array(
    '#theme' => 'picture',
    '#width' => isset($item['width']) ? $item['width'] : NULL,
    '#height' => isset($item['height']) ? $item['height'] : NULL,
    '#style_name' => $variables['image_style'],
    '#breakpoints' => $variables['breakpoints'],
    '#uri' => $item['uri'],
    '#alt' => isset($item['alt']) ? $item['alt'] : '',
    '#attributes' => isset($item['attributes']) ? $item['attributes'] : NULL,
    '#timestamp' => isset($item['timestamp']) ? $item['timestamp'] : NULL,
    '#lazyload' => !empty($variables['lazyload']),
    '#lazyload_aspect_ratio' => !empty($variables['lazyload_aspect_ratio']),
  );
  if (isset($item['title']) && drupal_strlen($item['title']) != 0) {
    $responsive_image['#title'] = $item['title'];
  }
  if (!empty($item['class'])) {
    $class = !empty($responsive_image['#attributes']['class']) ? $responsive_image['#attributes']['class'] : array();
    if (!is_array($class)) {
      $class = array(
        $class,
      );
    }
    $responsive_image['#attributes']['class'] = $class + $item['class'];
  }
  if (is_array($variables['path']) && isset($variables['path']['path'])) {
    $path = $variables['path']['path'];
    $options = isset($variables['path']['options']) ? $variables['path']['options'] : array();
    $options['html'] = TRUE;
    return l(drupal_render($responsive_image), $path, $options);
  }
  return drupal_render($responsive_image);
}

/**
 * Theme pictue sizes formatter.
 */
function theme_picture_sizes_formatter($variables) {
  $output_method = variable_get('picture_img_sizes_output_method', 'picture_element');
  $variables['image_style'] = $variables['fallback_image_style'];
  if ($output_method == 'picture_element') {
    $variables['breakpoints'][BREAKPOINTS_SOURCE_TYPE_MODULE . '.picture.empty_srcset']['1x'] = array(
      'mapping_type' => 'sizes',
      'sizes' => $variables['sizes'],
      'sizes_image_styles' => $variables['image_styles'],
      'lazyload' => !empty($variables['lazyload']),
      'lazyload_aspect_ratio' => !empty($variables['lazyload_aspect_ratio']),
    );
    return theme('picture_formatter', $variables);
  }
  if ($output_method == 'img_srcset') {
    $image = $variables['item'];

    // Map image attributes.
    $variables['uri'] = _picture_image_style_url($variables['fallback_image_style'], $image['uri']);
    if (!empty($image['alt'])) {
      $variables['alt'] = $image['alt'];
    }
    if (!empty($image['title'])) {
      $variables['title'] = $image['title'];
    }
    if (!empty($image['attributes'])) {
      $variables['attributes'] = $image['attributes'];
    }

    // Check for lazyload_data_attributes.
    if (isset($variables['lazyload_data_attributes']) && $variables['lazyload_data_attributes']) {
      if (!isset($variables['attributes'])) {
        $variables['attributes'] = array();
      }
      if (!isset($variables['attributes']['class'])) {
        $variables['attributes']['class'] = array();
      }
      $variables['attributes']['class'] = array_merge($variables['attributes']['class'], explode(',', $variables['lazyload_class']));
    }

    // Set up image source options.
    $variables['srcset'] = array();

    // Get the dimensions of the original image.
    $image_info = image_get_info($image['uri']);
    foreach ($variables['image_styles'] as $image_style) {
      $image_style_uri = _picture_image_style_url($image_style, $image['uri']);
      $dimensions = array(
        'width' => $image_info['width'],
        'height' => $image_info['height'],
      );

      // Let the the image style transform the dimensions. This way we avoid the
      // performance cost of actually having to generate the derivative. The
      // image style can transform the dimensions without the derivative having
      // to exist.
      $dimensions = picture_get_image_dimensions($image_style, $dimensions);
      $variables['srcset'][] = array(
        'uri' => $image_style_uri,
        'width' => $dimensions['width'] . 'w',
      );
    }
    $output = theme('image_srcset', $variables);
    if (isset($variables['lazyload_data_attributes']) && $variables['lazyload_data_attributes']) {
      $output = str_replace(array(
        'src="',
        'srcset="',
      ), array(
        'data-src="',
        'data-srcset="',
      ), $output);
    }
    return $output;
  }
}

/**
 * Theme function to add support for colorbox.
 */
function theme_picture_formatter_colorbox($variables) {
  if (!isset($variables['breakpoints']) || empty($variables['breakpoints'])) {
    return theme('image_formatter', $variables);
  }
  $item = $variables['item'];

  // Do not output an empty 'title' attribute.
  if (isset($item['title']) && drupal_strlen($item['title']) == 0) {
    unset($item['title']);
  }
  $item['style_name'] = $variables['image_style'];
  $item['breakpoints'] = $variables['breakpoints'];
  if (!isset($item['path']) && isset($variables['uri'])) {
    $item['path'] = $variables['uri'];
  }
  $output = theme('picture', $item);

  // If an error happened skip further processing.
  if (empty($output)) {
    return $output;
  }
  if (isset($variables['colorbox_group'])) {
    $item['breakpoints'] = $variables['colorbox_group'];
    $item['style_name'] = $variables['colorbox_image_style'];
    $id = 'picture-colorbox-' . user_password();
    $colorbox = '<div style="display: none;"><div id="' . $id . '" class="picture-colorbox-container">' . theme('picture', $item) . '</div></div>';
    $options = array(
      'attributes' => array(
        'class' => array(
          'colorbox-inline',
        ),
      ),
      'query' => array(
        'maxWidth' => '80%',
        'maxHeight' => '80%',
        'inline' => 'true',
      ),
      'fragment' => $id,
      'html' => TRUE,
    );
    if (!empty($variables['colorbox_group_id'])) {
      $options['attributes']['rel'] = $variables['colorbox_group_id'];
    }
    if (!empty($variables['colorbox_caption'])) {
      $options['query']['title'] = $variables['colorbox_caption'];
    }

    // Do not load picture automatically.
    $colorbox = str_replace('<picture', '<span lazyload="lazyload"', $colorbox);
    $colorbox = str_replace('</picture>', '</span>', $colorbox);
    $colorbox = str_replace(' srcset="', ' data-srcset="', $colorbox);
    $output = l($output, current_path(), $options) . $colorbox;
  }
  return $output;
}

/**
 * Returns HTML for a picture.
 *
 * @param array $variables
 *   An associative array containing:
 *   - uri: Either the path of the image file (relative to base_path()) or a
 *     full URL.
 *   - width: The width of the image (if known).
 *   - height: The height of the image (if known).
 *   - alt: The alternative text for text-based browsers.
 *   - breakpoints: An array containing breakpoints.
 *   - attributes: An array containing attributes.
 *
 * @ingroup themeable
 */
function theme_picture(array $variables) {
  $image_styles = image_styles();

  // Make sure that width and height are proper values.
  // If they exists we'll output them.
  // @see http://www.w3.org/community/respimg/2012/06/18/florians-compromise/
  if (isset($variables['width']) && empty($variables['width'])) {
    unset($variables['width']);
    unset($variables['height']);
  }
  elseif (isset($variables['height']) && empty($variables['height'])) {
    unset($variables['width']);
    unset($variables['height']);
  }
  if (isset($variables['metadata']['width']) && isset($variables['metadata']['height'])) {
    $variables['width'] = $variables['metadata']['width'];
    $variables['height'] = $variables['metadata']['height'];
  }
  $sources = array();
  $output = array();

  // If we know width and height we use them, if not we should get image info.
  if (!empty($variables['width']) && !empty($variables['height'])) {
    $dimensions = array(
      'width' => $variables['width'],
      'height' => $variables['height'],
    );
  }
  else {
    if (($image_info = image_get_info($variables['uri'])) === FALSE) {
      watchdog('Picture', 'Unable to load image: %image', array(
        '%image' => $variables['uri'],
      ));

      // If the image couldn't be loaded return nothing.
      return NULL;
    }
    $dimensions = array(
      'width' => $image_info['width'],
      'height' => $image_info['height'],
    );
  }

  // All breakpoints and multipliers.
  foreach ($variables['breakpoints'] as $breakpoint_name => $multipliers) {
    $breakpoint = breakpoints_breakpoint_load_by_fullkey($breakpoint_name);
    if ($breakpoint) {
      $sizes = array();
      $srcset = array();
      foreach ($multipliers as $multiplier => $mapping_definition) {
        switch ($mapping_definition['mapping_type']) {
          case 'sizes':
            foreach (array_filter($mapping_definition['sizes_image_styles']) as $image_style_name) {
              $dimensions_clone = $dimensions;
              picture_get_image_dimensions($image_style_name, $dimensions_clone);

              // Get mime type.
              // $derivative_mime_type = $mime_type;
              // $image_style->transformMimeType($derivative_mime_type);
              // $derivative_mime_types[] = $derivative_mime_type;
              $srcset[$dimensions_clone['width']] = _picture_image_style_url($image_style_name, $variables['uri'], $variables['timestamp']) . ' ' . $dimensions_clone['width'] . 'w';
              $sizes = array_merge(explode(',', $mapping_definition['sizes']), $sizes);
            }
            break;
          case 'image_style':

            // Get mime type (not implemented in Drupal 7.
            // $derivative_mime_type = $mime_type;
            // $image_style->transformMimeType($derivative_mime_type);
            // $derivative_mime_types[] = $derivative_mime_type;
            $srcset[$multiplier] = _picture_image_style_url($mapping_definition['image_style'], $variables['uri'], $variables['timestamp']) . ' ' . $multiplier;
            break;
        }
      }

      // Sort srcset.
      ksort($srcset);
      $aspect_ratio = '';
      if (!empty($variables['lazyload_aspect_ratio']) && !empty($image_styles[$mapping_definition['image_style']])) {
        $dimensions = array(
          'width' => $variables['width'],
          'height' => $variables['height'],
        );
        $dimensions = picture_get_image_dimensions($mapping_definition['image_style'], $dimensions);

        // store as a string, JS plugin will do calculation
        $aspect_ratio = $dimensions['width'] . '/' . $dimensions['height'];
      }
      $sources[] = array(
        '#theme' => 'picture_source',
        '#srcset' => implode(', ', array_unique($srcset)),
        '#media' => $breakpoint->breakpoint,
        '#sizes' => implode(', ', array_unique($sizes)),
        '#lazyload' => !empty($variables['lazyload']),
        '#lazyload_aspect_ratio' => $aspect_ratio,
      );
    }
  }
  $attributes = array();
  foreach (array(
    'alt',
    'title',
    'attributes',
  ) as $key) {
    $field = sprintf('field_file_image_%s_text', $key);
    if (isset($variables[$key]) && !empty($variables[$key])) {
      if ($key == 'attributes') {
        $attributes += $variables[$key];
      }
      else {
        $attributes[$key] = $variables[$key];
      }
    }
    elseif (isset($variables[$field]) && is_array($variables[$field]) && isset($variables[$field][LANGUAGE_NONE][0]['safe_value'])) {
      if ($key == 'attributes') {
        $attributes += $variables[$field][LANGUAGE_NONE][0]['safe_value'];
      }
      else {
        $attributes[$key] = $variables[$field][LANGUAGE_NONE][0]['safe_value'];
      }
    }
  }
  if (!empty($sources)) {
    $output[] = '<picture ' . drupal_attributes(array_diff_key($attributes, array(
      'alt' => '',
    ))) . '>';
    $output[] = '<!--[if IE 9]><video style="display: none;"><![endif]-->';
    $output = array_merge($output, array_map('drupal_render', $sources));
    $output[] = '<!--[if IE 9]></video><![endif]-->';
    $src = _picture_image_style_url($variables['style_name'], $variables['uri'], $variables['timestamp']);
    if (!empty($variables['lazyload'])) {
      if (!isset($attributes['class'])) {
        $attributes['class'] = array();
      }
      elseif (!is_array($attributes['class'])) {
        $attributes['class'] = (array) $attributes['class'];
      }
      $attributes['class'][] = 'lazyload';
    }
    if (!empty($variables['lazyload_aspect_ratio'])) {

      // img tag has to have data-aspectratio attribute even if its dummy placeholder
      $attributes['data-aspectratio'] = '';
    }
    if (variable_get('picture_fallback_method', 'src') === 'src') {
      $min_ie9_fallback = array(
        '#theme' => 'image_srcset',
        '#uri' => $src,
        '#alt' => isset($attributes['alt']) ? $attributes['alt'] : '',
        '#title' => isset($attributes['title']) ? $attributes['title'] : '',
        '#attributes' => array_diff_key($attributes, array(
          'alt' => '',
          'title' => '',
        )),
        '#lazyload' => !empty($variables['lazyload']),
      );
      $output[] = drupal_render($min_ie9_fallback);
    }
    else {

      // Fallback image for < IE9.
      $min_ie9_fallback = array(
        '#theme' => 'image_srcset',
        '#uri' => $src,
        '#alt' => isset($attributes['alt']) ? $attributes['alt'] : '',
        '#title' => isset($attributes['title']) ? $attributes['title'] : '',
        '#attributes' => array_diff_key($attributes, array(
          'alt' => '',
          'title' => '',
        )),
        '#lazyload' => !empty($variables['lazyload']),
      );
      $output[] = '<!--[if lt IE 9]>';
      $output[] = drupal_render($min_ie9_fallback);
      $output[] = '<![endif]-->';

      // Fallback image for > IE8.
      $srcset = array(
        'uri' => $src,
      );
      $dimensions_clone = $dimensions;
      picture_get_image_dimensions($variables['style_name'], $dimensions_clone);
      if (!empty($dimensions_clone['width'])) {
        $srcset['width'] = $dimensions_clone['width'] . 'w';
      }
      $plus_ie8_fallback = array(
        '#theme' => 'image_srcset',
        '#srcset' => array(
          $srcset,
        ),
        '#alt' => isset($attributes['alt']) ? $attributes['alt'] : '',
        '#title' => isset($attributes['title']) ? $attributes['title'] : '',
        '#attributes' => array_diff_key($attributes, array(
          'alt' => '',
          'title' => '',
        )),
        '#lazyload' => !empty($variables['lazyload']),
      );
      $output[] = '<!--[if !lt IE 9]><!-->';
      $output[] = drupal_render($plus_ie8_fallback);
      $output[] = '<!-- <![endif]-->';
    }
    $output[] = '</picture>';
    return implode("\n", $output);
  }
}

/**
 * Returns HTML for an image.
 *
 * @param array $variables
 *   An associative array containing:
 *   - path: Either the path of the image file (relative to base_path()) or a
 *     full URL.
 *   - srcset: Array of multiple URI's and sizes/multipliers.
 *   - sizes: The value for the sizes attribute.
 *   - width: The width of the image (if known).
 *   - height: The height of the image (if known).
 *   - alt: The alternative text for text-based browsers. HTML 4 and XHTML 1.0
 *     always require an alt attribute. The HTML 5 draft allows the alt
 *     attribute to be omitted in some cases. Therefore, this variable defaults
 *     to an empty string, but can be set to NULL for the attribute to be
 *     omitted. Usually, neither omission nor an empty string satisfies
 *     accessibility requirements, so it is strongly encouraged for code
 *     calling theme('image') to pass a meaningful value for this variable.
 *     - http://www.w3.org/TR/REC-html40/struct/objects.html#h-13.8
 *     - http://www.w3.org/TR/xhtml1/dtds.html
 *     - http://dev.w3.org/html5/spec/Overview.html#alt
 *   - 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.
 */
function theme_image_srcset(array $variables) {

  // Sizes is mandatory, so make sure the key is set, default is 100vw.
  if (!isset($variables['sizes']) || empty($variables['sizes'])) {
    $attributes['sizes'] = '100vw';
  }

  // Make sure we have an array.
  $attributes = isset($variables['attributes']) ? $variables['attributes'] : array();
  if (isset($variables['uri']) && !empty($variables['uri'])) {
    $attr_name = !empty($variables['lazyload']) ? 'data-src' : 'src';

    // TODO: remove when https://www.drupal.org/node/1342504 is fixed.
    if (strpos($variables['uri'], 'data:') === 0) {
      $attributes[$attr_name] = $variables['uri'];
    }
    else {
      $attributes[$attr_name] = file_create_url($variables['uri']);
    }
  }
  if (isset($variables['srcset']) && !empty($variables['srcset'])) {
    $srcsets = array();
    foreach ($variables['srcset'] as $src) {

      // URI is mandatory.
      if (isset($src['uri']) && !empty($src['uri'])) {
        $key = '';

        // TODO: remove when https://www.drupal.org/node/1342504 is fixed.
        if (strpos($src['uri'], 'data:') === 0) {
          $source = $src['uri'];
        }
        else {
          $source = file_create_url($src['uri']);
        }
        if (isset($src['width']) && !empty($src['width'])) {
          $source .= ' ' . $src['width'];
          $key = (int) drupal_substr($src['width'], 0, -1);
        }
        elseif (isset($src['multiplier']) && !empty($src['multiplier'])) {
          $source .= ' ' . $src['multiplier'];

          // Convert 1x to 100, 1.5x to 150  and so on.
          $key = (int) ((double) drupal_substr($src['multiplier'], 0, -1) * 100);
        }
        $srcsets[$key] = $source;
      }
    }
    ksort($srcsets);
    $attr_name = !empty($variables['lazyload']) ? 'data-srcset' : 'srcset';
    $attributes[$attr_name] = implode(', ', $srcsets);
  }
  foreach (array(
    'width',
    'height',
    'alt',
    'title',
    'sizes',
  ) as $key) {
    if (isset($variables[$key])) {
      $attributes[$key] = $variables[$key];
    }
  }
  if (isset($variables['path']) && !empty($variables['path'])) {
    return l('<img ' . drupal_attributes($attributes) . ' />', $variables['path']['path'], array(
      'html' => TRUE,
    ) + $variables['path']['options']);
  }
  return '<img ' . drupal_attributes($attributes) . ' />';
}

/**
 * Returns HTML for a source tag.
 *
 * @param array $variables
 *   An associative array containing:
 *   - media: The media query to use.
 *   - src: Either the path of the image file (relative to base_path()) or a
 *     full URL.
 *   - dimensions: The width and height of the image (if known).
 *
 * @ingroup themeable
 */
function theme_picture_source(array $variables) {
  if (!empty($variables['lazyload'])) {
    $attributes['data-srcset'] = $variables['srcset'];
    if (!empty($variables['lazyload_aspect_ratio'])) {
      $attributes['data-aspectratio'] = $variables['lazyload_aspect_ratio'];
    }
  }
  else {
    $attributes['srcset'] = $variables['srcset'];
  }
  if (isset($variables['media']) && !empty($variables['media'])) {
    $attributes['media'] = $variables['media'];
  }
  if (isset($variables['mime_type']) && !empty($variables['mime_type'])) {
    $attributes['type'] = $variables['mime_type'];
  }
  if (isset($variables['sizes']) && !empty($variables['sizes'])) {
    $attributes['sizes'] = $variables['sizes'];
  }
  return '<source' . drupal_attributes($attributes) . ' />';
}

/**
 * Determines the dimensions of an image.
 *
 * @param string $image_style_name
 *   The name of the style to be used to alter the original image.
 * @param array $dimensions
 *   Dimensions to be modified - an array with components width and height, in
 *   pixels.
 */
function picture_get_image_dimensions($image_style_name, array &$dimensions) {
  if ($image_style_name == PICTURE_EMPTY_IMAGE) {
    $dimensions = array(
      'width' => 1,
      'height' => 1,
    );
  }
  elseif ($image_style_name == PICTURE_ORIGINAL_IMAGE) {

    // NOOP.
  }
  else {
    image_style_transform_dimensions($image_style_name, $dimensions);
  }
  return $dimensions;
}

/**
 * Implements hook_filter_info().
 */
function picture_filter_info() {
  $filters = array();
  $filters['picture'] = array(
    'title' => t('Make images responsive with the picture module'),
    'description' => t('Replace img tags with markup that contains media width breakpoints. The appropriate image file size will be chosen.'),
    'process callback' => '_picture_filter_process',
    'tips callback' => '_picture_filter_tips',
    'type' => 'FILTER_TYPE_TRANSFORM_REVERSIBLE',
  );
  return $filters;
}

/**
 * Process callback for inline image filter.
 */
function _picture_filter_process($text, $filter) {

  // Find all img tags with a data-picture-mapping attribute.
  preg_match_all('/<img [^>]*data-picture-(mapping|group)\\s*=\\s*["\'][^>]*>/i', $text, $images);
  if (!empty($images)) {
    foreach ($images[0] as $image) {

      // Create the render array expected by theme_picture_formatter.
      $image_render_array = _picture_filter_prepare_image($image);
      if (!$image_render_array) {
        continue;
      }

      // Get the responsive markup for this image.
      $new_markup = drupal_render($image_render_array);

      // Replace the original img tag with the responsive markup.
      $text = str_replace($image, $new_markup, $text);
    }
  }
  return $text;
}

/**
 * Prepares a Render Array for theme_picture_formatter().
 *
 * It is similar to picture_field_formatter_view()
 * with modifications for inline images.
 *
 * @param string $image
 *   An img tag.
 *
 * @see picture_field_formatter_view()
 */
function _picture_filter_prepare_image($image) {

  // Make sure the closing tag is right.
  $image = str_replace('/>', '>', $image);
  $image = str_replace('>', ' />', $image);
  $image = str_replace("&nbsp;", '', $image);
  $image = htmlspecialchars($image);

  // Parse the tag as xml.
  $xml = simplexml_load_string('<image>' . html_entity_decode($image, ENT_QUOTES, "utf-8") . '</image>');
  if (isset($xml->img[0]) && is_object($xml->img[0])) {
    $attributes = array();
    foreach ($xml->img[0]
      ->attributes() as $a => $b) {
      $attributes[$a] = (string) $b;
    }
  }
  $fallback_image_style = '';
  if (isset($attributes['data-picture-mapping'])) {
    $mapping_id = $attributes['data-picture-mapping'];
  }
  elseif (isset($attributes['data-picture-group'])) {
    $mapping_id = $attributes['data-picture-group'];
  }
  else {
    $mapping_id = null;
  }
  $mappings = picture_mapping_load($mapping_id);

  // Make sure we have valid mappings.
  if (empty($mappings)) {
    return FALSE;
  }
  $breakpoint_styles = picture_get_mapping_breakpoints($mappings, $fallback_image_style);

  // Make sure we have a src attribute.
  if (!isset($attributes['src'])) {
    return FALSE;
  }
  $src = $attributes['src'];
  unset($attributes['src']);
  $alt = isset($attributes['alt']) ? $attributes['alt'] : '';
  unset($attributes['alt']);
  $title = isset($attributes['title']) ? $attributes['title'] : '';
  unset($attributes['title']);

  // Ensure that class attributes are properly split into array.
  if (!empty($attributes['class']) && !is_array($attributes['class'])) {
    $attributes['class'] = array_filter(explode(' ', $attributes['class']));
  }

  // Make sure we have map src to uri.
  $uri = picture_image_uri($src);
  if (!$uri) {
    return FALSE;
  }
  $image_info = image_get_info($uri);

  // Check if stage_file_proxy is active.
  if (!$image_info && module_exists('stage_file_proxy')) {
    $local = stage_file_proxy_process_file_uri($uri);
    $image_info = image_get_info($uri);
  }
  if (!$image_info) {

    // It's not an image.
    return FALSE;
  }
  $picture_mappings = variable_get('picture_ckeditor_mappings', array());

  // Use srcset if possible and allowed.
  if (variable_get('picture_use_sizes_in_wysiwyg', FALSE)) {
    $bps = picture_get_mapping_breakpoints($mappings);
    $bp = reset($bps);
    $info = reset($bp);
    if (count($bps) == 1 && count($bp) == 1 && $info['mapping_type'] === 'image_style') {
      $image_render_array = array(
        '#theme' => 'image_style',
        '#path' => $uri,
        '#style_name' => $info['image_style'],
        '#alt' => $alt,
        '#title' => $title,
        '#attributes' => array(
          'data-picture-mapping' => $mapping_id,
        ) + $attributes,
      );
      return $image_render_array;
    }
    if (count($bps) == 1 && count($bp) == 1 && $info['mapping_type'] === 'sizes') {
      $image_styles = $info['sizes_image_styles'];
      $image_render_array = array(
        '#theme' => 'image_srcset',
        '#sizes' => $info['sizes'],
      );

      // Map image attributes.
      $image_render_array['#uri'] = _picture_image_style_url($picture_mappings[$mapping_id]['fallback'], $uri);

      // Set up image source options.
      $image_render_array['#srcset'] = array();
      foreach ($image_styles as $image_style) {
        $image_style_uri = _picture_image_style_url($image_style, $uri);
        $dimensions = array(
          'width' => $image_info['width'],
          'height' => $image_info['height'],
        );

        // Let the the image style transform the dimensions. This way we avoid the
        // performance cost of actually having to generate the derivative. The
        // image style can transform the dimensions without the derivative having
        // to exist.
        $dimensions = picture_get_image_dimensions($image_style, $dimensions);
        $image_render_array['#srcset'][] = array(
          'uri' => $image_style_uri,
          'width' => $dimensions['width'] . 'w',
        );
      }
      $image_render_array['#attributes'] = array(
        'data-picture-mapping' => $mapping_id,
      ) + $attributes;
      return $image_render_array;
    }
  }
  $image_render_array = array(
    '#theme' => 'picture',
    '#style_name' => $picture_mappings[$mapping_id]['fallback'],
    '#uri' => $uri,
    '#width' => $image_info['width'],
    '#height' => $image_info['height'],
    '#alt' => $alt,
    '#title' => $title,
    '#attributes' => array(
      'data-picture-mapping' => $mapping_id,
    ) + $attributes,
    '#breakpoints' => $breakpoint_styles,
    '#lazyload' => isset($picture_mappings[$mapping_id]['lazyload']) ? $picture_mappings[$mapping_id]['lazyload'] : FALSE,
    '#lazyload_aspect_ratio' => isset($picture_mappings[$mapping_id]['lazyload_aspect_ratio']) ? $picture_mappings[$mapping_id]['lazyload_aspect_ratio'] : FALSE,
  );
  return $image_render_array;
}

/**
 * Implements callback_filter_tips().
 */
function _picture_filter_tips($filter, $format, $long = FALSE) {
  return t('Images with a data-picture-mapping attribute will be responsive, with a file size appropriate for the browser width.');
}

/**
 * Implements hook_page_build().
 *
 * Add the image processing javascript to every page. This allows these scripts
 * to get included in aggregation, which is probably good since there will be
 * pictures needing this javascript on most pages. The library does not get
 * added twice, even if it's attached to multiple fields that are also being
 * displayed with responsive images. Maybe this should check that the
 * page is not an admin theme page?
 */
function picture_page_build(&$page) {
  drupal_add_library('picture', 'picturefill_head', TRUE);
  drupal_add_library('picture', 'picturefill', TRUE);
  drupal_add_library('picture', 'picture.ajax', TRUE);

  // Integrate with the WYSIWYG module, and the CKEditor module.
  $picture_mappings = picture_get_mapping_options();
  $ckeditor_mappings = variable_get('picture_ckeditor_mappings', array());
  $mappings = array();

  // CKEditor library expects an array of options formatted as
  // ['Display name', 'machine_name'].
  foreach ($ckeditor_mappings as $machine_name => $parameters) {
    if (array_key_exists($machine_name, $picture_mappings)) {
      if ($parameters['enabled'] == 1) {
        $mappings[] = array(
          $picture_mappings[$machine_name],
          $machine_name,
        );
        if (!empty($ckeditor_mappings[$machine_name]['lazyload'])) {
          drupal_add_library('picture', 'lazysizes');
          if (!empty($ckeditor_mappings[$machine_name]['lazyload_aspect_ratio'])) {
            drupal_add_library('picture', 'lazysizes_aspect_ratio');
          }
        }
      }
    }
  }
  if (!empty($mappings)) {
    $label = variable_get('picture_ckeditor_label', 'Image size');
    if (variable_get('picture_ckeditor_required', TRUE)) {
      $label .= ' (required)';
    }
    $mappings[] = array(
      'Not Set',
      'not_set',
    );
    drupal_add_js(array(
      'picture' => array(
        'required' => variable_get('picture_ckeditor_required', TRUE),
        'mappings' => $mappings,
        'label' => $label,
        'ckeditorDefaultMapping' => variable_get('picture_ckeditor_default_mapping', 'not_set'),
        'editorCSS' => picture_ckeditor_css(),
      ),
    ), 'setting');

    // Do not add inline css, unless when editing.
    if (!isset($page['content']['system_main']['#node_edit_form']) || !$page['content']['system_main']['#node_edit_form']) {
      return;
    }
    drupal_add_css(picture_ckeditor_css('.ImagePreviewBox'), 'inline');
  }
}

/**
 * Generate css to handle more realistic display of image styles.
 */
function picture_ckeditor_css($prefix = '') {
  $image_styles = image_styles();
  $picture_mappings = picture_mapping_load_all();
  $ckeditor_mappings = variable_get('picture_ckeditor_mappings', array());
  $mappings = array();
  foreach ($ckeditor_mappings as $machine_name => $parameters) {
    if (!array_key_exists($machine_name, $picture_mappings)) {
      continue;
    }
    if ($parameters['enabled'] != 1) {
      continue;
    }
    if (array_key_exists($parameters['fallback'], $image_styles)) {
      $dimensions = array(
        'width' => 1000,
        'height' => 1000,
      );

      // Transform dimensions.
      image_style_transform_dimensions($parameters['fallback'], $dimensions);

      // Add unit if needed.
      if (strpos($dimensions['width'], '%') === FALSE) {
        $dimensions['width'] .= 'px';
      }
      if (strpos($dimensions['height'], '%') === FALSE) {
        $dimensions['height'] .= 'px';
      }
      $mappings[$machine_name] = $prefix . ' [data-picture-mapping="' . $machine_name . '"] {';
      $mappings[$machine_name] .= ' width: ' . $dimensions['width'] . ';';
      $mappings[$machine_name] .= ' height: ' . $dimensions['height'] . ';';
      $mappings[$machine_name] .= '}';
    }
  }
  return implode("\n", $mappings);
}

/**
 * Helper function to figure out the uri of an image source.
 *
 * @param string $src
 *   Image src starting with http://, https://, or root relative /.
 */
function picture_image_uri($src) {
  foreach (module_implements('picture_image_uri') as $module) {
    $function = $module . '_picture_image_uri';
    if ($uri = $function($src)) {
      $uri = file_stream_wrapper_uri_normalize($uri);
      return urldecode($uri);
    }
  }
  return FALSE;
}

/**
 * Implements hook_picture_image_uri().
 */
function picture_picture_image_uri($src) {
  global $base_path;
  $uri = '';

  // Prepare the src by removing http:// or https://.
  $src = parse_url($src, PHP_URL_PATH);

  // Remove leading or trailing slashes.
  $src = trim($src, '/');

  // List all visible stream wrappers.
  $visible_stream_wrappers = array_intersect_key(file_get_stream_wrappers(STREAM_WRAPPERS_WRITE), file_get_stream_wrappers(STREAM_WRAPPERS_VISIBLE));
  $needles = array();
  $matches = array();
  foreach ($visible_stream_wrappers as $scheme => $data) {
    $class = file_stream_wrapper_get_class($scheme);
    $stream_wrapper = new $class();

    // Trim leading or trailing slashes since the Directory could be root
    // relative.
    if (method_exists($stream_wrapper, 'getDirectoryPath')) {
      $needles[$scheme] = trim($base_path . $stream_wrapper
        ->getDirectoryPath(), '/');

      // Check whether the file stream directory is at the beginning of
      // the image src. Use === since strpos could return false.
      if (!empty($needles[$scheme]) && strpos($src, $needles[$scheme]) === 0) {
        $matches[$scheme] = $needles[$scheme];
      }
    }
  }

  // No file stream wrapper is starting with the image src.
  // So it's not a public file.
  if (empty($matches)) {

    // Check for managed/private file with relative path like system/files/..
    $src_exploded = explode('/', $src);
    if ($src_exploded[0] == 'system' && $src_exploded[1] == 'files') {

      // Managed/private file recognized.
      // Check for image style derivatives.
      if ($src_exploded[2] == 'styles') {

        // Image style recognized.
        $unwanted_part = 'system/files/styles/' . $src_exploded[3] . '/private/';
        $uri = str_replace($unwanted_part, 'private://', $src);
        $uri = file_stream_wrapper_uri_normalize($uri);
        return urldecode($uri);
      }
      else {

        // No image style recognized; must be an original.
        $unwanted_part = 'system/files/';
        $uri = str_replace($unwanted_part, 'private://', $src);
        $uri = file_stream_wrapper_uri_normalize($uri);
        return urldecode($uri);
      }
    }
    else {

      // Can't figure out the Drupal uri.
      return FALSE;
    }
  }

  // If one file scheme directory is a subdirectory of another file
  // scheme directory, choose the longer one. This issue is possible with
  // the following scenario:
  // public file dir: /sites/default/files/
  // private file dir: /sites/default/files/private/
  // image src: /sites/default/files/private/the-image.jpg
  // In this example, the intended scheme would be 'private'.
  if (empty($matches)) {

    // Can't figure out the Drupal uri.
    return FALSE;
  }

  // Find the length of each matching directory path.
  $lengths = array_map('strlen', $matches);

  // Determine the key of the longest one.
  $the_scheme = array_search(max($lengths), $lengths);

  // Construct the Drupal uri.
  $uri = $the_scheme . '://' . str_replace($matches[$the_scheme], '', $src);
  return $uri;
}

/**
 * Implements hook_wysiwyg_plugin().
 *
 * Modify the CKEditor image dialog for use with the picture module.
 */
function picture_wysiwyg_plugin($editor, $version) {
  if ($editor == 'ckeditor') {
    return array(
      'picture_ckeditor' => array(
        'path' => drupal_get_path('module', 'picture') . '/ckeditor/plugins/',
        'load' => TRUE,
        'extensions' => array(
          'picture_ckeditor' => t('Responsive images with the Picture Module'),
        ),
        'url' => 'https://www.drupal.org/project/picture',
      ),
    );
  }
}

/**
 * Implements hook_ckeditor_plugin().
 *
 * Modify the CKEditor image dialog for use with the picture module.
 */
function picture_ckeditor_plugin() {
  return array(
    'picture_ckeditor' => array(
      'name' => 'picture_ckeditor',
      'desc' => t('Support responsive images with the Picture module.'),
      'buttons' => FALSE,
      'path' => drupal_get_path('module', 'picture') . '/ckeditor/plugins/',
      'default' => 't',
    ),
  );
}

/**
 * Wrapper around image_style_url() so we can return an empty image.
 */
function _picture_image_style_url($style_name, $path, $timestamp = NULL) {
  if ($style_name == PICTURE_EMPTY_IMAGE) {
    return '';
  }
  if ($style_name == PICTURE_ORIGINAL_IMAGE) {
    $url = file_create_url($path);
  }
  else {
    $url = image_style_url($style_name, $path);
  }
  $token_query = array();
  if (!empty($timestamp) && !variable_get('image_suppress_itok_output', FALSE)) {
    $token_query['timestamp'] = $timestamp;
  }

  // Allow modules to alter the $token_query for the given style.
  $context = array(
    'uri' => $url,
    'style_name' => $style_name,
    'path' => $path,
  );
  drupal_alter('image_style_uri_token_query', $token_query, $context);

  // Append the query string with the token, if necessary.
  if ($token_query) {
    $url .= (strpos($url, '?') !== FALSE ? '&' : '?') . drupal_http_build_query($token_query);
  }
  return $url;
}

/**
 * Implements hook_wysiwyg_editor_settings_alter().
 */
function picture_wysiwyg_editor_settings_alter(&$settings, $context) {
  if ($context['profile']->editor == 'ckeditor') {
    if (!isset($settings['extraAllowedContent'])) {
      $settings['extraAllowedContent'] = array(
        'img[src,title,alt,style,width,height,class,hspace,vspace,view_mode,format,fid,data-picture-mapping,data-picture-group,data-picture-align]',
      );
    }
    else {

      // @todo: try finding the img entry and add data- attributes if needed.
    }
  }
}

/**
 * Returns a list with the image styles of a mapping configuration.
 *
 * @param object $picture_mapping
 *   The mapping configuration.
 * @param string $fallback_image_style
 *   Reference to access the evaluated fallback image style.
 *
 * @return array
 *   List with the image styles of a mapping configuration.
 *   The array has following structure:
 *   array(
 *     breakpoint_name => array(
 *       multiplier => array(
 *         mapping_type => _none|sizes|image_style,
 *         image_style => image_style_name,
 *         sizes => '(min-width: 700px) 700px, 100vw',
 *         sizes_image_styles => array(
 *           thumbnail => thumbnail,
 *           medium => medium,
 *           large => large,
 *         ),
 *       ),
 *     ),
 *   );
 */
function picture_get_mapping_breakpoints(PictureMapping $picture_mapping, &$fallback_image_style = NULL) {
  $breakpoint_styles = array();
  $image_styles = array();
  foreach ($picture_mapping
    ->getMappings() as $breakpoint_name => $multipliers) {

    // Make sure there are multipliers.
    if (!empty($multipliers)) {
      $breakpoint = breakpoints_breakpoint_load_by_fullkey($breakpoint_name);
      if ($breakpoint && $breakpoint->status) {

        // Determine the enabled multipliers.
        $multipliers = array_intersect_key($multipliers, $breakpoint->multipliers);
        foreach ($multipliers as $multiplier => $mapping_definition) {

          // Make sure the multiplier still exists.
          if (!PictureMapping::isEmptyMappingDefinition($mapping_definition)) {
            if ($mapping_definition['mapping_type'] == 'image_style') {
              $image_styles[] = $mapping_definition['image_style'];
            }
            elseif ($mapping_definition['mapping_type'] == 'sizes') {
              $mapping_definition['sizes_image_styles'] = array_filter($mapping_definition['sizes_image_styles']);
              $image_styles = array_merge($image_styles, $mapping_definition['sizes_image_styles']);
            }
            if (!isset($breakpoint_styles[$breakpoint_name])) {
              $breakpoint_styles[$breakpoint_name] = array();
            }
            $breakpoint_styles[$breakpoint_name][$multiplier] = $mapping_definition;
          }
        }
      }
    }
  }

  // Check if the user defined a custom fallback image style.
  if (!$fallback_image_style) {
    $fallback_image_style = end($image_styles);
  }
  return $breakpoint_styles;
}

/**
 * Helper function to compute the list of possible link types.
 */
function picture_link_types($instance) {
  $link_types = array(
    'content' => t('Content'),
    'file' => t('File'),
  );
  if (module_exists('colorbox')) {
    $link_types['colorbox'] = t('Colorbox');
  }

  // If the link module is installed, also allow any link fields to be used.
  foreach (field_info_fields() as $field_key => $field_info) {
    if ($field_info['type'] == 'link_field') {
      $field_instance = field_info_instance($instance['entity_type'], $field_info['field_name'], $instance['bundle']);
      if ($field_instance) {
        $link_types[$field_key] = "{$field_instance['label']} ({$field_info['field_name']})";
      }
    }
  }

  // Allow linking to a specific image style.
  $image_styles = image_style_options(FALSE);
  foreach ($image_styles as $image_style => $label) {
    $link_types[$image_style] = t('Image style: @style', array(
      '@style' => $label,
    ));
  }
  return $link_types;
}

/**
 * Picture mapping factory.
 */
function picture_mapping_object_factory($schema, $data) {
  if (!class_exists('PictureMapping')) {
    module_load_include('php', 'picture', 'includes/PictureMapping');
  }
  $mapping = new PictureMapping();
  $mapping
    ->setValues($schema, $data);
  return $mapping;
}

/**
 * Create picture mapping.
 */
function picture_mapping_create($set_defaults) {
  $schema = ctools_export_get_schema('picture_mapping');
  $data = ctools_export_new_object('picture_mapping', $set_defaults);
  return picture_mapping_object_factory($schema, $data);
}

/**
 * Export picture mapping.
 */
function picture_mapping_export(PictureMapping $mapping, $indent) {
  return $mapping
    ->export($indent);
}

/**
 * Implements hook_ctools_plugin_api().
 */
function picture_ctools_plugin_api($module = NULL, $api = NULL) {
  if ($module == "breakpoints" && $api == "default_breakpoints") {
    return array(
      "version" => "1",
    );
  }
}

/**
 * Implements hook_default_breakpoints().
 */
function picture_default_breakpoints() {
  $export = array();
  $breakpoint = new stdClass();
  $breakpoint->disabled = FALSE;

  /* Edit this to true to make a default breakpoint disabled initially */
  $breakpoint->api_version = 1;
  $breakpoint->machine_name = BREAKPOINTS_SOURCE_TYPE_MODULE . '.picture.empty_srcset';
  $breakpoint->name = 'Empty srcset';
  $breakpoint->breakpoint = '';
  $breakpoint->source = 'picture';
  $breakpoint->source_type = BREAKPOINTS_SOURCE_TYPE_MODULE;
  $breakpoint->status = 1;
  $breakpoint->weight = 0;
  $breakpoint->multipliers = array(
    '1x' => '1x',
  );
  $export[BREAKPOINTS_SOURCE_TYPE_MODULE . '.picture.empty_srcset'] = $breakpoint;
  return $export;
}

Functions

Namesort descending Description
picture_ckeditor_css Generate css to handle more realistic display of image styles.
picture_ckeditor_plugin Implements hook_ckeditor_plugin().
picture_ctools_plugin_api Implements hook_ctools_plugin_api().
picture_ctools_plugin_directory Implements hook_ctools_plugin_directory().
picture_default_breakpoints Implements hook_default_breakpoints().
picture_field_formatter_info Implements hook_field_formatter_info().
picture_field_formatter_picture_sizes_formatter_view Helper function.
picture_field_formatter_picture_view Helper function.
picture_field_formatter_settings_form Implements hook_field_formatter_settings_form().
picture_field_formatter_settings_picture_form Helper function.
picture_field_formatter_settings_picture_sizes_formatter_form Helper function.
picture_field_formatter_settings_picture_sizes_formatter_summary Helper function.
picture_field_formatter_settings_picture_summary Helper function.
picture_field_formatter_settings_summary Implements hook_field_formatter_settings_summary().
picture_field_formatter_view Implements hook_field_formatter_view().
picture_filter_info Implements hook_filter_info().
picture_flush_caches Implements hook_flush_caches().
picture_get_image_dimensions Determines the dimensions of an image.
picture_get_mapping_breakpoints Returns a list with the image styles of a mapping configuration.
picture_get_mapping_options Returns a list of picture mappings for use in a select list.
picture_help Implements hook_help().
picture_image_uri Helper function to figure out the uri of an image source.
picture_library Implements hook_library().
picture_link_types Helper function to compute the list of possible link types.
picture_mapping_create Create picture mapping.
picture_mapping_export Export picture mapping.
picture_mapping_load Load mappings.
picture_mapping_load_all Load all mappings.
picture_mapping_object_factory Picture mapping factory.
picture_mapping_save Save mappings.
picture_menu Implements hook_menu().
picture_modules_enabled Implements hook_modules_enabled().
picture_page_build Implements hook_page_build().
picture_permission Implements hook_permission().
picture_picture_image_uri Implements hook_picture_image_uri().
picture_theme Implements hook_theme().
picture_wysiwyg_editor_settings_alter Implements hook_wysiwyg_editor_settings_alter().
picture_wysiwyg_plugin Implements hook_wysiwyg_plugin().
theme_image_srcset Returns HTML for an image.
theme_picture Returns HTML for a picture.
theme_picture_formatter Theme picture formatter.
theme_picture_formatter_colorbox Theme function to add support for colorbox.
theme_picture_sizes_formatter Theme pictue sizes formatter.
theme_picture_source Returns HTML for a source tag.
_picture_filter_prepare_image Prepares a Render Array for theme_picture_formatter().
_picture_filter_process Process callback for inline image filter.
_picture_filter_tips Implements callback_filter_tips().
_picture_image_style_url Wrapper around image_style_url() so we can return an empty image.

Constants

Namesort descending Description
PICTURE_EMPTY_IMAGE @file Picture formatter.
PICTURE_ORIGINAL_IMAGE