FAQ

1. Quelle est exactement la licence de FPDF ? Y a-t-il des restrictions d'utilisation ?
2. Lorsque j'essaie de créer un PDF, plein de caractères bizarres s'affichent à l'écran.
3. J'essaie de générer un PDF et IE m'affiche une page blanche. Que se passe-t-il ?
4. J'envoie des paramètres en utilisant la méthode POST et les valeurs n'apparaissent pas dans le PDF.
5. Lorsque j'utilise une session PHP, IE n'affiche plus mon PDF mais propose de le télécharger.
6. Quand je suis en SSL, IE n'arrive pas à ouvrir le PDF.
7. Quand j'exécute un script j'obtiens le message "FPDF error: Don't alter the locale before including class file".
8. J'essaie de mettre un PNG et Acrobat me dit "Une erreur est survenue lors du traitement d'une page. Une erreur d'affichage est survenue".
9. J'essaie de mettre une image et Acrobat me dit "Une erreur est survenue lors du traitement d'une page. Type d'opérande incorrect".
10. Je voudrais mettre mon image en taille réelle dans mon PDF. Comment convertir les pixels en mm ?
11. J'ai l'erreur suivante quand j'essaie de générer un PDF : Warning: Cannot add header information - headers already sent by (output started at script.php:X)
12. J'essaie d'afficher une variable dans la méthode Header mais rien ne s'imprime.
13. J'ai défini les méthodes Header et Footer dans ma classe PDF mais rien ne s'affiche.
14. Je n'arrive pas à faire de retour à la ligne. J'ai bien mis \n dans la chaîne imprimée par MultiCell mais ça ne marche pas.
15. J'essaie de mettre le caractère euro mais je n'y arrive pas.
16. Je dessine un cadre avec des dimensions très précises, mais à l'impression je constate des écarts.
17. Je voudrais utiliser toute la surface de la page mais à l'impression j'ai toujours des marges. Comment les enlever ?
18. Quelle est la taille limite des fichiers que je peux générer avec FPDF ?
19. Est-ce que je peux modifier un PDF avec FPDF ?
20. Je voudrais faire un moteur de recherche en PHP et indexer des PDF. Est-ce que je peux le faire avec FPDF ?
21. Est-ce que je peux transformer une page HTML en PDF avec FPDF ?
22. Est-ce que je peux concaténer des PDF avec FPDF ?
23. Comment faire pour activer les protections sur un PDF ? Je voudrais empêcher qu'on puisse copier-coller le texte ou modifier le document.


1. Quelle est exactement la licence de FPDF ? Y a-t-il des restrictions d'utilisation ?

