color.module 27.4 KB
Newer Older
1
<?php
2 3 4 5
/**
 * @file
 * Allows users to change the color scheme of themes.
 */
6

7
use Drupal\Component\Utility\Unicode;
8
use Drupal\Core\Asset\CssOptimizer;
9 10
use Drupal\Component\Utility\Bytes;
use Drupal\Component\Utility\Environment;
11
use Drupal\Component\Utility\SafeMarkup;
12
use Drupal\Component\Utility\String;
13
use Drupal\Core\Cache\Cache;
14
use Drupal\Core\Form\FormStateInterface;
15
use Drupal\Core\Language\LanguageInterface;
16
use Drupal\Core\Render\Element\Textfield;
17
use Drupal\Core\Routing\RouteMatchInterface;
18

19
/**
20
 * Implements hook_help().
21
 */
22
function color_help($route_name, RouteMatchInterface $route_match) {
23 24
  switch ($route_name) {
    case 'help.page.color':
25
      $output = '<h3>' . t('About') . '</h3>';
26
      $output .= '<p>' . t('The Color module allows users with the <em>Administer site configuration</em> permission to change the color scheme (color of links, backgrounds, text, and other theme elements) of themes that are compatible with it. For more information, see <a href="!color_do">the online documentation for the Color module</a>.', array('!color_do' => 'https://drupal.org/documentation/modules/color')) . '</p>';
27 28
      $output .= '<h3>' . t('Uses') . '</h3>';
      $output .= '<dl>';
29
      $output .= '<dt>' . t('Changing colors') . '</dt>';
30 31
      $output .= '<dd><p>' . t('To change the color settings, select the <em>Settings</em> link for your theme on the <a href="!appearance">Appearance</a> page. If the color picker does not appear than the theme is not compatible with the Color module.', array('!appearance' => \Drupal::url('system.themes_page'))) . '</p>';
      $output .= '<p>' . t('The Color module saves a modified copy of the theme\'s specified stylesheets in the files directory. This means that if you make any manual changes to your theme\'s stylesheet, <em>you must save your color settings again, even if they haven\'t changed</em>. This step is required because the module stylesheets (in the files directory) need to be recreated to include your changes.') . '</p></dd>';
32
      $output .= '</dl>';
33 34 35 36
      return $output;
  }
}

37
/**
38
 * Implements hook_theme().
39 40 41 42
 */
function color_theme() {
  return array(
    'color_scheme_form' => array(
43
      'render element' => 'form',
44 45 46
    ),
  );
}
47

48
/**
49
 * Implements hook_form_FORM_ID_alter().
50
 */
51
function color_form_system_theme_settings_alter(&$form, FormStateInterface $form_state) {
52 53
  $build_info = $form_state->getBuildInfo();
  if (isset($build_info['args'][0]) && ($theme = $build_info['args'][0]) && color_get_info($theme) && function_exists('gd_info')) {
54
    $form['color'] = array(
55
      '#type' => 'details',
56
      '#title' => t('Color scheme'),
57
      '#open' => TRUE,
58 59 60 61
      '#weight' => -1,
      '#attributes' => array('id' => 'color_scheme_form'),
      '#theme' => 'color_scheme_form',
    );
62
    $form['color'] += color_scheme_form($form, $form_state, $theme);
63
    $form['#validate'][] = 'color_scheme_form_validate';
64 65
    // Ensure color submission happens first so we can unset extra values.
    array_unshift($form['#submit'], 'color_scheme_form_submit');
66
  }
67 68
}

69
/**
70
 * Implements hook_library_info_alter().
71 72 73
 *
 * Replaces style sheets declared in libraries with color-altered style sheets.
 */
