2010-02-03 90 views
11

這是我到目前爲止有:如何從PHP讀取PNG元數據?

<?php 

$file = "18201010338AM16390621000846.png"; 

$test = file_get_contents($file, FILE_BINARY); 

echo str_replace("\n","<br>",$test); 

?> 

輸出是八九不離十我想要的,但我真的只需要3-7行(包括性)。這就是現在輸出的結果:http://silentnoobs.com/pbss/collector/test.php。我試圖從「PunkBuster Screenshot(±)AAO Bridge Crossing」到「結果:w = 394 X h = 196 sample = 2」獲取數據。我認爲讀取文件是相當直接的,並將每行存儲在一個數組中,第[0]行需要是「PunkBuster截圖(±)AAO Bridge Crossing」等等。所有這些線都會發生變化,所以我不能只搜索有限的東西。

我已經嘗試了幾天,並沒有幫助很多,我很窮的PHP。

+0

對不起,既不瞭解,也沒有目標的問題... – deceze 2010-02-03 07:11:50

+1

PNG被分成塊(http://www.libpng.org/pub/png/spec/ 1.2/PNG-Chunks.html)。你可能正在尋找包含註釋的'tEXt'塊(用'comment'關鍵字表示)。 – Gumbo 2010-02-03 07:24:35

回答

16

PNG file format定義將PNG文檔分成多個數據塊。因此,你必須找到你想要的塊。

想要提取的數據似乎是在tEXt塊中定義的。我已經寫了下面的類來允許你從PNG文件中提取塊。

class PNG_Reader 
{ 
    private $_chunks; 
    private $_fp; 

    function __construct($file) { 
     if (!file_exists($file)) { 
      throw new Exception('File does not exist'); 
     } 

     $this->_chunks = array(); 

     // Open the file 
     $this->_fp = fopen($file, 'r'); 

     if (!$this->_fp) 
      throw new Exception('Unable to open file'); 

     // Read the magic bytes and verify 
     $header = fread($this->_fp, 8); 

     if ($header != "\x89PNG\x0d\x0a\x1a\x0a") 
      throw new Exception('Is not a valid PNG image'); 

     // Loop through the chunks. Byte 0-3 is length, Byte 4-7 is type 
     $chunkHeader = fread($this->_fp, 8); 

     while ($chunkHeader) { 
      // Extract length and type from binary data 
      $chunk = @unpack('Nsize/a4type', $chunkHeader); 

      // Store position into internal array 
      if ($this->_chunks[$chunk['type']] === null) 
       $this->_chunks[$chunk['type']] = array(); 
      $this->_chunks[$chunk['type']][] = array (
       'offset' => ftell($this->_fp), 
       'size' => $chunk['size'] 
      ); 

      // Skip to next chunk (over body and CRC) 
      fseek($this->_fp, $chunk['size'] + 4, SEEK_CUR); 

      // Read next chunk header 
      $chunkHeader = fread($this->_fp, 8); 
     } 
    } 

    function __destruct() { fclose($this->_fp); } 

    // Returns all chunks of said type 
    public function get_chunks($type) { 
     if ($this->_chunks[$type] === null) 
      return null; 

     $chunks = array(); 

     foreach ($this->_chunks[$type] as $chunk) { 
      if ($chunk['size'] > 0) { 
       fseek($this->_fp, $chunk['offset'], SEEK_SET); 
       $chunks[] = fread($this->_fp, $chunk['size']); 
      } else { 
       $chunks[] = ''; 
      } 
     } 

     return $chunks; 
    } 
} 

你可以使用它是這樣來提取所需tEXt塊這樣:

$file = '18201010338AM16390621000846.png'; 
$png = new PNG_Reader($file); 

$rawTextData = $png->get_chunks('tEXt'); 

$metadata = array(); 

foreach($rawTextData as $data) { 
    $sections = explode("\0", $data); 

    if($sections > 1) { 
     $key = array_shift($sections); 
     $metadata[$key] = implode("\0", $sections); 
    } else { 
     $metadata[] = $data; 
    } 
} 
+0

如果他只需要tEXt塊,加載整個PNG數據會浪費內存(即'fread' vs'fseek')。 – Matthew 2010-02-03 08:03:24

+0

@konforce:我重新考慮了這個類只存儲塊的偏移量。它們是在有用的基礎上閱讀的。我想盡可能保持上述課程的多樣性。 – 2010-02-03 08:23:40

2
<?php 
    $fp = fopen('18201010338AM16390621000846.png', 'rb'); 
    $sig = fread($fp, 8); 
    if ($sig != "\x89PNG\x0d\x0a\x1a\x0a") 
    { 
    print "Not a PNG image"; 
    fclose($fp); 
    die(); 
    } 

    while (!feof($fp)) 
    { 
    $data = unpack('Nlength/a4type', fread($fp, 8)); 
    if ($data['type'] == 'IEND') break; 

    if ($data['type'] == 'tEXt') 
    { 
     list($key, $val) = explode("\0", fread($fp, $data['length'])); 
     echo "<h1>$key</h1>"; 
     echo nl2br($val); 

     fseek($fp, 4, SEEK_CUR); 
    } 
    else 
    { 
     fseek($fp, $data['length'] + 4, SEEK_CUR); 
    } 
    } 


    fclose($fp); 
?> 

它假定基本形成良好的PNG文件。