2 /***************************************************************************\
3 * SPIP, Systeme de publication pour l'internet *
5 * Copyright (c) 2001-2013 *
6 * Arnaud Martin, Antoine Pitrou, Philippe Riviere, Emmanuel Saint-James *
8 * Ce programme est un logiciel libre distribue sous licence GNU/GPL. *
9 * Pour plus de details voir le fichier COPYING.txt ou l'aide en ligne. *
10 \***************************************************************************/
12 if (!defined('_ECRIRE_INC_VERSION')) return;
14 // librairie de base du core
15 include_spip('inc/filtres_images_mini');
18 // Image typographique
19 // Fonctions pour l'arabe
21 // http://doc.spip.org/@rtl_mb_ord
22 function rtl_mb_ord($char){
24 if (($c = ord($char)) < 216) return $c;
25 else return 256 * rtl_mb_ord(substr($char, 0, -1)) +
ord(substr($char, -1));
27 /* return (strlen($char) < 2) ?
28 ord($char) : 256 * mb_ord(substr($char, 0, -1))
29 + ord(substr($char, -1));
35 // http://doc.spip.org/@rtl_reverse
36 function rtl_reverse($mot, $rtl_global) {
37 $rtl_prec = $rtl_global;
39 $ponctuations = array("/", "-", "«","»", "“", "”", ",", ".", " ", ":", ";", "(", ")", "،", "؟", "?", "!", " ");
40 foreach($ponctuations as $ponct) {
41 $ponctuation[$ponct] = true;
46 for ($i = 0; $i < spip_strlen($mot); $i++
) {
47 $lettre = spip_substr($mot, $i, 1);
49 $code = rtl_mb_ord($lettre);
50 # echo "<li>$lettre - $code";
52 if (($code >= 54928 && $code <= 56767) ||
($code >= 15707294 && $code <= 15711164)) {
57 if ($lettre == "٠" ||
$lettre == "١" ||
$lettre == "٢" ||
$lettre == "٣" ||
$lettre == "٤" ||
$lettre == "٥"
58 ||
$lettre == "٦" ||
$lettre == "٧" ||
$lettre == "٨" ||
$lettre == "٩") $rtl = false;
60 if ($ponctuation[$lettre]) {
61 # le truc mega casse-gueule de l'inversion unicode:
62 # traiter le sens de placement en fonction de la lettre precedente
63 # (et non automatiquement le rtl_global)
68 case "(": $lettre = ")"; break;
69 case ")": $lettre = "("; break;
70 case "«": $lettre = "»"; break;
71 case "»": $lettre = "«"; break;
72 case "“": $lettre = "”"; break;
73 case "”": $lettre = "“"; break;
79 if ($rtl) $res = $lettre.$res;
80 else $res = $res.$lettre;
90 // http://doc.spip.org/@rtl_visuel
91 function rtl_visuel($texte, $rtl_global) {
92 // hebreu + arabe: 54928 => 56767
93 // hebreu + presentation A: 15707294 => 15710140
94 // arabe presentation: 15708336 => 15711164
96 # echo hexdec("efb7bc");
98 // premiere passe pour determiner s'il y a du rtl
99 // de facon a placer ponctuation et mettre les mots dans l'ordre
102 $arabic_letters = array(
103 array("ي", // lettre 0
108 array("ب", // lettre 0
113 array("ا", // lettre 0
118 array("إ", // lettre 0
123 array("ل", // lettre 0
128 array("خ", // lettre 0
133 array("ج", // lettre 0
138 array("س", // lettre 0
143 array("ن", // lettre 0
148 array("ش", // lettre 0
153 array("ق", // lettre 0
158 array("ح", // lettre 0
163 array("م", // lettre 0
168 array("ر", // lettre 0
173 array("ع", // lettre 0
178 array("و", // lettre 0
183 array("ة", // lettre 0
188 array("ف", // lettre 0
193 array("ﻻ", // lettre 0
198 array("ح", // lettre 0
203 array("ت", // lettre 0
208 array("ض", // lettre 0
213 array("ك", // lettre 0
218 array("ه", // lettre 0
223 array("ي", // lettre 0
228 array("ئ", // lettre 0
233 array("ص", // lettre 0
238 array("ث", // lettre 0
243 array("ﻷ", // lettre 0
248 array("د", // lettre 0
253 array("ذ", // lettre 0
258 array("ط", // lettre 0
263 array("آ", // lettre 0
268 array("أ", // lettre 0
273 array("ؤ", // lettre 0
278 array("ز", // lettre 0
283 array("ظ", // lettre 0
288 array("غ", // lettre 0
293 array("ى", // lettre 0
298 array("پ", // lettre 0
303 array("چ", // lettre 0
310 if(init_mb_string() AND mb_regex_encoding() !== "UTF-8") echo "Attention: dans php.ini, il faut indiquer:<br /><strong>mbstring.internal_encoding = UTF-8</strong>";
313 $texte = explode(" ", $texte);
315 foreach ($texte as $mot) {
318 // Inserer des indicateurs de debut/fin
321 $mot = preg_replace(", ,u", " ", $mot);
322 $mot = preg_replace(",«,u", "«", $mot);
323 $mot = preg_replace(",»,u", "»", $mot);
326 $ponctuations = array("/", "-", "«","»", "“", "”", ",", ".", " ", ":", ";", "(", ")", "،", "؟", "?", "!"," ");
327 foreach($ponctuations as $ponct) {
328 $mot = str_replace("$ponct", "^$ponct^", $mot);
331 // lettres forcant coupure
332 $mot = preg_replace(",ا,u", "ا^", $mot);
333 $mot = preg_replace(",د,u", "د^", $mot);
334 $mot = preg_replace(",أ,u", "أ^", $mot);
335 $mot = preg_replace(",إ,u", "إ^", $mot);
336 $mot = preg_replace(",أ,u", "أ^", $mot);
337 $mot = preg_replace(",ر,u", "ر^", $mot);
338 $mot = preg_replace(",ذ,u", "ذ^", $mot);
339 $mot = preg_replace(",ز,u", "ز^", $mot);
340 $mot = preg_replace(",و,u", "و^", $mot);
341 $mot = preg_replace(",و,u", "و^", $mot);
342 $mot = preg_replace(",ؤ,u", "ؤ^", $mot);
343 $mot = preg_replace(",ة,u", "ة^", $mot);
344 // $mot = preg_replace(",ل,u", "^ل", $mot);
345 // $mot = preg_replace(",,", "^", $mot);
348 $mot = preg_replace(",٠,u", "^٠^", $mot);
349 $mot = preg_replace(",١,u", "^١^", $mot);
350 $mot = preg_replace(",٢,u", "^٢^", $mot);
351 $mot = preg_replace(",٣,u", "^٣^", $mot);
352 $mot = preg_replace(",٤,u", "^٤^", $mot);
353 $mot = preg_replace(",٥,u", "^٥^", $mot);
354 $mot = preg_replace(",٦,u", "^٦^", $mot);
355 $mot = preg_replace(",٧,u", "^٧^", $mot);
356 $mot = preg_replace(",٨,u", "^٨^", $mot);
357 $mot = preg_replace(",٩,u", "^٩^", $mot);
361 $mot = preg_replace(",لا,u", "ﻻ", $mot);
362 $mot = preg_replace(",لأ,u", "ﻷ", $mot);
365 foreach ($arabic_letters as $a_l) {
366 $mot = preg_replace(",([^\^])".$a_l[0]."([^\^]),u", "\\1".$a_l[3]."\\2", $mot);
367 $mot = preg_replace(",\^".$a_l[0]."([^\^]),u", "^".$a_l[2]."\\1", $mot);
368 $mot = preg_replace(",([^\^])".$a_l[0]."\^,u", "\\1".$a_l[4]."^", $mot);
369 // il semble qu'il ne soit pas necessaire de remplacer
371 // $mot = preg_replace(",\^".$a_l[0]."\^,u", "^".$a_l[1]."^", $mot);
374 $mot = preg_replace(",\^,u", "", $mot);
377 $res = rtl_reverse($mot, $rtl_global);
381 for ($i = 0; $i < spip_strlen($mot); $i++) {
382 $lettre = spip_substr($mot, $i, 1);
383 $code = rtl_mb_ord($lettre);
384 if (($code >= 54928 && $code <= 56767) || ($code >= 15708336 && $code <= 15711164)) $rtl = true;
389 if ($rtl_global) $retour = $res . " " . $retour;
390 else $retour = $retour. " ".$res;
398 // http://doc.spip.org/@printWordWrapped
399 function printWordWrapped($image, $top, $left, $maxWidth, $font, $couleur, $text, $textSize, $align="left", $hauteur_ligne = 0) {
400 static $memps = array();
402 // imageftbbox exige un float, et settype aime le double pour php < 4.2.0
403 settype($textSize, 'double');
405 // calculer les couleurs ici, car fonctionnement different selon TTF ou PS
406 $black = imagecolorallocatealpha($image, hexdec("0x{".substr($couleur, 0,2)."}"), hexdec("0x{".substr($couleur, 2,2)."}"), hexdec("0x{".substr($couleur, 4,2)."}"), 0);
407 $grey2 = imagecolorallocatealpha($image, hexdec("0x{".substr($couleur, 0,2)."}"), hexdec("0x{".substr($couleur, 2,2)."}"), hexdec("0x{".substr($couleur, 4,2)."}"), 127);
409 // Gaffe, T1Lib ne fonctionne carrement pas bien des qu'on sort de ASCII
410 // C'est dommage, parce que la rasterisation des caracteres est autrement plus jolie qu'avec TTF.
411 // A garder sous le coude en attendant que ca ne soit plus une grosse bouse.
412 // Si police Postscript et que fonction existe...
415 strtolower(substr($font,-4)) == ".pfb"
416 AND function_exists("imagepstext")) {
417 // Traitement specifique pour polices PostScript (experimental)
418 $textSizePs = round(1.32 * $textSize);
419 if (!$fontps = $memps["$font"]) {
420 $fontps = imagepsloadfont($font);
421 // Est-ce qu'il faut reencoder? Pas testable proprement, alors...
422 // imagepsencodefont($fontps,find_in_path('polices/standard.enc'));
423 $memps["$font"] = $fontps;
428 for ($i = 0; $i < spip_strlen($text); $i++
) {
429 $lettre = spip_substr($text, $i, 1);
430 $code = rtl_mb_ord($lettre);
431 if (($code >= 54928 && $code <= 56767) ||
($code >= 15707294 && $code <= 15711164)) {
437 // split the text into an array of single words
438 $words = explode(' ', $text);
441 foreach($words as $k=>$v)
442 $words[$k] = str_replace(array('~'), array(' '), $v);
445 if ($hauteur_ligne == 0) $lineHeight = floor($textSize * 1.3);
446 else $lineHeight = $hauteur_ligne;
448 $dimensions_espace = imageftbbox($textSize, 0, $font, ' ', array());
449 if ($dimensions_espace[2] < 0)
450 $dimensions_espace = imageftbbox($textSize, 0, $font, $line, array());
451 $largeur_espace = $dimensions_espace[2] - $dimensions_espace[0];
452 $retour["espace"] = $largeur_espace;
456 while (count($words) > 0) {
460 if ($rtl_global) $mot = rtl_visuel($mot,$rtl_global);
462 $dimensions = imageftbbox($textSize, 0, $font, $line.' '.$mot, array());
463 $lineWidth = $dimensions[2] - $dimensions[0]; // get the length of this line, if the word is to be included
464 if ($lineWidth > $maxWidth) { // if this makes the text wider that anticipated
465 $lines[] = $line; // add the line to the others
466 $line = ''; // empty it (the word will be added outside the loop)
468 $line .= ' '.$words[0]; // add the word to the current sentence
469 $words = array_slice($words, 1); // remove the word from the array
471 if ($line != '') { $lines[] = $line; } // add the last line to the others, if it isn't empty
472 $height = count($lines) * $lineHeight; // the height of all the lines total
473 // do the actual printing
476 // Deux passes pour recuperer, d'abord, largeur_ligne
477 // necessaire pour alignement right et center
478 foreach ($lines as $line) {
479 if ($rtl_global) $line = rtl_visuel($line, $rtl_global);
481 $dimensions = imageftbbox($textSize, 0, $font, $line, array());
482 $largeur_ligne = $dimensions[2] - $dimensions[0];
483 if ($largeur_ligne > $largeur_max) $largeur_max = $largeur_ligne;
486 foreach ($lines as $i => $line) {
487 if ($rtl_global) $line = rtl_visuel($line, $rtl_global);
489 $dimensions = imageftbbox($textSize, 0, $font, $line, array());
490 $largeur_ligne = $dimensions[2] - $dimensions[0];
491 if ($align == "right") $left_pos = $largeur_max - $largeur_ligne;
492 else if ($align == "center") $left_pos = floor(($largeur_max - $largeur_ligne)/2);
498 imagepstext ($image, "$line", $fontps, $textSizePs, $black, $grey2, $left +
$left_pos, $top +
$lineHeight * $i, 0, 0, 0, 16);
500 else imagefttext($image, $textSize, 0, $left +
$left_pos, $top +
$lineHeight * $i, $black, $font, trim($line), array());
502 $retour["height"] = $height;# + round(0.3 * $hauteur_ligne);
503 $retour["width"] = $largeur_max;
508 //array imagefttext ( resource image, float size, float angle, int x, int y, int col, string font_file, string text [, array extrainfo] )
509 //array imagettftext ( resource image, float size, float angle, int x, int y, int color, string fontfile, string text )
511 // http://doc.spip.org/@produire_image_typo
512 function produire_image_typo() {
516 $texte : le texte a transformer; attention: c'est toujours le premier argument, et c'est automatique dans les filtres
517 $couleur : la couleur du texte dans l'image - pas de dieze
518 $police: nom du fichier de la police (inclure terminaison)
519 $largeur: la largeur maximale de l'image ; attention, l'image retournee a une largeur inferieure, selon les limites reelles du texte
520 $hauteur_ligne: la hauteur de chaque ligne de texte si texte sur plusieurs lignes
521 (equivalent a "line-height")
522 $padding: forcer de l'espace autour du placement du texte; necessaire pour polices a la con qui "depassent" beaucoup de leur boite
523 $align: alignement left, right, center
528 // Recuperer les differents arguments
529 $numargs = func_num_args();
530 $arg_list = func_get_args();
531 $texte = $arg_list[0];
532 for ($i = 1; $i < $numargs; $i++
) {
533 if (($p = strpos($arg_list[$i], "="))!==FALSE) {
534 $nom_variable = substr($arg_list[$i], 0, $p);
535 $val_variable = substr($arg_list[$i], $p+
1);
537 $variable["$nom_variable"] = $val_variable;
542 // Construire requete et nom fichier
543 $text = str_replace(" ", "~", $texte);
544 $text = preg_replace(",(\r|\n)+,ms", " ", $text);
545 include_spip('inc/charsets');
546 $text = html2unicode(strip_tags($text));
547 if (strlen($text) == 0) return "";
549 $taille = $variable["taille"];
550 if ($taille < 1) $taille = 16;
552 $couleur = couleur_html_to_hex($variable["couleur"]);
553 if (strlen($couleur) < 6) $couleur = "000000";
557 $align = $variable["align"];
558 if (!$variable["align"]) $align="left";
560 $police = $variable["police"];
561 if (strlen($police) < 2) $police = "dustismo.ttf";
563 $largeur = $variable["largeur"];
564 if ($largeur < 5) $largeur = 600;
566 if ($variable["hauteur_ligne"] > 0) $hauteur_ligne = $variable["hauteur_ligne"];
567 else $hauteur_ligne = 0;
568 if ($variable["padding"] > 0) $padding = $variable["padding"];
573 $string = "$text-$taille-$couleur-$align-$police-$largeur-$hauteur_ligne-$padding";
574 $query = md5($string);
575 $dossier = sous_repertoire(_DIR_VAR
, 'cache-texte');
576 $fichier = "$dossier$query.png";
578 $flag_gd_typo = function_exists("imageftbbox")
579 && function_exists('imageCreateTrueColor');
582 if (@file_exists
($fichier))
584 else if (!$flag_gd_typo)
587 $font = find_in_path('polices/'.$police);
589 spip_log(_T('fichier_introuvable', array('fichier' => $police)));
590 $font = find_in_path('polices/'."dustismo.ttf");
593 $imgbidon = imageCreateTrueColor($largeur, 45);
594 $retour = printWordWrapped($imgbidon, $taille+
5, 0, $largeur, $font, $couleur, $text, $taille, 'left', $hauteur_ligne);
595 $hauteur = $retour["height"];
596 $largeur_reelle = $retour["width"];
597 $espace = $retour["espace"];
598 imagedestroy($imgbidon);
600 $im = imageCreateTrueColor($largeur_reelle-$espace+
(2*$padding), $hauteur+
5+
(2*$padding));
601 imagealphablending ($im, FALSE );
602 imagesavealpha ( $im, TRUE );
604 // Creation de quelques couleurs
606 $grey2 = imagecolorallocatealpha($im, hexdec("0x{".substr($couleur, 0,2)."}"), hexdec("0x{".substr($couleur, 2,2)."}"), hexdec("0x{".substr($couleur, 4,2)."}"), 127);
607 ImageFilledRectangle ($im,0,0,$largeur_reelle+
(2*$padding),$hauteur+
5+
(2*$padding),$grey2);
609 // Le texte a dessiner
610 printWordWrapped($im, $taille+
5+
$padding, $padding, $largeur, $font, $couleur, $text, $taille, $align, $hauteur_ligne);
613 // Utiliser imagepng() donnera un texte plus claire,
614 // compare a l'utilisation de la fonction imagejpeg()
615 _image_gd_output($im,array('fichier_dest'=>$fichier,'format_dest'=>'png'));
622 $dimensions = getimagesize($image);
623 $largeur = $dimensions[0];
624 $hauteur = $dimensions[1];
626 return inserer_attribut("<img src='$image' width='$largeur' height='$hauteur' style='width:".$largeur."px;height:".$hauteur."px;' />", 'alt', $alt);