74 75 76 77
function color_library_info_alter(&$libraries, $extension) {
  $themes =  array_keys(\Drupal::service('theme_handler')->listInfo());
  if (in_array($extension, $themes)) {
    $color_paths = \Drupal::config('color.theme.' . $extension)->get('stylesheets');
78
    if (!empty($color_paths)) {
79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100
      foreach (array_keys($libraries) as $name) {
        // Override stylesheets.
        foreach ($libraries[$name]['css'] as $category => $css_assets) {
          foreach ($css_assets as $path  => $metadata) {
            // Loop over the path array with recolored CSS files to find matching
            // paths which could replace the non-recolored paths.
            foreach ($color_paths as $color_path) {
              // Color module currently requires unique file names to be used,
              // which allows us to compare different file paths.
              if (drupal_basename($path) == drupal_basename($color_path)) {
                // Replace the path to the new css file.
                // This keeps the order of the stylesheets intact.
                $index = array_search($path, array_keys($libraries[$name]['css'][$category]));
                $preceding_css_assets = array_slice($libraries[$name]['css'][$category], 0, $index);
                $succeeding_css_assets = array_slice($libraries[$name]['css'][$category], $index + 1);
                $libraries[$name]['css'][$category] = array_merge(
                  $preceding_css_assets,
                  [$color_path => $metadata],
                  $succeeding_css_assets
                );
              }
            }
101 102 103 104 105 106 107
          }
        }
      }
    }
  }
}

108
/**
109
 * Implements hook_preprocess_page().
110
 *
111
 * Replace the logo with the colored version if available.
112
 */
113
function color_preprocess_page(&$variables) {
114
  $theme_key = \Drupal::theme()->getActiveTheme()->getName();
115

116
  // Override logo.
117
  $logo = \Drupal::config('color.theme.' . $theme_key)->get('logo');
118
  if ($logo && $variables['logo'] && preg_match('!' . $theme_key . '/logo.svg$!', $variables['logo'])) {
119
    $variables['logo'] = file_create_url($logo);
120 121 122 123
  }
}

/**
124
 * Retrieves the Color module information for a particular theme.
125 126
 */
function color_get_info($theme) {
127 128 129 130 131 132
  static $theme_info = array();

  if (isset($theme_info[$theme])) {
    return $theme_info[$theme];
  }

133
  $path = drupal_get_path('theme', $theme);
134
  $file = \Drupal::root() . '/' . $path . '/color/color.inc';
135 136
  if ($path && file_exists($file)) {
    include $file;
137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152
    // Add in default values.
    $info += array(
      // CSS files (excluding @import) to rewrite with new color scheme.
      'css' => array(),
      // Files to copy.
      'copy' => array(),
      // Gradient definitions.
      'gradients' => array(),
      // Color areas to fill (x, y, width, height).
      'fill' => array(),
      // Coordinates of all the theme slices (x, y, width, height) with their
      // filename as used in the stylesheet.
      'slices' => array(),
      // Reference color used for blending.
      'blend_target' => '#ffffff',
    );
153
    $theme_info[$theme] = $info;
154 155 156 157 158
    return $info;
  }
}

/**
159
 * Retrieves the color palette for a particular theme.
160
 */
161
function color_get_palette($theme, $default = FALSE) {
162
  // Fetch and expand default palette.
163
  $info = color_get_info($theme);
164
  $palette = $info['schemes']['default']['colors'];
165

166 167 168 169
  if ($default) {
    return $palette;
  }

170
  // Load variable.
171
  // @todo Default color config should be moved to yaml in the theme.
172 173 174 175
  // Getting a mutable override-free object because this function is only used
  // in forms. Color configuration is used to write CSS to the file system
  // making configuration overrides pointless.
  return \Drupal::configFactory()->getEditable('color.theme.' . $theme)->get('palette') ?: $palette;
176 177 178
}

/**
179 180 181 182 183 184 185
 * Form constructor for the color configuration form for a particular theme.
 *
 * @param $theme
 *   The machine name of the theme whose color settings are being configured.
 *
 * @see color_scheme_form_validate()
 * @see color_scheme_form_submit()
186
 */
