Подтвердить что ты не робот

Могу ли я обнаружить анимированные gifs, используя php и gd?

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

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

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

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

Любые подсказки?

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

С уважением,

Kris

4b9b3361

Ответ 2

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

Я воспроизведу его здесь, потому что это может представлять интерес.

источник: 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;
}

Ответ 3

Здесь рабочая функция:

/**
 * 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;
}

Ответ 4

Чтение всего файла с помощью 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;
}

Ответ 5

Это улучшение текущего голосового ответа, но у меня нет достаточной репутации для комментариев. Проблема с этим ответом заключается в том, что он читает файл в блоках 100Kb, а маркер конца кадра может быть разделен на 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;
}

Ответ 6

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

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