source: extensions/exif_view/main.inc.php @ 12998

Last change on this file since 12998 was 7276, checked in by plg, 14 years ago

bug 1934 fixed: improve display for weird raw values from Nikon cameras on ExposureTime

  • Property svn:eol-style set to LF
  • Property svn:keywords set to Author Date Id Revision
File size: 11.1 KB
Line 
1<?php /*
2Plugin Name: Exif View
3Version: auto
4Description: Converts EXIF values to human readable localized values. Corresponds to EXIF specification 2.2, details in http://www.exif.org. Easily extensible.
5Plugin URI: http://piwigo.org/ext/extension_view.php?eid=155
6Author: Martin Javorek
7Author URI: mailto:maple@seznam.cz&subject=PWG%20EXIF%20View
8*/
9
10/*
11-------------------------------------------------------------------------------
12Change log:
13
140.2, 23th August 2007
15- exposurue bias fix, date time original formatting
16
170.1, 1st August 2007
18- initial version
19
20
21-------------------------------------------------------------------------------
22
23Extend your configuration in /include/config.local.inc.php file - example:
24
25$conf['show_exif_fields'] = array(
26  'Make',
27  'Model',
28  'ExifVersion',
29  'Software',
30  'DateTimeOriginal',
31  'FNumber',
32  'ExposureBiasValue',
33  'FILE;FileSize',
34  'ExposureTime',
35  'Flash',
36  'ISOSpeedRatings',
37  'FocalLength',
38  'FocalLengthIn35mmFilm',
39  'WhiteBalance',
40  'ExposureMode',
41  'MeteringMode',
42  'ExposureProgram',
43  'LightSource',
44  'Contrast',
45  'Saturation',
46  'Sharpness',
47  );
48
49*/
50
51add_event_handler('format_exif_data', 'exif_translation' );
52
53/**
54 * Date and time format.
55 * @see http://cz2.php.net/manual/en/function.date.php
56 */
57define('DATE_TIME_FORMAT', 'H:i:s j.n.Y');
58
59/**
60 * Truncates number.
61 *
62 * @param num number
63 * @param digits number of digits, default 0
64 */
65function truncate($num, $digits = 0) {
66    $shift = pow(10 , $digits);
67    return ((floor($num * $shift)) / $shift);
68}
69
70/**
71 * Format date.
72 *
73 * @param date given EXIF date
74 */
75function formatDate($date) {
76        $dateTime = explode(' ', $date);
77        $d = explode(':', $dateTime[0]);
78        $t = explode(':', $dateTime[1]);
79        // beware of american date format for mktime, it accepts date in M/D/Y ;-)
80        return date(DATE_TIME_FORMAT, mktime($t[0], $t[1], $t[2], $d[1], $d[2], $d[0]));
81}
82
83/**
84 * EXIF translation.
85 *
86 * @param $key EXIF key name
87 * @param $value EXIF key value
88 * @return translated value depending on key meaning and choosed language
89 */
90function exif_key_translation($key, $value) {
91   // EXIF
92        if (!(strpos($key, 'ExifVersion') === FALSE)) {
93      return $value[1].'.'.$value[2];
94   }
95   
96   // Date Time Original
97   if (!(strpos($key, 'DateTimeOriginal') === FALSE)) {
98     // to fix bug:1862 the easiest way without releasing a new version of
99     // Piwigo itself, it's better to bypass the date format function
100     //
101     // return formatDate($value);
102     return $value;
103   }
104
105   // exposure time
106         if (!(strpos($key, 'ExposureTime') === FALSE)) {
107      $tokens = explode('/', $value);
108      while ($tokens[0] % 10 == 0) {
109         $tokens[0] = $tokens[0] / 10;
110         $tokens[1] = $tokens[1] / 10;
111      }
112      if ($tokens[1] == 1) {
113         return $tokens[0].' s';
114      } else {
115        return '1/'.floor(1/($tokens[0]/$tokens[1])).' s';
116         // return $tokens[0].'/'.$tokens[1].' s';
117      }
118   }
119
120   // aperture
121         if (!(strpos($key, 'FNumber') === FALSE)) {
122      $tokens = explode('/', $value);
123      return $tokens[0]/$tokens[1];
124   }
125
126   // flash
127   if (!(strpos($key, 'Flash') === FALSE)) {
128      // 1st bit is fired/did not fired
129      if (($value & 1) > 0) {
130         $retValue = l10n('yes');
131      } else {
132         $retValue = l10n('no');
133      }
134      // 2nd+3rd bits are return light mode
135      $returnLight = $value & (3 << 1);
136      switch ($returnLight) {
137        case 2 << 1: $retValue .= ', '.l10n('exif_value_flash_return_light_not_detected');break;
138        case 3 << 1: $retValue .= ', '.l10n('exif_value_flash_return_light_detected');break;
139      }
140      // 4th+5th bits are mode
141      $mode = $value & (3 << 3);
142      switch ($mode) {
143        case 0: $retValue .= ', '.l10n('exif_value_flash_mode').': '.l10n('exif_value_flash_mode_unknown');break;
144        case 1 << 3: $retValue .= ', '.l10n('exif_value_flash_mode').': '.l10n('exif_value_flash_mode_compulsory');break;
145        case 2 << 3: $retValue .= ', '.l10n('exif_value_flash_mode').': '.l10n('exif_value_flash_mode_supress');break;
146        case 3 << 3: $retValue .= ', '.l10n('exif_value_flash_mode').': '.l10n('exif_value_flash_mode_auto');break;
147      }
148                        // 6th bit is red eye function
149      if (($value & (1 << 6)) > 0) {
150         $retValue .= ', '.l10n('exif_value_red_eye');
151      }
152      return $retValue;
153   }
154
155   // exposure bias
156   if (!(strpos($key, 'ExposureBiasValue') === FALSE)) {
157      $tokens = explode('/', $value);
158      $newValue = $tokens[0] / $tokens[1];
159      // max EV range +-
160      $maxEV = 5;
161      // default value
162      $retValue = $newValue;
163      $absValue = truncate(abs($newValue), 2);
164      $found = FALSE;
165      // find through 1/3
166      for ($i = 1; $i <= $maxEV * 3 ; $i++) {
167         $ev = floor($i * 1/3.0 * 100) / 100;
168         if ($ev == $absValue) {
169            if ($i > 3) {
170               $retValue = (truncate($i / 3)).' '.($i % 3).'/3';
171            } else {
172               $retValue = $i.'/3';
173            }
174            $found = TRUE;
175            break;
176         }
177      }
178      // find through 1/2
179      if (!$found) {
180         for ($i = 1; $i <= $maxEV * 2 ; $i++) {
181            $ev = floor($i * 1/2.0 * 100) / 100;
182            if ($ev == $absValue) {
183               if ($i > 2) {
184                  $retValue = ($i / 2).' '.($i % 2).'/2';
185               } else {
186                  $retValue = $i.'/2';
187               }
188               $found = TRUE;
189               break;
190            }
191         }
192      }
193      // signs
194      if (($newValue < 0) && $found) {
195         $retValue = '- '.$retValue;
196      }
197      if ($newValue > 0) {
198         $retValue = '+ '.$retValue;
199      }
200      return $retValue.' EV';
201   }
202
203   // focal length 35mm
204   if (!(strpos($key, 'FocalLengthIn35mmFilm') === FALSE)) {
205      return $value.' mm';
206   }
207
208   // focal length
209   if (!(strpos($key, 'FocalLength') === FALSE)) {
210      $tokens = explode('/', $value);
211      return ($tokens[0]/$tokens[1]).' mm';
212   }
213
214   // digital zoom
215   if (!(strpos($key, 'DigitalZoomRatio') === FALSE)) {
216      $tokens = explode('/', $value);
217      return ($tokens[0]/$tokens[1]);
218   }
219
220   // white balance
221   if (!(strpos($key, 'WhiteBalance') === FALSE)) {
222      switch ($value) {
223         case 0: return l10n('exif_value_white_balance_auto');
224         case 1: return l10n('exif_value_white_balance_manual');
225         default: return '';
226      }
227   }
228
229   // exposure mode
230   if (!(strpos($key, 'ExposureMode') === FALSE)) {
231      switch ($value) {
232         case 0: return l10n('exif_value_exposure_mode_auto');
233         case 1: return l10n('exif_value_exposure_mode_manual');
234         case 2: return l10n('exif_value_exposure_mode_auto_bracket');
235         default: return '';
236      }
237   }
238
239   // exposure metering mode
240   if (!(strpos($key, 'MeteringMode') === FALSE)) {
241      switch ($value) {
242         case 0: return l10n('exif_value_metering_mode_unknown');
243         case 1: return l10n('exif_value_metering_mode_average');
244         case 2: return l10n('exif_value_metering_mode_CenterWeightedAVG');
245         case 3: return l10n('exif_value_metering_mode_spot');
246         case 4: return l10n('exif_value_metering_mode_multispot');
247         case 5: return l10n('exif_value_metering_mode_pattern');
248         case 6: return l10n('exif_value_metering_mode_partial');
249         default: return '';
250      }
251   }
252
253   // exposure program
254   if (!(strpos($key, 'ExposureProgram') === FALSE)) {
255      switch ($value) {
256         case 0: return l10n('exif_value_exposure_program_not_defined');
257         case 1: return l10n('exif_value_exposure_program_manual');
258         case 2: return l10n('exif_value_exposure_program_normal');
259         case 3: return l10n('exif_value_exposure_program_aperture');
260         case 4: return l10n('exif_value_exposure_program_shutter');
261         case 5: return l10n('exif_value_exposure_program_creative');
262         case 6: return l10n('exif_value_exposure_program_action');
263         case 7: return l10n('exif_value_exposure_program_portrait');
264         case 8: return l10n('exif_value_exposure_program_landscape');
265         default: return '';
266      }
267   }
268   
269   // light source
270   if (!(strpos($key, 'LightSource') === FALSE)) {
271      switch ($value) {
272         case 0: return l10n('exif_value_light_source_unknown');
273         case 1: return l10n('exif_value_light_source_daylight');
274         case 2: return l10n('exif_value_light_source_fluorescent');
275         case 3: return l10n('exif_value_light_source_tungsten');
276         case 4: return l10n('exif_value_light_source_flash');
277         case 9: return l10n('exif_value_light_source_fine_weather');
278         case 10: return l10n('exif_value_light_source_cloudy_weather');
279         case 11: return l10n('exif_value_light_source_shade');
280         case 12: return l10n('exif_value_light_source_daylight_fluorescent_d');
281         case 13: return l10n('exif_value_light_source_daywhite_fluorescent_n');
282         case 14: return l10n('exif_value_light_source_coolwhite_fluorescent_w');
283         case 15: return l10n('exif_value_light_source_white_fluorescent');
284         case 17: return l10n('exif_value_light_source_standard_light_a');
285         case 18: return l10n('exif_value_light_source_standard_light_b');
286         case 19: return l10n('exif_value_light_source_standard_light_c');
287         case 20: return l10n('exif_value_light_source_D55');
288         case 21: return l10n('exif_value_light_source_D65');
289         case 22: return l10n('exif_value_light_source_D75');
290         case 23: return l10n('exif_value_light_source_D50');
291         case 24: return l10n('exif_value_light_source_iso_studio_tungsten');
292         case 255: return l10n('exif_value_light_source_other');
293         default: return '';
294      }
295   }
296
297   // contrast
298   if (!(strpos($key, 'Contrast') === FALSE)) {
299      switch ($value) {
300         case 0: return l10n('exif_value_contrast_normal');
301         case 1: return l10n('exif_value_contrast_soft');
302         case 2: return l10n('exif_value_contrast_hard');
303         default: return '';
304      }
305   }
306
307   // sharpness
308   if (!(strpos($key, 'Sharpness') === FALSE)) {
309      switch ($value) {
310         case 0: return l10n('exif_value_sharpness_normal');
311         case 1: return l10n('exif_value_sharpness_soft');
312         case 2: return l10n('exif_value_sharpness_hard');
313         default: return '';
314      }
315   }
316
317   // saturation
318   if (!(strpos($key, 'Saturation') === FALSE)) {
319      switch ($value) {
320         case 0: return l10n('exif_value_saturation_normal');
321         case 1: return l10n('exif_value_saturation_low');
322         case 2: return l10n('exif_value_saturation_hard');
323         default: return '';
324      }
325   }
326
327   // return value unchanged
328   return $value;
329}
330define('exif_DIR' , basename(dirname(__FILE__)));
331define('exif_PATH' , PHPWG_PLUGINS_PATH . exif_DIR . '/');
332        /**
333         * Loads plugin language file.
334         */
335  function loadLang() {
336    global $lang;
337    load_language('lang.exif', exif_PATH);
338  }
339
340/**
341 * EXIF translation.
342 *
343 * @param $key EXIF key name
344 * @param $value EXIF key value
345 * @return translated value dependend on key meaning and choosed language
346 */
347function exif_translation($exif) {
348         // translate all exif fields
349         if (is_array($exif)) {
350         loadLang();
351           foreach ($exif as $key => $value) {
352                         $exif[$key] = exif_key_translation($key, $value);
353           }
354         }
355   return $exif;
356}
357
358?>
Note: See TracBrowser for help on using the repository browser.