187
function color_scheme_form($complete_form, FormStateInterface $form_state, $theme) {
188 189 190
  $base = drupal_get_path('module', 'color');
  $info = color_get_info($theme);

191 192 193 194 195 196 197 198 199
  $info['schemes'][''] = array('title' => t('Custom'), 'colors' => array());
  $color_sets = array();
  $schemes = array();
  foreach ($info['schemes'] as $key => $scheme) {
    $color_sets[$key] = $scheme['title'];
    $schemes[$key] = $scheme['colors'];
    $schemes[$key] += $info['schemes']['default']['colors'];
  }

200
  // See if we're using a predefined scheme.
201
  // Note: we use the original theme when the default scheme is chosen.
202 203 204 205 206
  // Note: we use configuration without overrides since this information is used
  // in a form and therefore without doing this would bleed overrides into
  // active configuration. Furthermore, color configuration is used to write
  // CSS to the file system making configuration overrides pointless.
  $current_scheme = \Drupal::configFactory()->getEditable('color.theme.' . $theme)->get('palette');
207 208 209 210 211 212 213 214 215 216 217 218 219 220
  foreach ($schemes as $key => $scheme) {
    if ($current_scheme == $scheme) {
      $scheme_name = $key;
      break;
    }
  }
  if (empty($scheme_name)) {
    if (empty($current_scheme)) {
      $scheme_name = 'default';
    }
    else {
      $scheme_name = '';
    }
  }
221

222
  // Add scheme selector.
223
  $default_palette = color_get_palette($theme, TRUE);
224 225 226
  $form['scheme'] = array(
    '#type' => 'select',
    '#title' => t('Color set'),
227 228
    '#options' => $color_sets,
    '#default_value' => $scheme_name,
229 230
    '#attached' => array(
      'library' => array(
231
        'color/drupal.color',
232
        'color/admin',
233 234
      ),
      // Add custom JavaScript.
235 236
      'drupalSettings' => [
        'color' => [
237
          'reference' => $default_palette,
238 239 240 241
          'schemes' => $schemes,
        ],
        'gradients' => $info['gradients'],
      ],
242
    ),
243 244
  );

245 246
  // Add palette fields. Use the configuration if available.
  $palette = $current_scheme ?: $default_palette;
247
  $names = $info['fields'];
248
  $form['palette']['#tree'] = TRUE;
249
  foreach ($palette as $name => $value) {
250 251 252
    if (isset($names[$name])) {
      $form['palette'][$name] = array(
        '#type' => 'textfield',
253
        '#title' => String::checkPlain($names[$name]),
254
        '#value_callback' => 'color_palette_color_value',
255 256
        '#default_value' => $value,
        '#size' => 8,
257
        '#attributes' => array('dir' => LanguageInterface::DIRECTION_LTR),
258 259
      );
    }
260
  }
261
  $form['theme'] = array('#type' => 'value', '#value' => $theme);
262 263 264 265 266 267
  $form['info'] = array('#type' => 'value', '#value' => $info);

  return $form;
}

/**
268
 * Prepares variables for color scheme form templates.
269
 *
270 271 272
 * Default template: color-scheme-form.html.twig.
 *
 * @param array $variables
273 274
 *   An associative array containing:
 *   - form: A render element representing the form.
275
 */
276
function template_preprocess_color_scheme_form(&$variables) {
277
  $form = &$variables['form'];
278

279 280
  $theme = $form['theme']['#value'];
  $info = $form['info']['#value'];
281

282 283
  if (isset($info['preview_library'])) {
    $form['scheme']['#attached']['library'][] = $info['preview_library'];
284
  }
285

286
  // Attempt to load preview HTML if the theme provides it.
287
  $preview_html_path = \Drupal::root() . '/' . (isset($info['preview_html']) ? drupal_get_path('theme', $theme) . '/' . $info['preview_html'] : drupal_get_path('module', 'color') . '/preview.html');
288
  $variables['html_preview'] = SafeMarkup::set(file_get_contents($preview_html_path));
289 290
}

291 292 293 294 295 296 297 298
/**
 * Determines the value for a palette color field.
 *
 * @param array $element
 *   The form element whose value is being populated.
 * @param string|bool $input
 *   The incoming input to populate the form element. If this is FALSE,
 *   the element's default value should be returned.
299 300
 * @param \Drupal\Core\Form\FormStateInterface $form_state
 *   The current state of the form.
301 302
 *
 * @return string
303
 *   The data that will appear in the $form_state->getValues() collection for this
304 305
 *   element. Return nothing to use the default.
 */
306
function color_palette_color_value($element, $input = FALSE, FormStateInterface $form_state) {
307 308 309 310 311 312
  // If we suspect a possible cross-site request forgery attack, only accept
  // hexadecimal CSS color strings from user input, to avoid problems when this
  // value is used in the JavaScript preview.
  if ($input !== FALSE) {
    // Start with the provided value for this textfield, and validate that if
    // necessary, falling back on the default value.
313
    $value = Textfield::valueCallback($element, $input, $form_state);
314 315
    $complete_form = $form_state->getCompleteForm();
    if (!$value || !isset($complete_form['#token']) || color_valid_hexadecimal_string($value) || \Drupal::csrfToken()->validate($form_state->getValue('form_token'), $complete_form['#token'])) {
316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337
      return $value;
    }
    else {
      return $element['#default_value'];
    }
  }
}

