1 image.test ImageToolkitGdTestCase::testManipulations()

Since PHP can't visually check that our images have been manipulated properly, build a list of expected color values for each of the corners and the expected height and widths for the final images.

File

core/modules/simpletest/tests/image.test, line 255
Tests for core image handling API.

Class

ImageToolkitGdTestCase
Test the core GD image manipulation functions.

Code

function testManipulations() {
  // Typically the corner colors will be unchanged. These colors are in the
  // order of top-left, top-right, bottom-right, bottom-left.
  $default_corners = array($this->red, $this->green, $this->blue, $this->transparent);

  // A list of files that will be tested.
  $files = array(
    'image-test.png',
    'image-test.gif',
    'image-test.jpg',
  );

  // Setup a list of tests to perform on each type.
  $operations = array(
    'resize' => array(
      'function' => 'resize',
      'arguments' => array(20, 10),
      'width' => 20,
      'height' => 10,
      'corners' => $default_corners,
    ),
    'scale_x' => array(
      'function' => 'scale',
      'arguments' => array(20, NULL),
      'width' => 20,
      'height' => 10,
      'corners' => $default_corners,
    ),
    'scale_y' => array(
      'function' => 'scale',
      'arguments' => array(NULL, 10),
      'width' => 20,
      'height' => 10,
      'corners' => $default_corners,
    ),
    'upscale_x' => array(
      'function' => 'scale',
      'arguments' => array(80, NULL, TRUE),
      'width' => 80,
      'height' => 40,
      'corners' => $default_corners,
    ),
    'upscale_y' => array(
      'function' => 'scale',
      'arguments' => array(NULL, 40, TRUE),
      'width' => 80,
      'height' => 40,
      'corners' => $default_corners,
    ),
    'crop' => array(
      'function' => 'crop',
      'arguments' => array(12, 4, 16, 12),
      'width' => 16,
      'height' => 12,
      'corners' => array_fill(0, 4, $this->white),
    ),
    'scale_and_crop' => array(
      'function' => 'scale_and_crop',
      'arguments' => array(10, 8),
      'width' => 10,
      'height' => 8,
      'corners' => array_fill(0, 4, $this->black),
    ),
  );

  // Systems using non-bundled GD2 don't have imagerotate. Test if available.
  // @todo Remove the version check once https://www.drupal.org/node/2918570
  //   is resolved.
  if (function_exists('imagerotate') && (version_compare(PHP_VERSION, '7.0.26', '<') || (version_compare(PHP_VERSION, '7.1', '>=') && version_compare(PHP_VERSION, '7.1.12', '<')))) {
    $operations += array(
      'rotate_5' => array(
        'function' => 'rotate',
        'arguments' => array(5, 'FF00FF'), // Fuchsia background.
        'width' => 41,
        'height' => 23,
        'corners' => array_fill(0, 4, $this->fuchsia),
      ),
      'rotate_90' => array(
        'function' => 'rotate',
        'arguments' => array(90, 0xFF00FF), // Fuchsia background.
        'width' => 20,
        'height' => 40,
        'corners' => array($this->fuchsia, $this->red, $this->green, $this->blue),
      ),
      'rotate_transparent_5' => array(
        'function' => 'rotate',
        'arguments' => array(5),
        'width' => 41,
        'height' => 23,
        'corners' => array_fill(0, 4, $this->transparent),
      ),
      'rotate_transparent_90' => array(
        'function' => 'rotate',
        'arguments' => array(90),
        'width' => 20,
        'height' => 40,
        'corners' => array($this->transparent, $this->red, $this->green, $this->blue),
      ),
    );
  }

  // Systems using non-bundled GD2 don't have imagefilter. Test if available.
  if (function_exists('imagefilter')) {
    $operations += array(
      'desaturate' => array(
        'function' => 'desaturate',
        'arguments' => array(),
        'height' => 20,
        'width' => 40,
        // Grayscale corners are a bit funky. Each of the corners are a shade
        // of gray. The values of these were determined by looking at the
        // final image to see what desaturated colors end up being.
        'corners' => array(
          array_fill(0, 3, 76) + array(3 => 0),
          array_fill(0, 3, 149) + array(3 => 0),
          array_fill(0, 3, 29) + array(3 => 0),
          array_fill(0, 3, 0) + array(3 => 127)
        ),
      ),
    );
  }

  foreach ($files as $file) {
    foreach ($operations as $op => $values) {
      // Load up a fresh image.
      $image = image_load(backdrop_get_path('module', 'simpletest') . '/files/' . $file, 'gd');
      if (!$image) {
        $this->fail(t('Could not load image %file.', array('%file' => $file)));
        continue 2;
      }

      // Transparent GIFs and the imagefilter function don't work together.
      // There is a todo in image.gd.inc to correct this.
      if ($image->info['extension'] == 'gif') {
        if ($op == 'desaturate') {
          $values['corners'][3] = $this->white;
        }
      }

      // Perform our operation.
      $function = 'image_' . $values['function'];
      $arguments = array();
      $arguments[] = &$image;
      $arguments = array_merge($arguments, $values['arguments']);
      call_user_func_array($function, $arguments);

      // To keep from flooding the test with assert values, make a general
      // value for whether each group of values fail.
      $correct_dimensions_real = TRUE;
      $correct_dimensions_object = TRUE;
      $correct_colors = TRUE;

      // Collect the actual size of the images on disk.
      $width_actual = imagesx($image->resource);
      $height_actual = imagesy($image->resource);

      // PHP 5.6 and earlier rotate differently than newer PHP versions.
      // See https://bugs.php.net/bug.php?id=65148. For the test images, the
      // dimensions may be 1 pixel larger in both dimensions (though other
      // tests have shown a difference of 0 to 3 pixels in both dimensions).
      if (version_compare(PHP_VERSION, '7.0', '<') && $values['function'] === 'rotate' && $values['arguments'][0] % 90 != 0) {
        $values['width'] = $width_actual;
        $values['height'] = $height_actual;
      }

      // Check that the actual dimensions match the expected ones.
      if ($width_actual != $values['width'] || $height_actual != $values['height']) {
        $correct_dimensions_real = FALSE;
      }

      // Check that the image object has an accurate record of the dimensions.
      if ($image->info['width'] != $values['width'] || $image->info['height'] != $values['height']) {
        $correct_dimensions_object = FALSE;
      }
      // Now check each of the corners to ensure color correctness.
      foreach ($values['corners'] as $key => $corner) {
        // Get the location of the corner.
        switch ($key) {
          case 0:
            $x = 0;
            $y = 0;
            break;
          case 1:
            $x = $values['width'] - 1;
            $y = 0;
            break;
          case 2:
            $x = $values['width'] - 1;
            $y = $values['height'] - 1;
            break;
          case 3:
            $x = 0;
            $y = $values['height'] - 1;
            break;
        }
        $color = $this->getPixelColor($image, $x, $y);
        $correct_colors = $this->colorsAreEqual($color, $corner);
      }

      $directory = file_default_scheme() . '://image_tests';
      file_prepare_directory($directory, FILE_CREATE_DIRECTORY);
      image_save($image, $directory . '/' . $op . '.' . $image->info['extension']);

      $this->assertTrue($correct_dimensions_real, format_string('Image %file after %action action has proper dimensions.', array('%file' => $file, '%action' => $op)));
      $this->assertTrue($correct_dimensions_object, format_string('Image %file object after %action action is reporting the proper height and width values.', array('%file' => $file, '%action' => $op)));
      // JPEG colors will always be messed up due to compression.
      if (!in_array($image->info['extension'], array('jpg', 'jpeg'))) {
        $this->assertTrue($correct_colors, format_string('Image %file object after %action action has the correct color placement.', array('%file' => $file, '%action' => $op)));
      }
    }
  }
}