FPDF est Freeware (c'est indiqué au début du fichier source). Il n'y a pas de restriction d'usage. Vous pouvez l'incorporer librement dans votre application (commerciale ou non), avec ou sans modification.

2. Lorsque j'essaie de créer un PDF, plein de caractères bizarres s'affichent à l'écran.

Ces caractères "bizarres" sont en fait le contenu réel du PDF. Ce comportement est un bug d'IE. Lorsqu'il reçoit d'abord une page HTML, puis un PDF à partir de la même URL, il l'affiche directement sans lancer le plug-in Acrobat. Cela arrive fréquemment en cours de développement : à la moindre erreur de script, une page HTML est envoyée, et après correction, le PDF arrive.
Pour résoudre le problème, il suffit de fermer IE et de le relancer. On peut aussi aller sur une autre URL et revenir.
Pour éviter ce genre de désagrément durant le développement, on peut générer le PDF directement dans un fichier et l'ouvrir via l'explorateur.

3. J'essaie de générer un PDF et IE m'affiche une page blanche. Que se passe-t-il ?

Tout d'abord, vérifiez que vous n'envoyez rien au navigateur après le PDF (même pas un espace ou un retour-chariot). Vous pouvez mettre un exit juste après l'appel à la méthode Output() pour en être sûr.
Si ce n'est pas ça, c'est que vous êtes victime du syndrome de la "page blanche". IE utilisé en conjonction avec le plug-in Acrobat souffre de très nombreux bugs, quelles que soient les versions. Essayez de tester votre application avec le plus de versions d'IE possible (en tout cas si elle est sur Internet). Le problème survient surtout lorsqu'on utilise la méthode POST, c'est pourquoi il est fortement déconseillé de l'utiliser (d'autant qu'elle pose d'autres problèmes, voir la question suivante). Le GET marche mieux mais peut échouer lorsque l'URL devient trop longue : il ne faut pas dépasser 45 caractères pour la query string. Il existe cependant une astuce pour dépasser cette limite : terminer l'URL par .pdf, ce qui trompe IE. Si vous utilisez un formulaire, il suffit de rajouter un champ caché en dernière position :

<INPUT TYPE="HIDDEN" NAME="ext" VALUE=".pdf">

L'utilisation de session PHP cause également souvent des dysfonctionnements (il faut éviter d'envoyer des en-têtes HTTP empêchant la mise en cache). Voir la question 5 pour un moyen de résoudre le problème.

Pour éviter tous ces problèmes de manière fiable, il existe deux principales techniques :

- Désactiver le plug-in et utiliser Acrobat comme application externe. Pour cela, lancez Acrobat ; dans le menu Fichier, Préférences, Générales, désactivez l'option "Intégrer au navigateur Web" (pour Acrobat 5 : Edition, Préférences, Options, "Afficher dans le navigateur"). Puis, lorsque vous récupérez un PDF dans IE, ce dernier affiche la boîte "Ouvrir ce fichier" ou "Enregistrer ce fichier". Décochez la case "Toujours demander avant d'ouvrir ce type de fichier" et choisissez Ouvrir. Dorénavant les PDF s'ouvriront automatiquement dans une fenêtre Acrobat indépendante.
L'inconvénient de la méthode est qu'il faut toucher à la configuration du poste client, ce qu'on peut faire en intranet mais pas pour Internet.

- Utiliser une technique de redirection. Le principe consiste à générer le PDF dans un fichier temporaire sur le serveur et à rediriger le client dessus (en utilisant du JavaScript, pas l'en-tête HTTP Location qui pose aussi des problèmes). Par exemple, à la fin du script, on peut mettre :

//Détermination d'un nom de fichier temporaire dans le répertoire courant
$file=basename(tempnam(getcwd(),'tmp'));
//Sauvegarde du PDF dans le fichier
$pdf->Output($file);
//Redirection JavaScript
echo "<HTML><SCRIPT>document.location='getpdf.php?f=$file';</SCRIPT></HTML>";

Puis ceci dans getpdf.php :

<?php
$f=$HTTP_GET_VARS['f'];
//Contrôle du fichier (à ne pas oublier !)
if(substr($f,0,3)!='tmp' or strpos($f,'/') or strpos($f,'\\'))
    die("Nom de fichier incorrect");
if(!file_exists($f))
    die("Le fichier n'existe pas");
//Traitement de la requête spéciale IE au cas où
if($HTTP_SERVER_VARS['HTTP_USER_AGENT']=='contype')
{
    Header('Content-Type: application/pdf');
    exit;
}
//Envoi du PDF
Header('Content-Type: application/pdf');
Header('Content-Length: '.filesize($f));
readfile($f);
//Suppression du fichier
unlink($f);
exit;
?>

Cette méthode fonctionne dans la plupart des cas, mais pose encore des problèmes avec IE6. La méthode "ultime" consiste à rediriger directement sur le fichier temporaire. Ce dernier doit donc avoir l'extension .pdf :

//Détermination d'un nom de fichier temporaire dans le répertoire courant
$file=basename(tempnam(getcwd(),'tmp'));
rename($file,$file.'.pdf');
$file.='.pdf';
//Sauvegarde du PDF dans le fichier
$pdf->Output($file);
//Redirection JavaScript
echo "<HTML><SCRIPT>document.location='$file';</SCRIPT></HTML>";

Cette méthode transforme un PDF dynamique en PDF statique et évite ainsi tous les ennuis. Par contre, il faut prévoir une procédure de nettoyage pour effacer les fichiers temporaires. Par exemple :

function CleanFiles($dir)
{
    //Efface les fichiers temporaires
    $t=time();
    $h=opendir($dir);
    while($file=readdir($h))
    {
        if(substr($file,0,3)=='tmp' and substr($file,-4)=='.pdf')
        {
            $path=$dir.'/'.$file;
            if($t-filemtime($path)>3600)
                @unlink($path);
        }
    }
    closedir($h);
}

Cette fonction efface tous les fichiers de la forme tmp*.pdf dans le répertoire spécifié qui datent de plus d'une heure. Vous pouvez l'appeler où vous voulez, par exemple dans le script qui génère le PDF.

Remarque : il est nécessaire d'ouvrir une nouvelle fenêtre pour le PDF, car on ne peut plus revenir en arrière à cause de la redirection.

4. J'envoie des paramètres en utilisant la méthode POST et les valeurs n'apparaissent pas dans le PDF.

C'est un problème qui affecte certaines versions d'IE (en particulier la première 5.5). Voir la question précédente pour les moyens de le contourner.

5. Lorsque j'utilise une session PHP, IE n'affiche plus mon PDF mais propose de le télécharger.

C'est un problème qui affecte certaines versions d'IE. Pour le contourner, ajoutez la ligne suivante avant session_start() :

session_cache_limiter('private');

ou bien faites une redirection comme expliqué à la question 3.

6. Quand je suis en SSL, IE n'arrive pas à ouvrir le PDF.

Le problème peut être résolu en ajoutant cette ligne :

Header('Pragma: public');

7. Quand j'exécute un script j'obtiens le message "FPDF error: Don't alter the locale before including class file".

Lorsqu'on configure le séparateur décimal comme virgule avant d'inclure un fichier, il y a un bug dans PHP et les nombres décimaux sont tronqués. Il ne faut donc pas faire d'appel à setlocale() avant d'inclure la classe. Sous Unix, il ne faut pas non plus définir la variable d'environnement LC_ALL, car cela est équivalent à faire un appel à setlocale().

8. J'essaie de mettre un PNG et Acrobat me dit "Une erreur est survenue lors du traitement d'une page. Une erreur d'affichage est survenue".

Acrobat 5 a un bug et ne peut pas afficher les images transparentes monochromes (i.e. avec 1 bit par pixel). Enlevez la transparence ou passez votre image en 16 couleurs (4 bits par pixel) ou plus.

9. J'essaie de mettre une image et Acrobat me dit "Une erreur est survenue lors du traitement d'une page. Type d'opérande incorrect".

Il faut donner au moins une dimension à l'image ; la hauteur et la largeur ne peuvent être nulles en même temps.

10. Je voudrais mettre mon image en taille réelle dans mon PDF. Comment convertir les pixels en mm ?

Une image n'a pas de "taille réelle". La dimension qu'on lui donne dans le document est arbitraire. A moins que vous ne vouliez imposer une résolution particulière (par exemple 72 dpi, ce qui correspond à l'affichage sur un écran), auquel cas le rapport entre la taille en pixels et la résolution donne la longueur.

11. J'ai l'erreur suivante quand j'essaie de générer un PDF : Warning: Cannot add header information - headers already sent by (output started at script.php:X)

Il ne faut rien envoyer d'autre au navigateur que le PDF lui-même : pas d'HTML, pas d'espace, pas de retour-chariot, ni avant ni après. Le script envoie quelque chose à la ligne X.

12. J'essaie d'afficher une variable dans la méthode Header mais rien ne s'imprime.

Il faut utiliser le mot-clé global, par exemple :

function Header()
{
    global $titre;

    $this->SetFont('Arial','B',15);
    $this->Cell(0,10,$titre,1,1,'C');
}

13. J'ai défini les méthodes Header et Footer dans ma classe PDF mais rien ne s'affiche.

Il faut créer un objet de la classe PDF et non pas FPDF :

$pdf=new PDF();

14. Je n'arrive pas à faire de retour à la ligne. J'ai bien mis \n dans la chaîne imprimée par MultiCell mais ça ne marche pas.

Il faut mettre la chaîne entre guillemets et non pas entre apostrophes.

15. J'essaie de mettre le caractère euro mais je n'y arrive pas.

Pour les polices standard, le caractère euro a pour code 128. Vous pouvez par commodité définir une constante comme suit :

define('EURO',chr(128));

Note : Acrobat 4 ou supérieur est requis pour afficher l'euro.

16. Je dessine un cadre avec des dimensions très précises, mais à l'impression je constate des écarts.

Pour respecter les dimensions, il faut décocher la case "Ajuster" dans la boîte de dialogue d'impression.

17. Je voudrais utiliser toute la surface de la page mais à l'impression j'ai toujours des marges. Comment les enlever ?

Les imprimantes ont toutes des marges physiques (variables en fonction du modèle), il est donc impossible de les supprimer et d'imprimer sur la totalité de la page.

18. Quelle est la taille limite des fichiers que je peux générer avec FPDF ?

Il n'y a pas de limite particulière. Il existe cependant certaines contraintes :

- La taille mémoire allouée par défaut aux scripts PHP est de 8 Mo. Pour de très gros documents, en particulier avec des images, cette limite peut être atteinte (le fichier étant construit en mémoire). Elle est paramétrée dans php.ini.

- Le temps d'exécution alloué par défaut est de 30 secondes. Cette limite peut bien entendu être facilement dépassée. Elle est paramétrée dans php.ini et peut être éventuellement modifiée à l'exécution par set_time_limit().

- Les navigateurs ont généralement un time-out de 5 minutes. Si vous envoyez le PDF directement au navigateur et que vous dépassez cette limite, il sera perdu. Il est donc conseillé pour les très gros documents de les générer dans un fichier, et d'envoyer des données de temps en temps au navigateur (par exemple page 1, page 2... en utilisant flush() pour forcer l'envoi). Lorsque le fichier est terminé, vous pouvez effectuer une redirection dessus avec JavaScript ou bien créer un lien.
Remarque : même lorsque le navigateur part en time-out, il est possible que le script continue à s'exécuter sur le serveur.

19. Est-ce que je peux modifier un PDF avec FPDF ?

Non.

20. Je voudrais faire un moteur de recherche en PHP et indexer des PDF. Est-ce que je peux le faire avec FPDF ?

Non. Par contre il existe un utilitaire GPL en C, pdftotext, capable d'extraire le contenu textuel d'un PDF. Il est fourni avec l'archive de Xpdf :

http://www.foolabs.com/xpdf/

21. Est-ce que je peux transformer une page HTML en PDF avec FPDF ?

Non, on ne peut convertir que de l'HTML très simple, pas des pages complètes. Par contre il existe un utilitaire GPL en C, htmldoc, qui permet de le faire et qui donne de bons résultats :

http://www.easysw.com/htmldoc/

22. Est-ce que je peux concaténer des PDF avec FPDF ?

Non. Par contre il existe un utilitaire gratuit en C, mbtPdfAsm, capable de le faire :

http://thierry.schmit.free.fr/dev/mbtPdfAsm/mbtPdfAsm2.html

23. Comment faire pour activer les protections sur un PDF ? Je voudrais empêcher qu'on puisse copier-coller le texte ou modifier le document.

On ne peut pas actuellement. La fonctionnalité sera implémentée à l'avenir.