/**
 * Determines if a hexadecimal CSS color string is valid.
 *
 * @param string $color
 *   The string to check.
 *
 * @return bool
 *   TRUE if the string is a valid hexadecimal CSS color string, or FALSE if it
 *   isn't.
 */
function color_valid_hexadecimal_string($color) {
  return preg_match('/^#([a-f0-9]{3}){1,2}$/iD', $color);
}

338
/**
339 340 341
 * Form validation handler for color_scheme_form().
 *
 * @see color_scheme_form_submit()
342
 */
343
function color_scheme_form_validate($form, FormStateInterface $form_state) {
344
  // Only accept hexadecimal CSS color strings to avoid XSS upon use.
345
  foreach ($form_state->getValue('palette') as $key => $color) {
346
    if (!color_valid_hexadecimal_string($color)) {
347
      $form_state->setErrorByName('palette][' . $key, t('You must enter a valid hexadecimal color value for %name.', array('%name' => $form['color']['palette'][$key]['#title'])));
348 349 350 351
    }
  }
}

352
/**
353 354 355
 * Form submission handler for color_scheme_form().
 *
 * @see color_scheme_form_validate()
356
 */
357
function color_scheme_form_submit($form, FormStateInterface $form_state) {
358

359 360 361 362 363 364 365 366 367 368
  // Avoid color settings spilling over to theme settings.
  $color_settings = array('theme', 'palette', 'scheme');
  if ($form_state->hasValue('info')) {
    $color_settings[] = 'info';
  }
  foreach ($color_settings as $setting_name) {
    ${$setting_name} = $form_state->getValue($setting_name);
    $form_state->unsetValue($setting_name);
  }
  if (!isset($info)) {
369 370 371
    return;
  }

372
  $config = \Drupal::configFactory()->getEditable('color.theme.' . $theme);
373

374
  // Resolve palette.
375
  if ($scheme != '') {
376
    foreach ($palette as $key => $color) {
377 378
      if (isset($info['schemes'][$scheme]['colors'][$key])) {
        $palette[$key] = $info['schemes'][$scheme]['colors'][$key];
379
      }
380
    }
381
    $palette += $info['schemes']['default']['colors'];
382 383
  }

384
  // Make sure enough memory is available.
385 386 387 388 389 390 391 392 393 394 395 396 397 398
  if (isset($info['base_image'])) {
    // Fetch source image dimensions.
    $source = drupal_get_path('theme', $theme) . '/' . $info['base_image'];
    list($width, $height) = getimagesize($source);

    // We need at least a copy of the source and a target buffer of the same
    // size (both at 32bpp).
    $required = $width * $height * 8;
    // We intend to prevent color scheme changes if there isn't enough memory
    // available.  memory_get_usage(TRUE) returns a more accurate number than
    // memory_get_usage(), therefore we won't inadvertently reject a color
    // scheme change based on a faulty memory calculation.
    $usage = memory_get_usage(TRUE);
    $memory_limit = ini_get('memory_limit');
399 400
    $size = Bytes::toInt($memory_limit);
    if (!Environment::checkMemoryLimit($usage + $required, $memory_limit)) {
401 402 403
      drupal_set_message(t('There is not enough memory available to PHP to change this theme\'s color scheme. You need at least %size more. Check the <a href="@url">PHP documentation</a> for more information.', array('%size' => format_size($usage + $required - $size), '@url' => 'http://www.php.net/manual/ini.core.php#ini.sect.resource-limits')), 'error');
      return;
    }
404 405
  }

406
  // Delete old files.
407 408 409 410 411
  $files = $config->get('files');
  if (isset($files)) {
    foreach ($files as $file) {
      @drupal_unlink($file);
    }
412
  }
413
  if (isset($file) && $file = dirname($file)) {
414
    @drupal_rmdir($file);
415 416
  }

417
  // No change in color config, use the standard theme from color.inc.
418
  if (implode(',', color_get_palette($theme, TRUE)) == implode(',', $palette)) {
419
    $config->delete();
420 421 422
    return;
  }

423
  // Prepare target locations for generated files.
424
  $id = $theme . '-' . substr(hash('sha256', serialize($palette) . microtime()), 0, 8);
425
  $paths['color'] = 'public://color';
426
  $paths['target'] = $paths['color'] . '/' . $id;
427
  foreach ($paths as $path) {
428
    file_prepare_directory($path, FILE_CREATE_DIRECTORY);
429
  }
430
  $paths['target'] = $paths['target'] . '/';
431
  $paths['id'] = $id;
432
  $paths['source'] = drupal_get_path('theme', $theme) . '/';
433 434
  $paths['files'] = $paths['map'] = array();

435
  // Save palette and logo location.
436 437
  $config
    ->set('palette', $palette)
438
    ->set('logo', $paths['target'] . 'logo.svg')
439
    ->save();
440

441
  // Copy over neutral images.
442
  foreach ($info['copy'] as $file) {
443
    $base = drupal_basename($file);
444
    $source = $paths['source'] . $file;
445
    $filepath = file_unmanaged_copy($source, $paths['target'] . $base);
446
    $paths['map'][$file] = $base;
447
    $paths['files'][] = $filepath;
448 449
  }

450
  // Render new images, if image has been provided.
451
  if (isset($info['base_image'])) {
452 453
    _color_render_images($theme, $info, $paths, $palette);
  }
454

455 456 457
  // Rewrite theme stylesheets.
  $css = array();
  foreach ($info['css'] as $stylesheet) {
458
    // Build a temporary array with CSS files.
459 460 461 462 463 464
    $files = array();
    if (file_exists($paths['source'] . $stylesheet)) {
      $files[] = $stylesheet;
    }

    foreach ($files as $file) {
465
      $css_optimizer = new CssOptimizer();
466 467
      // Aggregate @imports recursively for each configured top level CSS file
      // without optimization. Aggregation and optimization will be
468
      // handled by drupal_build_css_cache() only.
469
      $style = $css_optimizer->loadFile($paths['source'] . $file, FALSE);
470 471 472

      // Return the path to where this CSS file originated from, stripping
      // off the name of the file at the end of the path.
473
      $css_optimizer->rewriteFileURIBasePath = base_path() . dirname($paths['source'] . $file) . '/';
474 475

      // Prefix all paths within this CSS file, ignoring absolute paths.
476
      $style = preg_replace_callback('/url\([\'"]?(?![a-z]+:|\/+)([^\'")]+)[\'"]?\)/i', array($css_optimizer, 'rewriteFileURI'), $style);
477 478 479

      // Rewrite stylesheet with new colors.
      $style = _color_rewrite_stylesheet($theme, $info, $paths, $palette, $style);
480
      $base_file = drupal_basename($file);
481 482
      $css[] = $paths['target'] . $base_file;
      _color_save_stylesheet($paths['target'] . $base_file, $style, $paths);
483
    }
484
  }
485

486
  // Maintain list of files.
487 488 489 490
  $config
    ->set('stylesheets', $css)
    ->set('files', $paths['files'])
    ->save();
491 492 493

  // Clear the library cache.
  Cache::invalidateTags(['library_info']);
494 495 496
}

