MediaWiki master
FormatMetadata.php
Go to the documentation of this file.
1<?php
33use MediaWiki\HookContainer\ProtectedHookAccessorTrait;
37
60 use ProtectedHookAccessorTrait;
61
67 protected $singleLang = false;
68
75 public function setSingleLanguage( $val ) {
76 $this->singleLang = $val;
77 }
78
92 public static function getFormattedData( $tags, $context = false ) {
93 $obj = new self;
94 if ( $context ) {
95 $obj->setContext( $context );
96 }
97
98 return $obj->makeFormattedData( $tags );
99 }
100
112 public function makeFormattedData( $tags ) {
113 $resolutionunit = !isset( $tags['ResolutionUnit'] ) || $tags['ResolutionUnit'] == 2 ? 2 : 3;
114 unset( $tags['ResolutionUnit'] );
115
116 / Ignore these complex values
117 unset( $tags['HasExtendedXMP'] );
118 unset( $tags['AuthorsPosition'] );
119 unset( $tags['LocationCreated'] );
120 unset( $tags['LocationShown'] );
121 unset( $tags['GPSAltitudeRef'] );
122
123 foreach ( $tags as $tag => &$vals ) {
124 / This seems ugly to wrap non-array's in an array just to unwrap again,
125 / especially when most of the time it is not an array
126 if ( !is_array( $vals ) ) {
127 $vals = [ $vals ];
128 }
129
130 / _type is a special value to say what array type
131 if ( isset( $vals['_type'] ) ) {
132 $type = $vals['_type'];
133 unset( $vals['_type'] );
134 } else {
135 $type = 'ul'; / default unordered list.
136 }
137
138 / _formatted is a special value to indicate the subclass
139 / already handled & formatted this tag as wikitext
140 if ( isset( $tags[$tag]['_formatted'] ) ) {
141 $tags[$tag] = $this->flattenArrayReal(
142 $tags[$tag]['_formatted'], $type
143 );
144 continue;
145 }
146
147 / This is done differently as the tag is an array.
148 if ( $tag === 'GPSTimeStamp' && count( $vals ) === 3 ) {
149 / hour min sec array
150
151 $h = explode( '/', $vals[0], 2 );
152 $m = explode( '/', $vals[1], 2 );
153 $s = explode( '/', $vals[2], 2 );
154
155 / this should already be validated
156 / when loaded from file, but it could
157 / come from a foreign repo, so be
158 / paranoid.
159 if ( !isset( $h[1] )
160 || !isset( $m[1] )
161 || !isset( $s[1] )
162 || $h[1] == 0
163 || $m[1] == 0
164 || $s[1] == 0
165 ) {
166 continue;
167 }
168 $vals = str_pad( (string)( (int)$h[0] / (int)$h[1] ), 2, '0', STR_PAD_LEFT )
169 . ':' . str_pad( (string)( (int)$m[0] / (int)$m[1] ), 2, '0', STR_PAD_LEFT )
170 . ':' . str_pad( (string)( (int)$s[0] / (int)$s[1] ), 2, '0', STR_PAD_LEFT );
171
172 $time = wfTimestamp( TS_MW, '1971:01:01 ' . $vals );
173 / the 1971:01:01 is just a placeholder, and not shown to user.
174 if ( $time && (int)$time > 0 ) {
175 $vals = $this->getLanguage()->time( $time );
176 }
177 continue;
178 }
179
180 / The contact info is a multi-valued field
181 / instead of the other props which are single
182 / valued (mostly) so handle as a special case.
183 if ( $tag === 'Contact' || $tag === 'CreatorContactInfo' ) {
184 $vals = $this->collapseContactInfo( $vals );
185 continue;
186 }
187
188 foreach ( $vals as &$val ) {
189 switch ( $tag ) {
190 case 'Compression':
191 switch ( $val ) {
192 case 1:
193 case 2:
194 case 3:
195 case 4:
196 case 5:
197 case 6:
198 case 7:
199 case 8:
200 case 32773:
201 case 32946:
202 case 34712:
203 $val = $this->exifMsg( $tag, $val );
204 break;
205 default:
206 /* If not recognized, display as is. */
207 $val = $this->literal( $val );
208 break;
209 }
210 break;
211
212 case 'PhotometricInterpretation':
213 switch ( $val ) {
214 case 0:
215 case 1:
216 case 2:
217 case 3:
218 case 4:
219 case 5:
220 case 6:
221 case 8:
222 case 9:
223 case 10:
224 case 32803:
225 case 34892:
226 $val = $this->exifMsg( $tag, $val );
227 break;
228 default:
229 /* If not recognized, display as is. */
230 $val = $this->literal( $val );
231 break;
232 }
233 break;
234
235 case 'Orientation':
236 switch ( $val ) {
237 case 1:
238 case 2:
239 case 3:
240 case 4:
241 case 5:
242 case 6:
243 case 7:
244 case 8:
245 $val = $this->exifMsg( $tag, $val );
246 break;
247 default:
248 /* If not recognized, display as is. */
249 $val = $this->literal( $val );
250 break;
251 }
252 break;
253
254 case 'PlanarConfiguration':
255 switch ( $val ) {
256 case 1:
257 case 2:
258 $val = $this->exifMsg( $tag, $val );
259 break;
260 default:
261 /* If not recognized, display as is. */
262 $val = $this->literal( $val );
263 break;
264 }
265 break;
266
267 / TODO: YCbCrSubSampling
268 case 'YCbCrPositioning':
269 switch ( $val ) {
270 case 1:
271 case 2:
272 $val = $this->exifMsg( $tag, $val );
273 break;
274 default:
275 /* If not recognized, display as is. */
276 $val = $this->literal( $val );
277 break;
278 }
279 break;
280
281 case 'XResolution':
282 case 'YResolution':
283 switch ( $resolutionunit ) {
284 case 2:
285 $val = $this->exifMsg( 'XYResolution', 'i', $this->formatNum( $val ) );
286 break;
287 case 3:
288 $val = $this->exifMsg( 'XYResolution', 'c', $this->formatNum( $val ) );
289 break;
290 default:
291 /* If not recognized, display as is. */
292 $val = $this->literal( $val );
293 break;
294 }
295 break;
296
297 / TODO: YCbCrCoefficients #p27 (see annex E)
298 case 'ExifVersion':
299 / PHP likes to be the odd one out with casing of FlashPixVersion;
300 / https://www.exif.org/Exif2-2.PDF#page=32 and
301 / https://www.digitalgalen.net/Documents/External/XMP/XMPSpecificationPart2.pdf#page=51
302 / both use FlashpixVersion. However, since at least 2002, PHP has used FlashPixVersion at
303 / https://github.com/php/php-src/blame/master/ext/exif/exif.c#L725
304 case 'FlashPixVersion':
305 / But we can still get the correct casing from
306 / Wikimedia\XMPReader on PDFs
307 case 'FlashpixVersion':
308 $val = $this->literal( (int)$val / 100 );
309 break;
310
311 case 'ColorSpace':
312 switch ( $val ) {
313 case 1:
314 case 65535:
315 $val = $this->exifMsg( $tag, $val );
316 break;
317 default:
318 /* If not recognized, display as is. */
319 $val = $this->literal( $val );
320 break;
321 }
322 break;
323
324 case 'ComponentsConfiguration':
325 switch ( $val ) {
326 case 0:
327 case 1:
328 case 2:
329 case 3:
330 case 4:
331 case 5:
332 case 6:
333 $val = $this->exifMsg( $tag, $val );
334 break;
335 default:
336 /* If not recognized, display as is. */
337 $val = $this->literal( $val );
338 break;
339 }
340 break;
341
342 case 'DateTime':
343 case 'DateTimeOriginal':
344 case 'DateTimeDigitized':
345 case 'DateTimeReleased':
346 case 'DateTimeExpires':
347 case 'GPSDateStamp':
348 case 'dc-date':
349 case 'DateTimeMetadata':
350 case 'FirstPhotoDate':
351 case 'LastPhotoDate':
352 if ( $val === null ) {
353 / T384879 - we don't need to call literal to turn this into a string, but
354 / we might as well call it for consistency and future proofing of the default value
355 $val = $this->literal( $val );
356 break;
357 }
358
359 if ( $val === '0000:00:00 00:00:00' || $val === ' : : : : ' ) {
360 $val = $this->msg( 'exif-unknowndate' )->text();
361 break;
362 }
363 if ( preg_match(
364 '/^(?:\d{4}):(?:\d\d):(?:\d\d) (?:\d\d):(?:\d\d):(?:\d\d)$/D',
365 $val
366 ) ) {
367 / Full date.
368 $time = wfTimestamp( TS_MW, $val );
369 if ( $time && (int)$time > 0 ) {
370 $val = $this->getLanguage()->timeanddate( $time );
371 break;
372 }
373 } elseif ( preg_match( '/^(?:\d{4}):(?:\d\d):(?:\d\d) (?:\d\d):(?:\d\d)$/D', $val ) ) {
374 / No second field. Still format the same
375 / since timeanddate doesn't include seconds anyways,
376 / but second still available in api
377 $time = wfTimestamp( TS_MW, $val . ':00' );
378 if ( $time && (int)$time > 0 ) {
379 $val = $this->getLanguage()->timeanddate( $time );
380 break;
381 }
382 } elseif ( preg_match( '/^(?:\d{4}):(?:\d\d):(?:\d\d)$/D', $val ) ) {
383 / If only the date but not the time is filled in.
384 $time = wfTimestamp( TS_MW, substr( $val, 0, 4 )
385 . substr( $val, 5, 2 )
386 . substr( $val, 8, 2 )
387 . '000000' );
388 if ( $time && (int)$time > 0 ) {
389 $val = $this->getLanguage()->date( $time );
390 break;
391 }
392 }
393 / else it will just output $val without formatting it.
394 $val = $this->literal( $val );
395 break;
396
397 case 'ExposureProgram':
398 switch ( $val ) {
399 case 0:
400 case 1:
401 case 2:
402 case 3:
403 case 4:
404 case 5:
405 case 6:
406 case 7:
407 case 8:
408 $val = $this->exifMsg( $tag, $val );
409 break;
410 default:
411 /* If not recognized, display as is. */
412 $val = $this->literal( $val );
413 break;
414 }
415 break;
416
417 case 'SubjectDistance':
418 $val = $this->exifMsg( $tag, '', $this->formatNum( $val ) );
419 break;
420
421 case 'MeteringMode':
422 switch ( $val ) {
423 case 0:
424 case 1:
425 case 2:
426 case 3:
427 case 4:
428 case 5:
429 case 6:
430 case 7:
431 case 255:
432 $val = $this->exifMsg( $tag, $val );
433 break;
434 default:
435 /* If not recognized, display as is. */
436 $val = $this->literal( $val );
437 break;
438 }
439 break;
440
441 case 'LightSource':
442 switch ( $val ) {
443 case 0:
444 case 1:
445 case 2:
446 case 3:
447 case 4:
448 case 9:
449 case 10:
450 case 11:
451 case 12:
452 case 13:
453 case 14:
454 case 15:
455 case 17:
456 case 18:
457 case 19:
458 case 20:
459 case 21:
460 case 22:
461 case 23:
462 case 24:
463 case 255:
464 $val = $this->exifMsg( $tag, $val );
465 break;
466 default:
467 /* If not recognized, display as is. */
468 $val = $this->literal( $val );
469 break;
470 }
471 break;
472
473 case 'Flash':
474 if ( $val === '' ) {
475 $val = 0;
476 }
477 $flashDecode = [
478 'fired' => $val & 0b00000001,
479 'return' => ( $val & 0b00000110 ) >> 1,
480 'mode' => ( $val & 0b00011000 ) >> 3,
481 'function' => ( $val & 0b00100000 ) >> 5,
482 'redeye' => ( $val & 0b01000000 ) >> 6,
483 / 'reserved' => ( $val & 0b10000000 ) >> 7,
484 ];
485 $flashMsgs = [];
486 # We do not need to handle unknown values since all are used.
487 foreach ( $flashDecode as $subTag => $subValue ) {
488 # We do not need any message for zeroed values.
489 if ( $subTag !== 'fired' && $subValue === 0 ) {
490 continue;
491 }
492 $fullTag = $tag . '-' . $subTag;
493 $flashMsgs[] = $this->exifMsg( $fullTag, $subValue );
494 }
495 $val = $this->getLanguage()->commaList( $flashMsgs );
496 break;
497
498 case 'FocalPlaneResolutionUnit':
499 switch ( $val ) {
500 case 2:
501 $val = $this->exifMsg( $tag, $val );
502 break;
503 default:
504 /* If not recognized, display as is. */
505 $val = $this->literal( $val );
506 break;
507 }
508 break;
509
510 case 'SensingMethod':
511 switch ( $val ) {
512 case 1:
513 case 2:
514 case 3:
515 case 4:
516 case 5:
517 case 7:
518 case 8:
519 $val = $this->exifMsg( $tag, $val );
520 break;
521 default:
522 /* If not recognized, display as is. */
523 $val = $this->literal( $val );
524 break;
525 }
526 break;
527
528 case 'FileSource':
529 switch ( $val ) {
530 case 3:
531 $val = $this->exifMsg( $tag, $val );
532 break;
533 default:
534 /* If not recognized, display as is. */
535 $val = $this->literal( $val );
536 break;
537 }
538 break;
539
540 case 'SceneType':
541 switch ( $val ) {
542 case 1:
543 $val = $this->exifMsg( $tag, $val );
544 break;
545 default:
546 /* If not recognized, display as is. */
547 $val = $this->literal( $val );
548 break;
549 }
550 break;
551
552 case 'CustomRendered':
553 switch ( $val ) {
554 case 0: /* normal */
555 case 1: /* custom */
556 /* The following are unofficial Apple additions */
557 case 2: /* HDR (no original saved) */
558 case 3: /* HDR (original saved) */
559 case 4: /* Original (for HDR) */
560 /* Yes 5 is not present ;) */
561 case 6: /* Panorama */
562 case 7: /* Portrait HDR */
563 case 8: /* Portrait */
564 $val = $this->exifMsg( $tag, $val );
565 break;
566 default:
567 /* If not recognized, display as is. */
568 $val = $this->literal( $val );
569 break;
570 }
571 break;
572
573 case 'ExposureMode':
574 switch ( $val ) {
575 case 0:
576 case 1:
577 case 2:
578 $val = $this->exifMsg( $tag, $val );
579 break;
580 default:
581 /* If not recognized, display as is. */
582 break;
583 }
584 break;
585
586 case 'WhiteBalance':
587 switch ( $val ) {
588 case 0:
589 case 1:
590 $val = $this->exifMsg( $tag, $val );
591 break;
592 default:
593 /* If not recognized, display as is. */
594 $val = $this->literal( $val );
595 break;
596 }
597 break;
598
599 case 'SceneCaptureType':
600 switch ( $val ) {
601 case 0:
602 case 1:
603 case 2:
604 case 3:
605 $val = $this->exifMsg( $tag, $val );
606 break;
607 default:
608 /* If not recognized, display as is. */
609 $val = $this->literal( $val );
610 break;
611 }
612 break;
613
614 case 'GainControl':
615 switch ( $val ) {
616 case 0:
617 case 1:
618 case 2:
619 case 3:
620 case 4:
621 $val = $this->exifMsg( $tag, $val );
622 break;
623 default:
624 /* If not recognized, display as is. */
625 $val = $this->literal( $val );
626 break;
627 }
628 break;
629
630 case 'Contrast':
631 switch ( $val ) {
632 case 0:
633 case 1:
634 case 2:
635 $val = $this->exifMsg( $tag, $val );
636 break;
637 default:
638 /* If not recognized, display as is. */
639 $val = $this->literal( $val );
640 break;
641 }
642 break;
643
644 case 'Saturation':
645 switch ( $val ) {
646 case 0:
647 case 1:
648 case 2:
649 $val = $this->exifMsg( $tag, $val );
650 break;
651 default:
652 /* If not recognized, display as is. */
653 $val = $this->literal( $val );
654 break;
655 }
656 break;
657
658 case 'Sharpness':
659 switch ( $val ) {
660 case 0:
661 case 1:
662 case 2:
663 $val = $this->exifMsg( $tag, $val );
664 break;
665 default:
666 /* If not recognized, display as is. */
667 $val = $this->literal( $val );
668 break;
669 }
670 break;
671
672 case 'SubjectDistanceRange':
673 switch ( $val ) {
674 case 0:
675 case 1:
676 case 2:
677 case 3:
678 $val = $this->exifMsg( $tag, $val );
679 break;
680 default:
681 /* If not recognized, display as is. */
682 $val = $this->literal( $val );
683 break;
684 }
685 break;
686
687 / The GPS...Ref values are kept for compatibility, probably won't be reached.
688 case 'GPSLatitudeRef':
689 case 'GPSDestLatitudeRef':
690 switch ( $val ) {
691 case 'N':
692 case 'S':
693 $val = $this->exifMsg( 'GPSLatitude', $val );
694 break;
695 default:
696 /* If not recognized, display as is. */
697 $val = $this->literal( $val );
698 break;
699 }
700 break;
701
702 case 'GPSLongitudeRef':
703 case 'GPSDestLongitudeRef':
704 switch ( $val ) {
705 case 'E':
706 case 'W':
707 $val = $this->exifMsg( 'GPSLongitude', $val );
708 break;
709 default:
710 /* If not recognized, display as is. */
711 $val = $this->literal( $val );
712 break;
713 }
714 break;
715
716 case 'GPSAltitude':
717 if ( $val < 0 ) {
718 $val = $this->exifMsg( 'GPSAltitude', 'below-sealevel', $this->formatNum( -$val, 3 ) );
719 } else {
720 $val = $this->exifMsg( 'GPSAltitude', 'above-sealevel', $this->formatNum( $val, 3 ) );
721 }
722 break;
723
724 case 'GPSStatus':
725 switch ( $val ) {
726 case 'A':
727 case 'V':
728 $val = $this->exifMsg( $tag, $val );
729 break;
730 default:
731 /* If not recognized, display as is. */
732 $val = $this->literal( $val );
733 break;
734 }
735 break;
736
737 case 'GPSMeasureMode':
738 switch ( $val ) {
739 case 2:
740 case 3:
741 $val = $this->exifMsg( $tag, $val );
742 break;
743 default:
744 /* If not recognized, display as is. */
745 $val = $this->literal( $val );
746 break;
747 }
748 break;
749
750 case 'GPSTrackRef':
751 case 'GPSImgDirectionRef':
752 case 'GPSDestBearingRef':
753 switch ( $val ) {
754 case 'T':
755 case 'M':
756 $val = $this->exifMsg( 'GPSDirection', $val );
757 break;
758 default:
759 /* If not recognized, display as is. */
760 $val = $this->literal( $val );
761 break;
762 }
763 break;
764
765 case 'GPSLatitude':
766 case 'GPSDestLatitude':
767 $val = $this->formatCoords( $val, 'latitude' );
768 break;
769 case 'GPSLongitude':
770 case 'GPSDestLongitude':
771 $val = $this->formatCoords( $val, 'longitude' );
772 break;
773
774 case 'GPSSpeedRef':
775 switch ( $val ) {
776 case 'K':
777 case 'M':
778 case 'N':
779 $val = $this->exifMsg( 'GPSSpeed', $val );
780 break;
781 default:
782 /* If not recognized, display as is. */
783 $val = $this->literal( $val );
784 break;
785 }
786 break;
787
788 case 'GPSDestDistanceRef':
789 switch ( $val ) {
790 case 'K':
791 case 'M':
792 case 'N':
793 $val = $this->exifMsg( 'GPSDestDistance', $val );
794 break;
795 default:
796 /* If not recognized, display as is. */
797 $val = $this->literal( $val );
798 break;
799 }
800 break;
801
802 case 'GPSDOP':
803 / See https://en.wikipedia.org/wiki/Dilution_of_precision_(GPS)
804 if ( $val <= 2 ) {
805 $val = $this->exifMsg( $tag, 'excellent', $this->formatNum( $val ) );
806 } elseif ( $val <= 5 ) {
807 $val = $this->exifMsg( $tag, 'good', $this->formatNum( $val ) );
808 } elseif ( $val <= 10 ) {
809 $val = $this->exifMsg( $tag, 'moderate', $this->formatNum( $val ) );
810 } elseif ( $val <= 20 ) {
811 $val = $this->exifMsg( $tag, 'fair', $this->formatNum( $val ) );
812 } else {
813 $val = $this->exifMsg( $tag, 'poor', $this->formatNum( $val ) );
814 }
815 break;
816
817 / This is not in the Exif standard, just a special
818 / case for our purposes which enables wikis to wikify
819 / the make, model and software name to link to their articles.
820 case 'Make':
821 case 'Model':
822 $val = $this->exifMsg( $tag, '', $this->literal( $val ) );
823 break;
824
825 case 'Software':
826 if ( is_array( $val ) ) {
827 if ( count( $val ) > 1 ) {
828 / if its a software, version array.
829 $val = $this->msg(
830 'exif-software-version-value',
831 $this->literal( $val[0] ),
832 $this->literal( $val[1] )
833 )->text();
834 } else {
835 / https://phabricator.wikimedia.org/T178130
836 $val = $this->exifMsg( $tag, '', $this->literal( $val[0] ) );
837 }
838 } else {
839 $val = $this->exifMsg( $tag, '', $this->literal( $val ) );
840 }
841 break;
842
843 case 'ExposureTime':
844 / Show the pretty fraction as well as decimal version
845 $val = $this->msg( 'exif-exposuretime-format',
846 $this->formatFraction( $val ), $this->formatNum( $val ) )->text();
847 break;
848 case 'ISOSpeedRatings':
849 / If it's 65535 that means it's at the
850 / limit of the size of Exif::short and
851 / is really higher.
852 if ( $val === '65535' ) {
853 $val = $this->exifMsg( $tag, 'overflow' );
854 } else {
855 $val = $this->formatNum( $val );
856 }
857 break;
858 case 'FNumber':
859 $val = $this->msg( 'exif-fnumber-format',
860 $this->formatNum( $val ) )->text();
861 break;
862
863 case 'FocalLength':
864 case 'FocalLengthIn35mmFilm':
865 $val = $this->msg( 'exif-focallength-format',
866 $this->formatNum( $val ) )->text();
867 break;
868
869 case 'MaxApertureValue':
870 if ( strpos( $val, '/' ) !== false ) {
871 / need to expand this earlier to calculate fNumber
872 [ $n, $d ] = explode( '/', $val, 2 );
873 if ( is_numeric( $n ) && is_numeric( $d ) ) {
874 $val = (int)$n / (int)$d;
875 }
876 }
877 if ( is_numeric( $val ) ) {
878 $fNumber = 2 ** ( $val / 2 );
879 if ( is_finite( $fNumber ) ) {
880 $val = $this->msg( 'exif-maxaperturevalue-value',
881 $this->formatNum( $val ),
882 $this->formatNum( $fNumber, 2 )
883 )->text();
884 break;
885 }
886 }
887 $val = $this->literal( $val );
888 break;
889
890 case 'iimCategory':
891 switch ( strtolower( $val ) ) {
892 / See pg 29 of IPTC photo
893 / metadata standard.
894 case 'ace':
895 case 'clj':
896 case 'dis':
897 case 'fin':
898 case 'edu':
899 case 'evn':
900 case 'hth':
901 case 'hum':
902 case 'lab':
903 case 'lif':
904 case 'pol':
905 case 'rel':
906 case 'sci':
907 case 'soi':
908 case 'spo':
909 case 'war':
910 case 'wea':
911 $val = $this->exifMsg(
912 'iimcategory',
913 $val
914 );
915 break;
916 default:
917 $val = $this->literal( $val );
918 }
919 break;
920 case 'SubjectNewsCode':
921 / Essentially like iimCategory.
922 / 8 (numeric) digit hierarchical
923 / classification. We decode the
924 / first 2 digits, which provide
925 / a broad category.
926 $val = $this->convertNewsCode( $val );
927 break;
928 case 'Urgency':
929 / 1-8 with 1 being highest, 5 normal
930 / 0 is reserved, and 9 is 'user-defined'.
931 $urgency = '';
932 if ( $val === 0 || $val === 9 ) {
933 $urgency = 'other';
934 } elseif ( $val < 5 && $val > 1 ) {
935 $urgency = 'high';
936 } elseif ( $val === 5 ) {
937 $urgency = 'normal';
938 } elseif ( $val <= 8 && $val > 5 ) {
939 $urgency = 'low';
940 }
941
942 if ( $urgency !== '' ) {
943 $val = $this->exifMsg( 'urgency',
944 $urgency, $this->literal( $val )
945 );
946 } else {
947 $val = $this->literal( $val );
948 }
949 break;
950
951 / Things that have a unit of pixels.
952 case 'OriginalImageHeight':
953 case 'OriginalImageWidth':
954 case 'PixelXDimension':
955 case 'PixelYDimension':
956 case 'ImageWidth':
957 case 'ImageLength':
958 $val = $this->formatNum( $val ) . ' ' . $this->msg( 'unit-pixel' )->text();
959 break;
960
961 / Do not transform fields with pure text.
962 / For some languages the formatNum()
963 / conversion results to wrong output like
964 / foo,bar@example,com or fooÙ«bar@exampleÙ«com.
965 / Also some 'numeric' things like Scene codes
966 / are included here as we really don't want
967 / commas inserted.
968 case 'ImageDescription':
969 case 'UserComment':
970 case 'Artist':
971 case 'Copyright':
972 case 'RelatedSoundFile':
973 case 'ImageUniqueID':
974 case 'SpectralSensitivity':
975 case 'GPSSatellites':
976 case 'GPSVersionID':
977 case 'GPSMapDatum':
978 case 'Keywords':
979 case 'WorldRegionDest':
980 case 'CountryDest':
981 case 'CountryCodeDest':
982 case 'ProvinceOrStateDest':
983 case 'CityDest':
984 case 'SublocationDest':
985 case 'WorldRegionCreated':
986 case 'CountryCreated':
987 case 'CountryCodeCreated':
988 case 'ProvinceOrStateCreated':
989 case 'CityCreated':
990 case 'SublocationCreated':
991 case 'ObjectName':
992 case 'SpecialInstructions':
993 case 'Headline':
994 case 'Credit':
995 case 'Source':
996 case 'EditStatus':
997 case 'FixtureIdentifier':
998 case 'LocationDest':
999 case 'LocationDestCode':
1000 case 'Writer':
1001 case 'JPEGFileComment':
1002 case 'iimSupplementalCategory':
1003 case 'OriginalTransmissionRef':
1004 case 'Identifier':
1005 case 'dc-contributor':
1006 case 'dc-coverage':
1007 case 'dc-publisher':
1008 case 'dc-relation':
1009 case 'dc-rights':
1010 case 'dc-source':
1011 case 'dc-type':
1012 case 'Lens':
1013 case 'SerialNumber':
1014 case 'CameraOwnerName':
1015 case 'Label':
1016 case 'Nickname':
1017 case 'RightsCertificate':
1018 case 'CopyrightOwner':
1019 case 'UsageTerms':
1020 case 'WebStatement':
1021 case 'OriginalDocumentID':
1022 case 'LicenseUrl':
1023 case 'MorePermissionsUrl':
1024 case 'AttributionUrl':
1025 case 'PreferredAttributionName':
1026 case 'PNGFileComment':
1027 case 'Disclaimer':
1028 case 'ContentWarning':
1029 case 'GIFFileComment':
1030 case 'SceneCode':
1031 case 'IntellectualGenre':
1032 case 'Event':
1033 case 'OrganisationInImage':
1034 case 'PersonInImage':
1035 case 'CaptureSoftware':
1036 case 'GPSAreaInformation':
1037 case 'GPSProcessingMethod':
1038 case 'StitchingSoftware':
1039 case 'SubSecTime':
1040 case 'SubSecTimeOriginal':
1041 case 'SubSecTimeDigitized':
1042 $val = $this->literal( $val );
1043 break;
1044
1045 case 'ProjectionType':
1046 switch ( $val ) {
1047 case 'equirectangular':
1048 $val = $this->exifMsg( $tag, $val );
1049 break;
1050 default:
1051 $val = $this->literal( $val );
1052 break;
1053 }
1054 break;
1055 case 'ObjectCycle':
1056 switch ( $val ) {
1057 case 'a':
1058 case 'p':
1059 case 'b':
1060 $val = $this->exifMsg( $tag, $val );
1061 break;
1062 default:
1063 $val = $this->literal( $val );
1064 break;
1065 }
1066 break;
1067 case 'Copyrighted':
1068 case 'UsePanoramaViewer':
1069 case 'ExposureLockUsed':
1070 switch ( $val ) {
1071 case 'True':
1072 case 'False':
1073 $val = $this->exifMsg( $tag, $val );
1074 break;
1075 default:
1076 $val = $this->literal( $val );
1077 break;
1078 }
1079 break;
1080 case 'Rating':
1081 if ( $val === '-1' ) {
1082 $val = $this->exifMsg( $tag, 'rejected' );
1083 } else {
1084 $val = $this->formatNum( $val );
1085 }
1086 break;
1087
1088 case 'LanguageCode':
1089 $lang = MediaWikiServices::getInstance()
1090 ->getLanguageNameUtils()
1091 ->getLanguageName( strtolower( $val ), $this->getLanguage()->getCode() );
1092 $val = $this->literal( $lang ?: $val );
1093 break;
1094
1095 default:
1096 $val = $this->formatNum( $val, false, $tag );
1097 break;
1098 }
1099 }
1100 / End formatting values, start flattening arrays.
1101 $vals = $this->flattenArrayReal( $vals, $type );
1102 }
1103
1104 return $tags;
1105 }
1106
1124 public function flattenArrayReal( $vals, $type = 'ul', $noHtml = false ) {
1125 if ( !is_array( $vals ) ) {
1126 return $vals; / do nothing if not an array;
1127 }
1128
1129 if ( isset( $vals['_type'] ) ) {
1130 $type = $vals['_type'];
1131 unset( $vals['_type'] );
1132 }
1133
1134 if ( count( $vals ) === 1 && $type !== 'lang' && isset( $vals[0] ) ) {
1135 return $vals[0];
1136 }
1137 if ( count( $vals ) === 0 ) {
1138 wfDebug( __METHOD__ . " metadata array with 0 elements!" );
1139
1140 return ""; / paranoia. This should never happen
1141 }
1142 / Check if $vals contains nested arrays
1143 $containsNestedArrays = in_array( true, array_map( 'is_array', $vals ), true );
1144 if ( $containsNestedArrays ) {
1145 wfLogWarning( __METHOD__ . ': Invalid $vals, contains nested arrays: ' . json_encode( $vals ) );
1146 }
1147
1148 /* @todo FIXME: This should hide some of the list entries if there are
1149 * say more than four. Especially if a field is translated into 20
1150 * languages, we don't want to show them all by default
1151 */
1152 switch ( $type ) {
1153 case 'lang':
1154 / Display default, followed by ContentLanguage,
1155 / followed by the rest in no particular order.
1156
1157 / Todo: hide some items if really long list.
1158
1159 $content = '';
1160
1161 $priorityLanguages = $this->getPriorityLanguages();
1162 $defaultItem = false;
1163 $defaultLang = false;
1164
1165 / If default is set, save it for later,
1166 / as we don't know if it's equal to one of the lang codes.
1167 / (In xmp you specify the language for a default property by having
1168 / both a default prop, and one in the language that are identical)
1169 if ( isset( $vals['x-default'] ) ) {
1170 $defaultItem = $vals['x-default'];
1171 unset( $vals['x-default'] );
1172 }
1173 foreach ( $priorityLanguages as $pLang ) {
1174 if ( isset( $vals[$pLang] ) ) {
1175 $isDefault = false;
1176 if ( $vals[$pLang] === $defaultItem ) {
1177 $defaultItem = false;
1178 $isDefault = true;
1179 }
1180 $content .= $this->langItem( $vals[$pLang], $pLang, $isDefault, $noHtml );
1181
1182 unset( $vals[$pLang] );
1183
1184 if ( $this->singleLang ) {
1185 return Html::rawElement( 'span', [ 'lang' => $pLang ], $vals[$pLang] );
1186 }
1187 }
1188 }
1189
1190 / Now do the rest.
1191 foreach ( $vals as $lang => $item ) {
1192 if ( $item === $defaultItem ) {
1193 $defaultLang = $lang;
1194 continue;
1195 }
1196 $content .= $this->langItem( $item, $lang, false, $noHtml );
1197 if ( $this->singleLang ) {
1198 return Html::rawElement( 'span', [ 'lang' => $lang ], $item );
1199 }
1200 }
1201 if ( $defaultItem !== false ) {
1202 $content = $this->langItem( $defaultItem, $defaultLang, true, $noHtml ) . $content;
1203 if ( $this->singleLang ) {
1204 return $defaultItem;
1205 }
1206 }
1207 if ( $noHtml ) {
1208 return $content;
1209 }
1210
1211 return '<ul class="metadata-langlist">' . $content . '</ul>';
1212 case 'ol':
1213 if ( $noHtml ) {
1214 return "\n#" . implode( "\n#", $vals );
1215 }
1216
1217 return "<ol><li>" . implode( "</li>\n<li>", $vals ) . '</li></ol>';
1218 case 'ul':
1219 default:
1220 if ( $noHtml ) {
1221 return "\n*" . implode( "\n*", $vals );
1222 }
1223
1224 return "<ul><li>" . implode( "</li>\n<li>", $vals ) . '</li></ul>';
1225 }
1226 }
1227
1237 private function langItem( $value, $lang, $default = false, $noHtml = false ) {
1238 if ( $lang === false && $default === false ) {
1239 throw new InvalidArgumentException( '$lang and $default cannot both be false.' );
1240 }
1241
1242 if ( $noHtml ) {
1243 $wrappedValue = $this->literal( $value );
1244 } else {
1245 $wrappedValue = '<span class="mw-metadata-lang-value">' . $this->literal( $value ) . '</span>';
1246 }
1247
1248 if ( $lang === false ) {
1249 $msg = $this->msg( 'metadata-langitem-default', $wrappedValue );
1250 if ( $noHtml ) {
1251 return $msg->text() . "\n\n";
1252 } /* else */
1253
1254 return '<li class="mw-metadata-lang-default">' . $msg->text() . "</li>\n";
1255 }
1256
1257 $lowLang = strtolower( $lang );
1258 $languageNameUtils = MediaWikiServices::getInstance()->getLanguageNameUtils();
1259 $langName = $languageNameUtils->getLanguageName( $lowLang );
1260 if ( $langName === '' ) {
1261 / try just the base language name. (aka en-US -> en ).
1262 $langPrefix = explode( '-', $lowLang, 2 )[0];
1263 $langName = $languageNameUtils->getLanguageName( $langPrefix );
1264 if ( $langName === '' ) {
1265 / give up.
1266 $langName = $lang;
1267 }
1268 }
1269 / else we have a language specified
1270
1271 $msg = $this->msg( 'metadata-langitem', $wrappedValue, $langName, $lang );
1272 if ( $noHtml ) {
1273 return '*' . $msg->text();
1274 } /* else: */
1275
1276 $item = '<li class="mw-metadata-lang-code-' . $lang;
1277 if ( $default ) {
1278 $item .= ' mw-metadata-lang-default';
1279 }
1280 $item .= '" lang="' . $lang . '">';
1281 $item .= $msg->text();
1282 $item .= "</li>\n";
1283
1284 return $item;
1285 }
1286
1294 protected function literal( $val ): string {
1295 if ( $val === null ) {
1296 return '';
1297 }
1298 / T266707: historically this has used htmlspecialchars to protect
1299 / the string contents, but it should probably be changed to use
1300 / wfEscapeWikitext() instead -- however, "we still want to auto-link
1301 / urls" so wfEscapeWikitext isn't *quite* right...
1302 return htmlspecialchars( $val );
1303 }
1304
1314 private function exifMsg( $tag, $val, $arg = null, $arg2 = null ) {
1315 if ( $val === '' ) {
1316 $val = 'value';
1317 }
1318
1319 return $this->msg(
1320 MediaWikiServices::getInstance()->getContentLanguage()->lc( "exif-$tag-$val" ),
1321 $arg,
1322 $arg2
1323 )->text();
1324 }
1325
1335 private function formatNum( $num, $round = false, $tagName = null ) {
1336 $m = [];
1337 if ( is_array( $num ) ) {
1338 $out = [];
1339 foreach ( $num as $number ) {
1340 $out[] = $this->formatNum( $number, $round, $tagName );
1341 }
1342
1343 return $this->getLanguage()->commaList( $out );
1344 }
1345 if ( is_numeric( $num ) ) {
1346 if ( $round !== false ) {
1347 $num = round( $num, $round );
1348 }
1349 return $this->getLanguage()->formatNum( $num );
1350 }
1351 $num ??= '';
1352 if ( preg_match( '/^(-?\d+)\/(\d+)$/', $num, $m ) ) {
1353 if ( $m[2] !== 0 ) {
1354 $newNum = (int)$m[1] / (int)$m[2];
1355 if ( $round !== false ) {
1356 $newNum = round( $newNum, $round );
1357 }
1358 } else {
1359 $newNum = $num;
1360 }
1361
1362 return $this->getLanguage()->formatNum( $newNum );
1363 }
1364 # T267370: there are a lot of strange EXIF tags floating around.
1365 LoggerFactory::getInstance( 'formatnum' )->warning(
1366 'FormatMetadata::formatNum with non-numeric value',
1367 [
1368 'tag' => $tagName,
1369 'value' => $num,
1370 ]
1371 );
1372 return $this->literal( $num );
1373 }
1374
1381 private function formatFraction( $num ) {
1382 $m = [];
1383 if ( preg_match( '/^(-?\d+)\/(\d+)$/', $num, $m ) ) {
1384 $numerator = (int)$m[1];
1385 $denominator = (int)$m[2];
1386 $gcd = $this->gcd( abs( $numerator ), $denominator );
1387 if ( $gcd !== 0 ) {
1388 / 0 shouldn't happen! ;)
1389 return $this->formatNum( $numerator / $gcd ) . '/' . $this->formatNum( $denominator / $gcd );
1390 }
1391 }
1392
1393 return $this->formatNum( $num );
1394 }
1395
1403 private function gcd( $a, $b ) {
1404 /*
1405 / https://en.wikipedia.org/wiki/Euclidean_algorithm
1406 / Recursive form would be:
1407 if ( $b == 0 )
1408 return $a;
1409 else
1410 return gcd( $b, $a % $b );
1411 */
1412 while ( $b != 0 ) {
1413 $remainder = $a % $b;
1414
1415 / tail recursion...
1416 $a = $b;
1417 $b = $remainder;
1418 }
1419
1420 return $a;
1421 }
1422
1435 private function convertNewsCode( $val ) {
1436 if ( !preg_match( '/^\d{8}$/D', $val ) ) {
1437 / Not a valid news code.
1438 return $val;
1439 }
1440 $cat = '';
1441 switch ( substr( $val, 0, 2 ) ) {
1442 case '01':
1443 $cat = 'ace';
1444 break;
1445 case '02':
1446 $cat = 'clj';
1447 break;
1448 case '03':
1449 $cat = 'dis';
1450 break;
1451 case '04':
1452 $cat = 'fin';
1453 break;
1454 case '05':
1455 $cat = 'edu';
1456 break;
1457 case '06':
1458 $cat = 'evn';
1459 break;
1460 case '07':
1461 $cat = 'hth';
1462 break;
1463 case '08':
1464 $cat = 'hum';
1465 break;
1466 case '09':
1467 $cat = 'lab';
1468 break;
1469 case '10':
1470 $cat = 'lif';
1471 break;
1472 case '11':
1473 $cat = 'pol';
1474 break;
1475 case '12':
1476 $cat = 'rel';
1477 break;
1478 case '13':
1479 $cat = 'sci';
1480 break;
1481 case '14':
1482 $cat = 'soi';
1483 break;
1484 case '15':
1485 $cat = 'spo';
1486 break;
1487 case '16':
1488 $cat = 'war';
1489 break;
1490 case '17':
1491 $cat = 'wea';
1492 break;
1493 }
1494 if ( $cat !== '' ) {
1495 $catMsg = $this->exifMsg( 'iimcategory', $cat );
1496 $val = $this->exifMsg( 'subjectnewscode', '', $this->literal( $val ), $catMsg );
1497 }
1498
1499 return $val;
1500 }
1501
1510 private function formatCoords( $coord, string $type ) {
1511 if ( !is_numeric( $coord ) ) {
1512 wfDebugLog( 'exif', __METHOD__ . ": \"$coord\" is not a number" );
1513 return $this->literal( (string)$coord );
1514 }
1515
1516 $ref = '';
1517 if ( $coord < 0 ) {
1518 $nCoord = -$coord;
1519 if ( $type === 'latitude' ) {
1520 $ref = 'S';
1521 } elseif ( $type === 'longitude' ) {
1522 $ref = 'W';
1523 }
1524 } else {
1525 $nCoord = (float)$coord;
1526 if ( $type === 'latitude' ) {
1527 $ref = 'N';
1528 } elseif ( $type === 'longitude' ) {
1529 $ref = 'E';
1530 }
1531 }
1532
1533 $deg = floor( $nCoord );
1534 $min = floor( ( $nCoord - $deg ) * 60 );
1535 $sec = round( ( ( $nCoord - $deg ) * 60 - $min ) * 60, 2 );
1536
1537 $deg = $this->formatNum( $deg );
1538 $min = $this->formatNum( $min );
1539 $sec = $this->formatNum( $sec );
1540
1541 / Note the default message "$1° $2′ $3″ $4" ignores the 5th parameter
1542 return $this->msg( 'exif-coordinate-format', $deg, $min, $sec, $ref, $this->literal( $coord ) )->text();
1543 }
1544
1559 public function collapseContactInfo( array $vals ) {
1560 if ( !( isset( $vals['CiAdrExtadr'] )
1561 || isset( $vals['CiAdrCity'] )
1562 || isset( $vals['CiAdrCtry'] )
1563 || isset( $vals['CiEmailWork'] )
1564 || isset( $vals['CiTelWork'] )
1565 || isset( $vals['CiAdrPcode'] )
1566 || isset( $vals['CiAdrRegion'] )
1567 || isset( $vals['CiUrlWork'] )
1568 ) ) {
1569 / We don't have any sub-properties
1570 / This could happen if its using old
1571 / iptc that just had this as a free-form
1572 / text value.
1573 / Note: people often insert >, etc into
1574 / the metadata which should not be interpreted
1575 / but we still want to auto-link urls.
1576 foreach ( $vals as &$val ) {
1577 $val = $this->literal( $val );
1578 }
1579
1580 return $this->flattenArrayReal( $vals );
1581 }
1582
1583 / We have a real ContactInfo field.
1584 / Its unclear if all these fields have to be
1585 / set, so assume they do not.
1586 $url = $tel = $street = $city = $country = '';
1587 $email = $postal = $region = '';
1588
1589 / Also note, some of the class names this uses
1590 / are similar to those used by hCard. This is
1591 / mostly because they're sensible names. This
1592 / does not (and does not attempt to) output
1593 / stuff in the hCard microformat. However it
1594 / might output in the adr microformat.
1595
1596 if ( isset( $vals['CiAdrExtadr'] ) ) {
1597 / Todo: This can potentially be multi-line.
1598 / Need to check how that works in XMP.
1599 $street = '<span class="extended-address">'
1600 . $this->literal(
1601 $vals['CiAdrExtadr'] )
1602 . '</span>';
1603 }
1604 if ( isset( $vals['CiAdrCity'] ) ) {
1605 $city = '<span class="locality">'
1606 . $this->literal( $vals['CiAdrCity'] )
1607 . '</span>';
1608 }
1609 if ( isset( $vals['CiAdrCtry'] ) ) {
1610 $country = '<span class="country-name">'
1611 . $this->literal( $vals['CiAdrCtry'] )
1612 . '</span>';
1613 }
1614 if ( isset( $vals['CiEmailWork'] ) ) {
1615 $emails = [];
1616 / Have to split multiple emails at commas/new lines.
1617 $splitEmails = explode( "\n", $vals['CiEmailWork'] );
1618 foreach ( $splitEmails as $e1 ) {
1619 / Also split on comma
1620 foreach ( explode( ',', $e1 ) as $e2 ) {
1621 $finalEmail = trim( $e2 );
1622 if ( $finalEmail === ',' || $finalEmail === '' ) {
1623 continue;
1624 }
1625 if ( strpos( $finalEmail, '<' ) !== false ) {
1626 / Don't do fancy formatting to
1627 / "My name" <[email protected]> style stuff
1628 $emails[] = $this->literal( $finalEmail );
1629 } else {
1630 $emails[] = '[mailto:'
1631 . $finalEmail
1632 . ' <span class="email">'
1633 . $this->literal( $finalEmail )
1634 . '</span>]';
1635 }
1636 }
1637 }
1638 $email = implode( ', ', $emails );
1639 }
1640 if ( isset( $vals['CiTelWork'] ) ) {
1641 $tel = '<span class="tel">'
1642 . $this->literal( $vals['CiTelWork'] )
1643 . '</span>';
1644 }
1645 if ( isset( $vals['CiAdrPcode'] ) ) {
1646 $postal = '<span class="postal-code">'
1647 . $this->literal( $vals['CiAdrPcode'] )
1648 . '</span>';
1649 }
1650 if ( isset( $vals['CiAdrRegion'] ) ) {
1651 / Note this is province/state.
1652 $region = '<span class="region">'
1653 . $this->literal( $vals['CiAdrRegion'] )
1654 . '</span>';
1655 }
1656 if ( isset( $vals['CiUrlWork'] ) ) {
1657 $url = '<span class="url">'
1658 . $this->literal( $vals['CiUrlWork'] )
1659 . '</span>';
1660 }
1661
1662 return $this->msg( 'exif-contact-value', $email, $url,
1663 $street, $city, $region, $postal, $country, $tel )->text();
1664 }
1665
1672 public static function getVisibleFields() {
1673 $fields = [];
1674 $lines = explode( "\n", wfMessage( 'metadata-fields' )->inContentLanguage()->text() );
1675 foreach ( $lines as $line ) {
1676 $matches = [];
1677 if ( preg_match( '/^\\*\s*(.*?)\s*$/', $line, $matches ) ) {
1678 $fields[] = $matches[1];
1679 }
1680 }
1681 $fields = array_map( 'strtolower', $fields );
1682
1683 return $fields;
1684 }
1685
1693 public function fetchExtendedMetadata( File $file ) {
1694 $cache = MediaWikiServices::getInstance()->getMainWANObjectCache();
1695
1696 / If revision deleted, exit immediately
1697 if ( $file->isDeleted( File::DELETED_FILE ) ) {
1698 return [];
1699 }
1700
1701 $cacheKey = $cache->makeKey(
1702 'getExtendedMetadata',
1703 $this->getLanguage()->getCode(),
1704 (int)$this->singleLang,
1705 $file->getSha1()
1706 );
1707
1708 $cachedValue = $cache->get( $cacheKey );
1709 if (
1710 $cachedValue
1711 && $this->getHookRunner()->onValidateExtendedMetadataCache( $cachedValue['timestamp'], $file )
1712 ) {
1713 $extendedMetadata = $cachedValue['data'];
1714 } else {
1715 $maxCacheTime = ( $file instanceof ForeignAPIFile ) ? 60 * 60 * 12 : 60 * 60 * 24 * 30;
1716 $fileMetadata = $this->getExtendedMetadataFromFile( $file );
1717 $extendedMetadata = $this->getExtendedMetadataFromHook( $file, $fileMetadata, $maxCacheTime );
1718 if ( $this->singleLang ) {
1719 $this->resolveMultilangMetadata( $extendedMetadata );
1720 }
1721 $this->discardMultipleValues( $extendedMetadata );
1722 / Make sure the metadata won't break the API when an XML format is used.
1723 / This is an API-specific function so it would be cleaner to call it from
1724 / outside fetchExtendedMetadata, but this way we don't need to redo the
1725 / computation on a cache hit.
1726 $this->sanitizeArrayForAPI( $extendedMetadata );
1727 $valueToCache = [ 'data' => $extendedMetadata, 'timestamp' => wfTimestampNow() ];
1728 $cache->set( $cacheKey, $valueToCache, $maxCacheTime );
1729 }
1730
1731 return $extendedMetadata;
1732 }
1733
1743 protected function getExtendedMetadataFromFile( File $file ) {
1744 / If this is a remote file accessed via an API request, we already
1745 / have remote metadata so we just ignore any local one
1746 if ( $file instanceof ForeignAPIFile ) {
1747 / In case of error we pretend no metadata - this will get cached.
1748 / Might or might not be a good idea.
1749 return $file->getExtendedMetadata() ?: [];
1750 }
1751
1752 $uploadDate = wfTimestamp( TS_ISO_8601, $file->getTimestamp() );
1753
1754 $fileMetadata = [
1755 / This is modification time, which is close to "upload" time.
1756 'DateTime' => [
1757 'value' => $uploadDate,
1758 'source' => 'mediawiki-metadata',
1759 ],
1760 ];
1761
1762 $title = $file->getTitle();
1763 if ( $title ) {
1764 $text = $title->getText();
1765 $pos = strrpos( $text, '.' );
1766
1767 if ( $pos ) {
1768 $name = substr( $text, 0, $pos );
1769 } else {
1770 $name = $text;
1771 }
1772
1773 $fileMetadata['ObjectName'] = [
1774 'value' => $name,
1775 'source' => 'mediawiki-metadata',
1776 ];
1777 }
1778
1779 return $fileMetadata;
1780 }
1781
1792 protected function getExtendedMetadataFromHook( File $file, array $extendedMetadata,
1793 &$maxCacheTime
1794 ) {
1795 $this->getHookRunner()->onGetExtendedMetadata(
1796 $extendedMetadata,
1797 $file,
1798 $this->getContext(),
1799 $this->singleLang,
1800 $maxCacheTime
1801 );
1802
1803 $visible = array_fill_keys( self::getVisibleFields(), true );
1804 foreach ( $extendedMetadata as $key => $value ) {
1805 if ( !isset( $visible[strtolower( $key )] ) ) {
1806 $extendedMetadata[$key]['hidden'] = '';
1807 }
1808 }
1809
1810 return $extendedMetadata;
1811 }
1812
1821 protected function resolveMultilangValue( $value ) {
1822 if (
1823 !is_array( $value )
1824 || !isset( $value['_type'] )
1825 || $value['_type'] !== 'lang'
1826 ) {
1827 return $value; / do nothing if not a multilang array
1828 }
1829
1830 / choose the language best matching user or site settings
1831 $priorityLanguages = $this->getPriorityLanguages();
1832 foreach ( $priorityLanguages as $lang ) {
1833 if ( isset( $value[$lang] ) ) {
1834 return $value[$lang];
1835 }
1836 }
1837
1838 / otherwise go with the default language, if set
1839 if ( isset( $value['x-default'] ) ) {
1840 return $value['x-default'];
1841 }
1842
1843 / otherwise just return any one language
1844 unset( $value['_type'] );
1845 if ( $value ) {
1846 return reset( $value );
1847 }
1848
1849 / this should not happen; signal error
1850 return null;
1851 }
1852
1862 protected function resolveMultivalueValue( $value ) {
1863 if ( !is_array( $value ) ) {
1864 return $value;
1865 }
1866 if ( isset( $value['_type'] ) && $value['_type'] === 'lang' ) {
1867 / if this is a multilang array, process fields separately
1868 $newValue = [];
1869 foreach ( $value as $k => $v ) {
1870 $newValue[$k] = $this->resolveMultivalueValue( $v );
1871 }
1872 return $newValue;
1873 }
1874 / _type is 'ul' or 'ol' or missing in which case it defaults to 'ul'
1875 $v = reset( $value );
1876 if ( key( $value ) === '_type' ) {
1877 $v = next( $value );
1878 }
1879 return $v;
1880 }
1881
1888 protected function resolveMultilangMetadata( &$metadata ) {
1889 if ( !is_array( $metadata ) ) {
1890 return;
1891 }
1892 foreach ( $metadata as &$field ) {
1893 if ( isset( $field['value'] ) ) {
1894 $field['value'] = $this->resolveMultilangValue( $field['value'] );
1895 }
1896 }
1897 }
1898
1905 protected function discardMultipleValues( &$metadata ) {
1906 if ( !is_array( $metadata ) ) {
1907 return;
1908 }
1909 foreach ( $metadata as $key => &$field ) {
1910 if ( $key === 'Software' || $key === 'Contact' ) {
1911 / we skip some fields which have composite values. They are not particularly interesting
1912 / and you can get them via the metadata / commonmetadata APIs anyway.
1913 continue;
1914 }
1915 if ( isset( $field['value'] ) ) {
1916 $field['value'] = $this->resolveMultivalueValue( $field['value'] );
1917 }
1918 }
1919 }
1920
1925 protected function sanitizeArrayForAPI( &$arr ) {
1926 if ( !is_array( $arr ) ) {
1927 return;
1928 }
1929
1930 $counter = 1;
1931 foreach ( $arr as $key => &$value ) {
1932 $sanitizedKey = $this->sanitizeKeyForAPI( $key );
1933 if ( $sanitizedKey !== $key ) {
1934 if ( isset( $arr[$sanitizedKey] ) ) {
1935 / Make the sanitized keys hopefully unique.
1936 / To make it definitely unique would be too much effort, given that
1937 / sanitizing is only needed for misformatted metadata anyway, but
1938 / this at least covers the case when $arr is numeric.
1939 $sanitizedKey .= $counter;
1940 ++$counter;
1941 }
1942 $arr[$sanitizedKey] = $arr[$key];
1943 unset( $arr[$key] );
1944 }
1945 if ( is_array( $value ) ) {
1946 $this->sanitizeArrayForAPI( $value );
1947 }
1948 }
1949 unset( $value );
1950
1951 / Handle API metadata keys (particularly "_type")
1952 $keys = array_filter( array_keys( $arr ), [ ApiResult::class, 'isMetadataKey' ] );
1953 if ( $keys ) {
1954 ApiResult::setPreserveKeysList( $arr, $keys );
1955 }
1956 }
1957
1964 protected function sanitizeKeyForAPI( $key ) {
1965 / drop all characters which are not valid in an XML tag name
1966 / a bunch of non-ASCII letters would be valid but probably won't
1967 / be used so we take the easy way
1968 $key = preg_replace( '/[^a-zA-Z0-9_:.\-]/', '', $key );
1969 / drop characters which are invalid at the first position
1970 $key = preg_replace( '/^[\d\-.]+/', '', $key );
1971
1972 if ( $key === '' ) {
1973 $key = '_';
1974 / special case for an internal keyword
1975 } elseif ( $key === '_element' ) {
1976 $key = 'element';
1977 }
1978
1979 return $key;
1980 }
1981
1988 protected function getPriorityLanguages() {
1989 $priorityLanguages = MediaWikiServices::getInstance()
1990 ->getLanguageFallback()
1991 ->getAllIncludingSiteLanguage( $this->getLanguage()->getCode() );
1992 $priorityLanguages = array_merge(
1993 (array)$this->getLanguage()->getCode(),
1994 $priorityLanguages[0],
1995 $priorityLanguages[1]
1996 );
1997
1998 return $priorityLanguages;
1999 }
2000}
wfDebug( $text, $dest='all', array $context=[])
Sends a line to the debug log if enabled or, optionally, to a comment in output.
wfTimestampNow()
Convenience function; returns MediaWiki timestamp for the present time.
wfLogWarning( $msg, $callerOffset=1, $level=E_USER_WARNING)
Send a warning as a PHP error and the debug log.
wfDebugLog( $logGroup, $text, $dest='all', array $context=[])
Send a line to a supplementary debug log file, if configured, or main debug log if not.
wfTimestamp( $outputtype=TS_UNIX, $ts=0)
Get a timestamp string in one of various formats.
wfMessage( $key,... $params)
This is the function for getting translated interface messages.
if(!defined('MW_SETUP_CALLBACK'))
Definition WebStart.php:81
Format Image metadata values into a human readable form.
resolveMultilangValue( $value)
Turns an XMP-style multilang array into a single value.
getPriorityLanguages()
Returns a list of languages (first is best) to use when formatting multilang fields,...
flattenArrayReal( $vals, $type='ul', $noHtml=false)
A function to collapse multivalued tags into a single value.
literal( $val)
Convenience function for getFormattedData()
getExtendedMetadataFromFile(File $file)
Get file-based metadata in standardized format.
collapseContactInfo(array $vals)
Format the contact info field into a single value.
fetchExtendedMetadata(File $file)
Get an array of extended metadata.
setSingleLanguage( $val)
Trigger only outputting single language for multilanguage fields.
sanitizeArrayForAPI(&$arr)
Makes sure the given array is a valid API response fragment.
discardMultipleValues(&$metadata)
Takes an array returned by the getExtendedMetadata* functions, and turns all fields into single-value...
makeFormattedData( $tags)
Numbers given by Exif user agents are often magical, that is they should be replaced by a detailed ex...
resolveMultivalueValue( $value)
Turns an XMP-style multivalue array into a single value by dropping all but the first value.
static getVisibleFields()
Get a list of fields that are visible by default.
getExtendedMetadataFromHook(File $file, array $extendedMetadata, &$maxCacheTime)
Get additional metadata from hooks in standardized format.
resolveMultilangMetadata(&$metadata)
Takes an array returned by the getExtendedMetadata* functions, and resolves multi-language values in ...
static getFormattedData( $tags, $context=false)
Numbers given by Exif user agents are often magical, that is they should be replaced by a detailed ex...
sanitizeKeyForAPI( $key)
Turns a string into a valid API identifier.
bool $singleLang
Only output a single language for multi-language fields.
This class represents the result of the API operations.
Definition ApiResult.php:43
The simplest way of implementing IContextSource is to hold a RequestContext as a member variable and ...
msg( $key,... $params)
Get a Message object with context set Parameters are the same as wfMessage()
Implements some public methods and some protected utility functions which are required by multiple ch...
Definition File.php:93
getSha1()
Get the SHA-1 base 36 hash of the file.
Definition File.php:2419
isDeleted( $field)
Is this file a "deleted" file in a private archive? STUB.
Definition File.php:2161
getTitle()
Return the associated title object.
Definition File.php:391
getTimestamp()
Get the 14-character timestamp of the file upload.
Definition File.php:2395
Foreign file accessible through api.php requests.
This class is a collection of static functions that serve two purposes:
Definition Html.php:56
Create PSR-3 logger objects.
Service locator for MediaWiki core services.
Interface for objects which can provide a MediaWiki context on request.
if(!file_exists( $CREDITS)) $lines

Follow Lee on X/Twitter - Father, Husband, Serial builder creating AI, crypto, games & web tools. We are friends :) AI Will Come To Life!

Check out: eBank.nz (Art Generator) | Netwrck.com (AI Tools) | Text-Generator.io (AI API) | BitBank.nz (Crypto AI) | ReadingTime (Kids Reading) | RewordGame | BigMultiplayerChess | WebFiddle | How.nz | Helix AI Assistant