Могу ли я обнаружить анимированные GIF с помощью php и gd?

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

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

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

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

какие-то зацепки?

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

С уважением,

Крис

6 ответов


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

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


при поиске решения той же проблемы я заметил, что 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;
}

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

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

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

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

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

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

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