/**
497
 * Rewrites the stylesheet to match the colors in the palette.
498
 */
499
function _color_rewrite_stylesheet($theme, &$info, &$paths, $palette, $style) {
500
  // Prepare color conversion table.
501 502
  $conversion = $palette;
  foreach ($conversion as $k => $v) {
503
    $conversion[$k] = Unicode::strtolower($v);
504
  }
505
  $default = color_get_palette($theme, TRUE);
506 507

  // Split off the "Don't touch" section of the stylesheet.
508
  $split = "Color Module: Don't touch";
509
  if (strpos($style, $split) !== FALSE) {
510 511
    list($style, $fixed) = explode($split, $style);
  }
512 513 514

  // Find all colors in the stylesheet and the chunks in between.
  $style = preg_split('/(#[0-9a-f]{6}|#[0-9a-f]{3})/i', $style, -1, PREG_SPLIT_DELIM_CAPTURE);
515
  $is_color = FALSE;
516 517 518
  $output = '';
  $base = 'base';

519
  // Iterate over all the parts.
520 521
  foreach ($style as $chunk) {
    if ($is_color) {
522
      $chunk = Unicode::strtolower($chunk);
523
      // Check if this is one of the colors in the default palette.
524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539
      if ($key = array_search($chunk, $default)) {
        $chunk = $conversion[$key];
      }
      // Not a pre-set color. Extrapolate from the base.
      else {
        $chunk = _color_shift($palette[$base], $default[$base], $chunk, $info['blend_target']);
      }
    }
    else {
      // Determine the most suitable base color for the next color.

      // 'a' declarations. Use link.
      if (preg_match('@[^a-z0-9_-](a)[^a-z0-9_-][^/{]*{[^{]+$@i', $chunk)) {
        $base = 'link';
      }
      // 'color:' styles. Use text.
540
      elseif (preg_match('/(?<!-)color[^{:]*:[^{#]*$/i', $chunk)) {
541 542 543 544 545 546 547 548 549 550
        $base = 'text';
      }
      // Reset back to base.
      else {
        $base = 'base';
      }
    }
    $output .= $chunk;
    $is_color = !$is_color;
  }
551 552 553 554
  // Append fixed colors segment.
  if (isset($fixed)) {
    $output .= $fixed;
  }
555

556
  // Replace paths to images.
557
  foreach ($paths['map'] as $before => $after) {
558 559
    $before = base_path() . $paths['source'] . $before;
    $before = preg_replace('`(^|/)(?!../)([^/]+)/../`', '$1', $before);
560 561 562
    $output = str_replace($before, $after, $output);
  }

563
  return $output;
564 565 566
}

