[4660] | 1 | <?php /* |
---|
| 2 | Plugin Name: Exif View |
---|
[13181] | 3 | Version: auto |
---|
[4660] | 4 | Description: Converts EXIF values to human readable localized values. Corresponds to EXIF specification 2.2, details in http://www.exif.org. Easily extensible. |
---|
[6989] | 5 | Plugin URI: http://piwigo.org/ext/extension_view.php?eid=155 |
---|
[4660] | 6 | Author: Martin Javorek |
---|
| 7 | Author URI: mailto:maple@seznam.cz&subject=PWG%20EXIF%20View |
---|
| 8 | */ |
---|
| 9 | |
---|
| 10 | /* |
---|
| 11 | ------------------------------------------------------------------------------- |
---|
| 12 | Change log: |
---|
| 13 | |
---|
| 14 | 0.2, 23th August 2007 |
---|
| 15 | - exposurue bias fix, date time original formatting |
---|
| 16 | |
---|
| 17 | 0.1, 1st August 2007 |
---|
| 18 | - initial version |
---|
| 19 | |
---|
| 20 | |
---|
| 21 | ------------------------------------------------------------------------------- |
---|
| 22 | |
---|
| 23 | Extend 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 | |
---|
| 51 | add_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 | */ |
---|
| 57 | define('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 | */ |
---|
| 65 | function 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 | */ |
---|
| 75 | function 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 | */ |
---|
| 90 | function 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)) { |
---|
[6989] | 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; |
---|
[4660] | 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 { |
---|
[7276] | 115 | return '1/'.floor(1/($tokens[0]/$tokens[1])).' s'; |
---|
| 116 | // return $tokens[0].'/'.$tokens[1].' s'; |
---|
[4660] | 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); |
---|
[20066] | 211 | return (round($tokens[0]/$tokens[1])).' mm'; |
---|
[4660] | 212 | } |
---|
| 213 | |
---|
| 214 | // digital zoom |
---|
| 215 | if (!(strpos($key, 'DigitalZoomRatio') === FALSE)) { |
---|
| 216 | $tokens = explode('/', $value); |
---|
| 217 | return ($tokens[0]/$tokens[1]); |
---|
| 218 | } |
---|
| 219 | |
---|
[13180] | 220 | // distance to subject |
---|
| 221 | if (!(strpos($key, 'SubjectDistance') === FALSE)) { |
---|
| 222 | $tokens = explode('/', $value); |
---|
[13192] | 223 | if (isset($tokens[1])) |
---|
| 224 | { |
---|
| 225 | $distance = $tokens[0]/$tokens[1]; |
---|
| 226 | } |
---|
| 227 | else |
---|
| 228 | { |
---|
| 229 | $distance = $value; |
---|
| 230 | } |
---|
| 231 | return $distance.' m'; |
---|
[13180] | 232 | } |
---|
| 233 | |
---|
[4660] | 234 | // white balance |
---|
| 235 | if (!(strpos($key, 'WhiteBalance') === FALSE)) { |
---|
| 236 | switch ($value) { |
---|
| 237 | case 0: return l10n('exif_value_white_balance_auto'); |
---|
| 238 | case 1: return l10n('exif_value_white_balance_manual'); |
---|
| 239 | default: return ''; |
---|
| 240 | } |
---|
| 241 | } |
---|
| 242 | |
---|
| 243 | // exposure mode |
---|
| 244 | if (!(strpos($key, 'ExposureMode') === FALSE)) { |
---|
| 245 | switch ($value) { |
---|
| 246 | case 0: return l10n('exif_value_exposure_mode_auto'); |
---|
| 247 | case 1: return l10n('exif_value_exposure_mode_manual'); |
---|
| 248 | case 2: return l10n('exif_value_exposure_mode_auto_bracket'); |
---|
| 249 | default: return ''; |
---|
| 250 | } |
---|
| 251 | } |
---|
| 252 | |
---|
| 253 | // exposure metering mode |
---|
| 254 | if (!(strpos($key, 'MeteringMode') === FALSE)) { |
---|
| 255 | switch ($value) { |
---|
| 256 | case 0: return l10n('exif_value_metering_mode_unknown'); |
---|
| 257 | case 1: return l10n('exif_value_metering_mode_average'); |
---|
| 258 | case 2: return l10n('exif_value_metering_mode_CenterWeightedAVG'); |
---|
| 259 | case 3: return l10n('exif_value_metering_mode_spot'); |
---|
| 260 | case 4: return l10n('exif_value_metering_mode_multispot'); |
---|
| 261 | case 5: return l10n('exif_value_metering_mode_pattern'); |
---|
| 262 | case 6: return l10n('exif_value_metering_mode_partial'); |
---|
| 263 | default: return ''; |
---|
| 264 | } |
---|
| 265 | } |
---|
| 266 | |
---|
| 267 | // exposure program |
---|
| 268 | if (!(strpos($key, 'ExposureProgram') === FALSE)) { |
---|
| 269 | switch ($value) { |
---|
| 270 | case 0: return l10n('exif_value_exposure_program_not_defined'); |
---|
| 271 | case 1: return l10n('exif_value_exposure_program_manual'); |
---|
| 272 | case 2: return l10n('exif_value_exposure_program_normal'); |
---|
| 273 | case 3: return l10n('exif_value_exposure_program_aperture'); |
---|
| 274 | case 4: return l10n('exif_value_exposure_program_shutter'); |
---|
| 275 | case 5: return l10n('exif_value_exposure_program_creative'); |
---|
| 276 | case 6: return l10n('exif_value_exposure_program_action'); |
---|
| 277 | case 7: return l10n('exif_value_exposure_program_portrait'); |
---|
| 278 | case 8: return l10n('exif_value_exposure_program_landscape'); |
---|
| 279 | default: return ''; |
---|
| 280 | } |
---|
| 281 | } |
---|
| 282 | |
---|
| 283 | // light source |
---|
| 284 | if (!(strpos($key, 'LightSource') === FALSE)) { |
---|
| 285 | switch ($value) { |
---|
| 286 | case 0: return l10n('exif_value_light_source_unknown'); |
---|
| 287 | case 1: return l10n('exif_value_light_source_daylight'); |
---|
| 288 | case 2: return l10n('exif_value_light_source_fluorescent'); |
---|
| 289 | case 3: return l10n('exif_value_light_source_tungsten'); |
---|
| 290 | case 4: return l10n('exif_value_light_source_flash'); |
---|
| 291 | case 9: return l10n('exif_value_light_source_fine_weather'); |
---|
| 292 | case 10: return l10n('exif_value_light_source_cloudy_weather'); |
---|
| 293 | case 11: return l10n('exif_value_light_source_shade'); |
---|
| 294 | case 12: return l10n('exif_value_light_source_daylight_fluorescent_d'); |
---|
| 295 | case 13: return l10n('exif_value_light_source_daywhite_fluorescent_n'); |
---|
| 296 | case 14: return l10n('exif_value_light_source_coolwhite_fluorescent_w'); |
---|
| 297 | case 15: return l10n('exif_value_light_source_white_fluorescent'); |
---|
| 298 | case 17: return l10n('exif_value_light_source_standard_light_a'); |
---|
| 299 | case 18: return l10n('exif_value_light_source_standard_light_b'); |
---|
| 300 | case 19: return l10n('exif_value_light_source_standard_light_c'); |
---|
| 301 | case 20: return l10n('exif_value_light_source_D55'); |
---|
| 302 | case 21: return l10n('exif_value_light_source_D65'); |
---|
| 303 | case 22: return l10n('exif_value_light_source_D75'); |
---|
| 304 | case 23: return l10n('exif_value_light_source_D50'); |
---|
| 305 | case 24: return l10n('exif_value_light_source_iso_studio_tungsten'); |
---|
| 306 | case 255: return l10n('exif_value_light_source_other'); |
---|
| 307 | default: return ''; |
---|
| 308 | } |
---|
| 309 | } |
---|
| 310 | |
---|
| 311 | // contrast |
---|
| 312 | if (!(strpos($key, 'Contrast') === FALSE)) { |
---|
| 313 | switch ($value) { |
---|
| 314 | case 0: return l10n('exif_value_contrast_normal'); |
---|
| 315 | case 1: return l10n('exif_value_contrast_soft'); |
---|
| 316 | case 2: return l10n('exif_value_contrast_hard'); |
---|
| 317 | default: return ''; |
---|
| 318 | } |
---|
| 319 | } |
---|
| 320 | |
---|
| 321 | // sharpness |
---|
| 322 | if (!(strpos($key, 'Sharpness') === FALSE)) { |
---|
| 323 | switch ($value) { |
---|
| 324 | case 0: return l10n('exif_value_sharpness_normal'); |
---|
| 325 | case 1: return l10n('exif_value_sharpness_soft'); |
---|
| 326 | case 2: return l10n('exif_value_sharpness_hard'); |
---|
| 327 | default: return ''; |
---|
| 328 | } |
---|
| 329 | } |
---|
| 330 | |
---|
| 331 | // saturation |
---|
| 332 | if (!(strpos($key, 'Saturation') === FALSE)) { |
---|
| 333 | switch ($value) { |
---|
| 334 | case 0: return l10n('exif_value_saturation_normal'); |
---|
| 335 | case 1: return l10n('exif_value_saturation_low'); |
---|
| 336 | case 2: return l10n('exif_value_saturation_hard'); |
---|
| 337 | default: return ''; |
---|
| 338 | } |
---|
| 339 | } |
---|
| 340 | |
---|
| 341 | // return value unchanged |
---|
| 342 | return $value; |
---|
| 343 | } |
---|
| 344 | define('exif_DIR' , basename(dirname(__FILE__))); |
---|
| 345 | define('exif_PATH' , PHPWG_PLUGINS_PATH . exif_DIR . '/'); |
---|
| 346 | /** |
---|
| 347 | * Loads plugin language file. |
---|
| 348 | */ |
---|
| 349 | function loadLang() { |
---|
| 350 | global $lang; |
---|
| 351 | load_language('lang.exif', exif_PATH); |
---|
| 352 | } |
---|
| 353 | |
---|
| 354 | /** |
---|
| 355 | * EXIF translation. |
---|
| 356 | * |
---|
| 357 | * @param $key EXIF key name |
---|
| 358 | * @param $value EXIF key value |
---|
| 359 | * @return translated value dependend on key meaning and choosed language |
---|
| 360 | */ |
---|
| 361 | function exif_translation($exif) { |
---|
| 362 | // translate all exif fields |
---|
| 363 | if (is_array($exif)) { |
---|
| 364 | loadLang(); |
---|
| 365 | foreach ($exif as $key => $value) { |
---|
| 366 | $exif[$key] = exif_key_translation($key, $value); |
---|
| 367 | } |
---|
| 368 | } |
---|
| 369 | return $exif; |
---|
| 370 | } |
---|
| 371 | |
---|
| 372 | ?> |
---|