Могу ли я обнаружить анимированные картинки с помощью php и gd? - PullRequest
43 голосов
/ 11 ноября 2008

В настоящее время у меня возникают проблемы с изменением размеров изображений с помощью GD.

Все работает нормально, пока я не захочу изменить размер анимированного GIF, который доставляет первый кадр на черном фоне.

Я пытался использовать getimagesize, но это только дает мне размеры и нечего различать между любым рисунком и анимацией.

Фактическое изменение размера не требуется для анимированных GIF-файлов, просто возможность их пропустить будет достаточно для наших целей.

Есть какие-нибудь подсказки?

PS. У меня нет доступа к imagemagick.

С уважением,

Kris

Ответы [ 6 ]

38 голосов
/ 06 января 2009

При поиске решения той же проблемы я заметил, что на сайте php.net есть продолжение кода, на который ссылаются Дэвиде и Крис, но, по мнению автора, он требует меньше памяти и, возможно, меньше диск интенсивно.

Я повторю здесь, потому что это может быть интересно.

источник: http://www.php.net/manual/en/function.imagecreatefromgif.php#88005

function is_ani($filename) {
    if(!($fh = @fopen($filename, 'rb')))
        return false;
    $count = 0;
    //an animated gif contains multiple "frames", with each frame having a
    //header made up of:
    // * a static 4-byte sequence (\x00\x21\xF9\x04)
    // * 4 variable bytes
    // * a static 2-byte sequence (\x00\x2C)

    // We read through the file til we reach the end of the file, or we've found
    // at least 2 frame headers
    while(!feof($fh) && $count < 2) {
        $chunk = fread($fh, 1024 * 100); //read 100kb at a time
        $count += preg_match_all('#\x00\x21\xF9\x04.{4}\x00[\x2C\x21]#s', $chunk, $matches);
    }

    fclose($fh);
    return $count > 1;
}
18 голосов
/ 11 ноября 2008

На странице руководства PHP функции imagecreatefromgif() приведен краткий фрагмент кода, который должен быть тем, что вам нужно:

imagecreatefromgif комментарий # 59787 от ZeBadger

7 голосов
/ 11 ноября 2008

Вот рабочая функция:

/**
 * Thanks to ZeBadger for original example, and Davide Gualano for pointing me to it
 * Original at http://it.php.net/manual/en/function.imagecreatefromgif.php#59787
 **/
function is_animated_gif( $filename )
{
    $raw = file_get_contents( $filename );

    $offset = 0;
    $frames = 0;
    while ($frames < 2)
    {
        $where1 = strpos($raw, "\x00\x21\xF9\x04", $offset);
        if ( $where1 === false )
        {
            break;
        }
        else
        {
            $offset = $where1 + 1;
            $where2 = strpos( $raw, "\x00\x2C", $offset );
            if ( $where2 === false )
            {
                break;
            }
            else
            {
                if ( $where1 + 8 == $where2 )
                {
                    $frames ++;
                }
                $offset = $where2 + 1;
            }
        }
    }

    return $frames > 1;
}
2 голосов
/ 12 февраля 2017

Чтение всего файла с file_get_contents может занять слишком много памяти, если данный файл слишком велик. Я повторно проанализировал ранее заданную функцию, которая считывает достаточное количество байтов для проверки кадров и возвращает, как только она находит как минимум 2 кадра.

<?php
/**
 * Detects animated GIF from given file pointer resource or filename.
 *
 * @param resource|string $file File pointer resource or filename
 * @return bool
 */
function is_animated_gif($file)
{
    $fp = null;

    if (is_string($file)) {
        $fp = fopen($file, "rb");
    } else {
        $fp = $file;

        /* Make sure that we are at the beginning of the file */
        fseek($fp, 0);
    }

    if (fread($fp, 3) !== "GIF") {
        fclose($fp);

        return false;
    }

    $frames = 0;

    while (!feof($fp) && $frames < 2) {
        if (fread($fp, 1) === "\x00") {
            /* Some of the animated GIFs do not contain graphic control extension (starts with 21 f9) */
            if (fread($fp, 1) === "\x2c" || fread($fp, 2) === "\x21\xf9") {
                $frames++;
            }
        }
    }

    fclose($fp);

    return $frames > 1;
}
0 голосов
/ 20 декабря 2017

Это улучшение текущего популярного ответа, но у меня пока недостаточно репутации, чтобы комментировать. Проблема с этим ответом состоит в том, что он читает файл в кусках по 100 Кб, а маркер конца кадра может быть разбит на 2 блока. Для этого необходимо добавить последние 20b предыдущего кадра к следующему:

<?php
function is_ani($filename) {
  if(!($fh = @fopen($filename, 'rb')))
    return false;
  $count = 0;
  //an animated gif contains multiple "frames", with each frame having a
  //header made up of:
  // * a static 4-byte sequence (\x00\x21\xF9\x04)
  // * 4 variable bytes
  // * a static 2-byte sequence (\x00\x2C) (some variants may use \x00\x21 ?)

  // We read through the file til we reach the end of the file, or we've found
  // at least 2 frame headers
  $chunk = false;
  while(!feof($fh) && $count < 2) {
    //add the last 20 characters from the previous string, to make sure the searched pattern is not split.
    $chunk = ($chunk ? substr($chunk, -20) : "") . fread($fh, 1024 * 100); //read 100kb at a time
    $count += preg_match_all('#\x00\x21\xF9\x04.{4}\x00(\x2C|\x21)#s', $chunk, $matches);
  }

  fclose($fh);
  return $count > 1;
}
0 голосов
/ 30 августа 2013

Анимированный GIF должен иметь следующую строку

"\x21\xFF\x0B\x4E\x45\x54\x53\x43\x41\x50\x45\x32\x2E\x30"

Я протестировал несколько анимированных GIF-файлов, и кажется, что строка находится в позиции 781 файла (найден с file_get_contents и strpos)

...