/**
567
 * Saves the rewritten stylesheet to disk.
568
 */
569
function _color_save_stylesheet($file, $style, &$paths) {
570
  $filepath = file_unmanaged_save_data($style, $file, FILE_EXISTS_REPLACE);
571
  $paths['files'][] = $filepath;
572 573

  // Set standard file permissions for webserver-generated files.
574
  drupal_chmod($file);
575 576 577
}

/**
578
 * Renders images that match a given palette.
579 580
 */
function _color_render_images($theme, &$info, &$paths, $palette) {
Steven Wittens's avatar
Steven Wittens committed
581
  // Prepare template image.
582
  $source = $paths['source'] . '/' . $info['base_image'];
583 584 585
  $source = imagecreatefrompng($source);
  $width = imagesx($source);
  $height = imagesy($source);
Steven Wittens's avatar
Steven Wittens committed
586 587

  // Prepare target buffer.
588
  $target = imagecreatetruecolor($width, $height);
589
  imagealphablending($target, TRUE);
590 591 592 593 594 595

  // Fill regions of solid color.
  foreach ($info['fill'] as $color => $fill) {
    imagefilledrectangle($target, $fill[0], $fill[1], $fill[0] + $fill[2], $fill[1] + $fill[3], _color_gd($target, $palette[$color]));
  }

596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612
  // Render gradients.
  foreach ($info['gradients'] as $gradient) {
    // Get direction of the gradient.
    if (isset($gradient['direction']) && $gradient['direction'] == 'horizontal') {
      // Horizontal gradient.
      for ($x = 0; $x < $gradient['dimension'][2]; $x++) {
        $color = _color_blend($target, $palette[$gradient['colors'][0]], $palette[$gradient['colors'][1]], $x / ($gradient['dimension'][2] - 1));
        imagefilledrectangle($target, ($gradient['dimension'][0] + $x), $gradient['dimension'][1], ($gradient['dimension'][0] + $x + 1), ($gradient['dimension'][1] + $gradient['dimension'][3]), $color);
      }
    }
    else {
      // Vertical gradient.
      for ($y = 0; $y < $gradient['dimension'][3]; $y++) {
        $color = _color_blend($target, $palette[$gradient['colors'][0]], $palette[$gradient['colors'][1]], $y / ($gradient['dimension'][3] - 1));
        imagefilledrectangle($target, $gradient['dimension'][0], $gradient['dimension'][1] + $y, $gradient['dimension'][0] + $gradient['dimension'][2], $gradient['dimension'][1] + $y + 1, $color);
      }
    }
613 614
  }

Steven Wittens's avatar
Steven Wittens committed
615
  // Blend over template.
616 617
  imagecopy($target, $source, 0, 0, 0, 0, $width, $height);

Steven Wittens's avatar
Steven Wittens committed
618
  // Clean up template image.
619 620
  imagedestroy($source);

Steven Wittens's avatar
Steven Wittens committed
621
  // Cut out slices.
622 623
  foreach ($info['slices'] as $file => $coord) {
    list($x, $y, $width, $height) = $coord;
624
    $base = drupal_basename($file);
625
    $image = drupal_realpath($paths['target'] . $base);
626

Steven Wittens's avatar
Steven Wittens committed
627
    // Cut out slice.
628 629 630
    if ($file == 'screenshot.png') {
      $slice = imagecreatetruecolor(150, 90);
      imagecopyresampled($slice, $target, 0, 0, $x, $y, 150, 90, $width, $height);
631
      \Drupal::configFactory()->getEditable('color.theme.' . $theme)
632 633
        ->set('screenshot', $image)
        ->save();
634 635 636 637 638 639
    }
    else {
      $slice = imagecreatetruecolor($width, $height);
      imagecopy($slice, $target, 0, 0, $x, $y, $width, $height);
    }

Steven Wittens's avatar
Steven Wittens committed
640
    // Save image.
641 642 643 644
    imagepng($slice, $image);
    imagedestroy($slice);
    $paths['files'][] = $image;

645
    // Set standard file permissions for webserver-generated files
646
    drupal_chmod($image);
647

648 649 650 651
    // Build before/after map of image paths.
    $paths['map'][$file] = $base;
  }

Steven Wittens's avatar
Steven Wittens committed
652
  // Clean up target buffer.
653 654 655 656
  imagedestroy($target);
}

/**
657
 * Shifts a given color, using a reference pair and a target blend color.
658 659 660 661
 *
 * Note: this function is significantly different from the JS version, as it
 * is written to match the blended images perfectly.
 *
662 663
 * Constraint: if (ref2 == target + (ref1 - target) * delta) for some fraction
 * delta then (return == target + (given - target) * delta).
664 665
 *
 * Loose constraint: Preserve relative positions in saturation and luminance
666
 * space.
667 668 669
 */
function _color_shift($given, $ref1, $ref2, $target) {
  // We assume that ref2 is a blend of ref1 and target and find
670
  // delta based on the length of the difference vectors.
671 672

  // delta = 1 - |ref2 - ref1| / |white - ref1|
673 674 675
  $target = _color_unpack($target, TRUE);
  $ref1 = _color_unpack($ref1, TRUE);
  $ref2 = _color_unpack($ref2, TRUE);
676 677
  $numerator = 0;
  $denominator = 0;
678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697
  for ($i = 0; $i < 3; ++$i) {
    $numerator += ($ref2[$i] - $ref1[$i]) * ($ref2[$i] - $ref1[$i]);
    $denominator += ($target[$i] - $ref1[$i]) * ($target[$i] - $ref1[$i]);
  }
  $delta = ($denominator > 0) ? (1 - sqrt($numerator / $denominator)) : 0;

  // Calculate the color that ref2 would be if the assumption was true.
  for ($i = 0; $i < 3; ++$i) {
    $ref3[$i] = $target[$i] + ($ref1[$i] - $target[$i]) * $delta;
  }

  // If the assumption is not true, there is a difference between ref2 and ref3.
  // We measure this in HSL space. Notation: x' = hsl(x).
  $ref2 = _color_rgb2hsl($ref2);
  $ref3 = _color_rgb2hsl($ref3);
  for ($i = 0; $i < 3; ++$i) {
    $shift[$i] = $ref2[$i] - $ref3[$i];
  }

  // Take the given color, and blend it towards the target.
698
  $given = _color_unpack($given, TRUE);
699 700 701 702 703 704 705 706 707 708 709 710 711
  for ($i = 0; $i < 3; ++$i) {
    $result[$i] = $target[$i] + ($given[$i] - $target[$i]) * $delta;
  }

  // Finally, we apply the extra shift in HSL space.
  // Note: if ref2 is a pure blend of ref1 and target, then |shift| = 0.
  $result = _color_rgb2hsl($result);
  for ($i = 0; $i < 3; ++$i) {
    $result[$i] = min(1, max(0, $result[$i] + $shift[$i]));
  }
  $result = _color_hsl2rgb($result);

  // Return hex color.
712
  return _color_pack($result, TRUE);
713 714 715
}

/**
716
 * Converts a hex triplet into a GD color.
717 718 719 720 721 722 723
 */
function _color_gd($img, $hex) {
  $c = array_merge(array($img), _color_unpack($hex));
  return call_user_func_array('imagecolorallocate', $c);
}

/**
724
 * Blends two hex colors and returns the GD color.
725 726 727 728 729 730 731 732
 */
function _color_blend($img, $hex1, $hex2, $alpha) {
  $in1 = _color_unpack($hex1);
  $in2 = _color_unpack($hex2);
  $out = array($img);
  for ($i = 0; $i < 3; ++$i) {
    $out[] = $in1[$i] + ($in2[$i] - $in1[$i]) * $alpha;
  }
733

734 735 736 737
  return call_user_func_array('imagecolorallocate', $out);
}

/**
738
 * Converts a hex color into an RGB triplet.
739
 */
740
function _color_unpack($hex, $normalize = FALSE) {
741 742 743 744 745 746 747
  if (strlen($hex) == 4) {
    $hex = $hex[1] . $hex[1] . $hex[2] . $hex[2] . $hex[3] . $hex[3];
  }
  $c = hexdec($hex);
  for ($i = 16; $i >= 0; $i -= 8) {
    $out[] = (($c >> $i) & 0xFF) / ($normalize ? 255 : 1);
  }
748

749 750 751 752
  return $out;
}

/**
753
 * Converts an RGB triplet to a hex color.
754
 */
755
function _color_pack($rgb, $normalize = FALSE) {
756
  $out = 0;
757 758 759
  foreach ($rgb as $k => $v) {
    $out |= (($v * ($normalize ? 255 : 1)) << (16 - $k * 8));
  }
760

761
  return '#' . str_pad(dechex($out), 6, 0, STR_PAD_LEFT);
762 763 764
}

/**
765
 * Converts an HSL triplet into RGB.
766 767 768 769 770 771 772
 */
function _color_hsl2rgb($hsl) {
  $h = $hsl[0];
  $s = $hsl[1];
  $l = $hsl[2];
  $m2 = ($l <= 0.5) ? $l * ($s + 1) : $l + $s - $l*$s;
  $m1 = $l * 2 - $m2;
773

774 775 776 777 778
  return array(
    _color_hue2rgb($m1, $m2, $h + 0.33333),
    _color_hue2rgb($m1, $m2, $h),
    _color_hue2rgb($m1, $m2, $h - 0.33333),
  );
779 780 781 782 783 784 785 786 787 788
}

/**
 * Helper function for _color_hsl2rgb().
 */
function _color_hue2rgb($m1, $m2, $h) {
  $h = ($h < 0) ? $h + 1 : (($h > 1) ? $h - 1 : $h);
  if ($h * 6 < 1) return $m1 + ($m2 - $m1) * $h * 6;
  if ($h * 2 < 1) return $m2;
  if ($h * 3 < 2) return $m1 + ($m2 - $m1) * (0.66666 - $h) * 6;
789

790 791 792 793
  return $m1;
}

/**
794
 * Converts an RGB triplet to HSL.
795 796 797 798 799 800 801 802 803 804
 */
function _color_rgb2hsl($rgb) {
  $r = $rgb[0];
  $g = $rgb[1];
  $b = $rgb[2];
  $min = min($r, min($g, $b));
  $max = max($r, max($g, $b));
  $delta = $max - $min;
  $l = ($min + $max) / 2;
  $s = 0;
805

806
  if ($l > 0 && $l < 1) {
Steven Wittens's avatar
Steven Wittens committed
807
    $s = $delta / ($l < 0.5 ? (2 * $l) : (2 - 2 * $l));
808
  }
809

810 811
  $h = 0;
  if ($delta > 0) {
Steven Wittens's avatar
Steven Wittens committed
812 813 814 815
    if ($max == $r && $max != $g) $h += ($g - $b) / $delta;
    if ($max == $g && $max != $b) $h += (2 + ($b - $r) / $delta);
    if ($max == $b && $max != $r) $h += (4 + ($r - $g) / $delta);
    $h /= 6;
816
  }
817

818
  return array($h, $s, $l);
819
}