Tri alphabétique des auteurs

Lister tous vos auteurs par ordre alphabétique selon leur nom

Cet article est une variante de la contribution Tri alphabétique tout en SPIP pour afficher la liste alphabétique de vos auteurs en fonction de leur nom

SPIP stocke le nom dans un seul champ. Les auteurs rentrent généralement le prénom nom et on désire un classement par nom
ex. Marcel Duchamp doit être rangé dans D et non dans M.

Ajout d’un filtre pour récupérer le nom

On crée un filtre (à ajouter dans mes_fonctions.php) pour déterminer la 1re lettre du nom des auteurs.
ex. Marcel Duchamp renvoie D

// ---------------------------------------
// Filtre lastfirstletter
// extrait la première lettre du dernier mot et la passe en majuscules
// ex. marcel duchamp -> D
// ---------------------------------------
function lastfirstletter($texte) {

  $pos = strrpos(trim($texte), " ");
  if ($pos === false) { // pas trouvé -> 1er lettre
      $texte = $texte{0};
  } else {
      $texte = $texte{$pos+1};
  }
	// remplacement des caractères accentués
	$texte = strtr($texte, "\xA1\xAA\xBA\xBF\xC0\xC1\xC2\xC3\xC5\xC7\xC8\xC9\xCA\xCB\xCC\xCD\xCE\xCF\xD0\xD1\xD2\xD3\xD4\xD5\xD8\xD9\xDA\xDB\xDD\xE0\xE1\xE2\xE3\xE5\xE7\xE8\xE9\xEA\xEB\xEC\xED\xEE\xEF\xF0\xF1\xF2\xF3\xF4\xF5\xF8\xF9\xFA\xFB\xFD\xFF", "!ao?AAAAACEEEEIIIIDNOOOOOUUUYaaaaaceeeeiiiidnooooouuuyy");
	$texte = strtr($texte, array("\xC4"=>"Ae", "\xC6"=>"AE", "\xD6"=>"Oe", "\xDC"=>"Ue", "\xDE"=>"TH", "\xDF"=>"ss", "\xE4"=>"ae", "\xE6"=>"ae", "\xF6"=>"oe", "\xFC"=>"ue", "\xFE"=>"th"));
	$texte = strtoupper($texte);          // tout en majuscules
	if (is_numeric($texte)) $texte = "";  // on supprime les chiffres
	return $texte;
	
} // lastfirstletter

Intégration dans les boucles

On va utiliser le même principe que la contribution originale à savoir :

  • une boucle pour générer la liste alphabétique
  • une boucle pour stocker les auteurs dans un tableau
  • une boucle pour lister les auteurs correspondant à la lettre passée en paramètre
<BOUCLE_listalpha(AUTEURS){par nom}{tout}>[(#SET{beginby[(#NOM|lastfirstletter)],[(#GET{beginby[(#NOM|lastfirstletter)]}|array_merge{#ARRAY{#COMPTEUR_BOUCLE,#ID_AUTEUR}})]})][<a href="[(#SELF|parametre_url{lettre,[(#NOM|lastfirstletter)]})]" [(#NOM|lastfirstletter|=={[(#ENV{lettre,A})]}|?{class='on'})]>(#NOM|lastfirstletter|unique)</a> ]</BOUCLE_listalpha>
          
[(#REM)2eme boucle stocke dans chaque Arraylettre les noms des auteurs]
<BOUCLE_listalphadeux(AUTEURS){par nom}{tout}>[(#SET{beginby[(#NOM|lastfirstletter)],[(#GET{beginby[(#NOM|lastfirstletter)]}|array_merge{#ARRAY{#COMPTEUR_BOUCLE,#ID_AUTEUR}})]})]</BOUCLE_listalphadeux> 

[(#REM)3eme boucle affiche les auteurs stockés dans Arraylettre si seulement ils correspondent bien à la lettre]
[<h2>(#ENV{lettre,A})</h2>]
<BOUCLE_artlettre(AUTEURS){id_auteur IN #GET{beginby#ENV{lettre,A}}}{tout}{doublons alphabet}>
<a href="#URL_AUTEUR">#NOM</a><br />  
</BOUCLE_artlettre>

Note : le critère tout est facultatif, il permet de lister tous les auteurs. Si vous l’enlevez, vous n’afficherez que les auteurs qui ont déjà publiés un article dans l’espace public.

Dernière modification de cette page le 8 décembre 2008

Discussion

7 discussions

  • Une approche pour un SPIP >=2 ? serait d’utiliser le critère fusion (produit la fonction SQL GROUPE BY) avec la fonction SQL LEFT (extraire le nombre de premiers caractères indiqués, ici ce sera 1). Donc :

    <BOUCLE_listalpha1(AUTEURS){par nom}{fusion LEFT(nom,1)}>
      #NOM
    </BOUCLE_listalpha1>

    Depuis SPIP 2, je crois, on peut utiliser pratiquement n’importe quelle fonction PHP comme filtre... on va utiliser la fonction PHP substr pour n’afficher que les lettres (sinon le resultt n’a pas trop de sens) :

    <BOUCLE_listalpha1(AUTEURS){par nom}{fusion LEFT(nom,1)}>
      [(#NOM|substr{0,1})]
    </BOUCLE_listalpha1>

    Maintenant, y a plus qu’a positionner les liens :

    <BOUCLE_listalpha1(AUTEURS){par nom}{fusion LEFT(nom,1)}>
      [(#SET{lalettre,[#NOM|substr{0,1}]})]
      [<a href="[(#SELF|parametre_url{lettre,[(#GET{lalettre})]})]" [(#GET{lalettre}=={[(#ENV{lettre,A})]}|oui)class='on']> [(#GET{lalettre})] </a>
    </BOUCLE_listalpha1>

    Voila, voila.
    (disclamer : boucle non testée, c’est pour exposer le principe... que je rescite cette contrib..)

    Répondre à ce message

  • 2

    Bonjour,

    Super script et très utile ! est-ce que quelqu’un à régler le problème de l’affichage de la liste alphabétique qui est totalement désordonnée chez moi aussi (et ce malgré le critère par nom), cela fait deux jours que je cherche et je bloque complètement, alors merci d’avance ;=)

    • Bonjour,
      Sur un site externe sous spip 2.0.2 j’ai ça comme titre de liste Alpha des auteurs dans la zone administrateur. La liste commence par les administrateurs avant de présenter les rédacteurs.
      A G L M P V Z B C D E F H I J K N O Q R S T U W X Y

      Sur un autre site en local sous Ubuntu avec la même base j’ai une liste alpha normale. Dans ce cas les administrateurs sont mélangés aux rédacteurs !!!
      Comprend qui peux ?
      Merci pour votre aide.
      Alain

    • Bonjour à tous,

      J’ai essayé d’adapter cette contrib aux articles (j’ai une rubrique où mes titres d’articles sont tous sous la forme « Prénom Nom »).
      Cela marche plutôt bien, le seul hic est que, comme Alain Bourdeau et Corinne, j’ai mes ancres de pagination qui ne sont pas triées alphabétiquement.

      J’ai choisi dans un premier temps de contourner cela en écrivant les ancres de pagination en « dur » dans mon squelette. C’est pas génial comme solution (pas très propre), mais ça marche.

      • j’ai supprimé [<a href="[(#SELF|parametre_url{lettre,[(#NOM|lastfirstletter)]})]" [(#NOM|lastfirstletter|=={[(#ENV{lettre,A})]}|?{class='on'})]>(#NOM|lastfirstletter|unique)</a> ] de la boucle BOUCLE_listalpha
      • J’ai inséré 26 liens entre BOUCLE_listalpha et BOUCLE_listalphadeux
        • <a href="[(#SELF|parametre_url{lettre,A})]" [(#ENV{lettre,A}|=={A}|?{class='on'})]>A</a>
        • <a href="[(#SELF|parametre_url{lettre,B})]" [(#ENV{lettre,A}|=={B}|?{class='on'})]>B</a>
        • <a href="[(#SELF|parametre_url{lettre,C})]" [(#ENV{lettre,A}|=={C}|?{class='on'})]>C</a>
        • etc... (notez bien que #ENV{lettre,A} ne change pas d’un lien à l’autre)

      En espérant trouver une solution plus propre prochainement.

    Répondre à ce message

  • formidable mais ya pas de s a Duchamp

    Répondre à ce message

  • Alexandre

    Pour ceux qui se cassent la tête avec une pagination alphabétique des signatures :

    -  fichier mes_fonctions.php (fichier à créer dans le dossier squelettes, si vous ne l’avez déjà fait) :

    <?php
    //squelettes/mes_fonctions.php
    //extrait la première lettre et la passe en majuscules
    function lastfirstletter($texte) {
    	$texte = $texte{0}; // première lettre
    	// remplacement des caractères accentués
    	// exemple trouvé la: 
    	// http://be.php.net/manual/fr/function.strtr.php#52098
    	$texte = strtr($texte, "\xA1\xAA\xBA\xBF\xC0\xC1\xC2\xC3\xC5\xC7\xC8\xC9\xCA\xCB\xCC\xCD\xCE\xCF\xD0\xD1\xD2\xD3\xD4\xD5\xD8\xD9\xDA\xDB\xDD\xE0\xE1\xE2\xE3\xE5\xE7\xE8\xE9\xEA\xEB\xEC\xED\xEE\xEF\xF0\xF1\xF2\xF3\xF4\xF5\xF8\xF9\xFA\xFB\xFD\xFF", "!ao?AAAAACEEEEIIIIDNOOOOOUUUYaaaaaceeeeiiiidnooooouuuyy");
    	$texte = strtr($texte, array("\xC4"=>"Ae", "\xC6"=>"AE", "\xD6"=>"Oe", "\xDC"=>"Ue", "\xDE"=>"TH", "\xDF"=>"ss", "\xE4"=>"ae", "\xE6"=>"ae", "\xF6"=>"oe", "\xFC"=>"ue", "\xFE"=>"th"));
    	$texte = strtoupper($texte); // tout en majuscules
    	//if($texte!='&')
    	return $texte;
    }
    
    ?>

    -  dans votre squelette :

    <BOUCLE_tot(SIGNATURES){id_article}>
    </BOUCLE_tot>
    
       <?php
               $l=[(#ENV{lettre,a})];
               //liste des lettres de l'alphabet...
               for ($i=ord("a");$i<=ord("z");$i++){
                   echo (($l !=chr($i)) ? ('<a href="#URL_PAGE{signatures,id_article=#ID_ARTICLE&lettre='.chr($i).'"><strong>'.mb_strtoupper(chr($i)).'</strong></a>') : mb_strtoupper(chr($i)));
                   echo " &nbsp; ";
               }
       ?>
    </B_tot>
    
    <BOUCLE_listalpha (SIGNATURES){par nom_email}{tout}>
    
    [(#SET{beginby[(#NOM|lastfirstletter)],[(#GET{beginby[(#NOM|lastfirstletter)]}|array_merge{#ARRAY{#COMPTEUR_BOUCLE,#ID_SIGNATURE}})]})]
    
    [<a href="[(#SELF|parametre_url{lettre,[(#NOM|lastfirstletter)]})]" [(#NOM|lastfirstletter|=={[(#ENV{lettre,a})]}|?{class='on'})]>(#NOM|lastfirstletter|unique)</a> ]
    
    </BOUCLE_listalpha>
    
    <BOUCLE_signatures_alpha(SIGNATURES) {nom_email==^#ENV{lettre,a}} {par nom_email}>[(#NOM)]<br />
    </BOUCLE_signatures_alpha>

    Je ne sais pas si c’est tout à fait propre mais ça marche !

    Et si vous avez des problèmes d’accents utilisez le filtre |utf8_encode ou |utf8_decode (à choisir en fonction du charset de départ) sur la balise [(#NOM)]. Ce qui donne [(#NOM|utf8_encode)]

    Répondre à ce message

  • 1

    C’est bien joli, mais la prémisse sur laquelle votre code est basé est fausse : le nom n’est pas nécessairement le second mot de la chaîne. C’est bien sûr très courant dans le monde anglo-saxon, mais on trouve aussi des occurrences en français. Bref, ce procédé n’est pas fiable.

    • En fait si vous lisez bien la fonction, le nom est le dernier « mot » de la chaine, l’auteur pour avoir plusieurs prénoms (ou aucun) :

      • Paul Robert Parker
      • John Jack Gilles Ferreira
      • Inès Fakhir-Négrin
      • Cabu
      • ...

      Par contre, cela peut poser problème avec des noms à particules séparés

      • Pierre La Motte du Petit Bois (retourne Bois)
      • Angus Mac Angus (retourne Angus, dans ce cas, écrire Angus McAngus)
      • Barack Bush Jr
      • ...

      Dernier cas particulier qui peut poser problème :

      • Louis XVI
      • Pie VI
      • ...

      La solution proposée n’est donc pas infaillible mais permet une mise en place facile sans à avoir à renommer tous vos auteurs et reprendre les autres pages de votre squelette. Si vous voulez une solution « parfaite », utiliser la méthode de Fil (voir lien plus bas)

    Répondre à ce message

  • 6

    Je cherchai à faire ça depuis un bon moment et voilà que ça semble me tomber tout cuit dans le bec ! Mais... lorsque j’essaye de mettre cette contrib en place sur mon site qui contient 83 auteurs ( 80 6forum et 3 administrateurs) j’ai le message suivant qui se répète 5 fois : "Erreur : filtre « array_merge5 » non défini, _listalpha".

    Si, dans la partie squelette, j’enlève le paramètre tout dans les 3 boucles, j’obtiens la liste de lettre correspondant aux initiales des administrateurs C L, la lettrine A mais après, plus rien ne s’affiche.

    Si, dans la partie squelette, j’enlève le paramètre tout dans la 1re boucle, j’obtiens cette fois-ci toujours bien la liste de lettre correspondant aux initiales des administrateurs C L, la lettrine A et les 4 auteurs dont le nom commence par un A.

    Franchement, je n’y comprends rien. Si quelqu’un à une idée...

    D’avance merci.

    • c’est du à l’évolution de la fonction array_merge de php4 à php5

      si vous êtes en PHP4
      utiliser l’exemple de la contrib avec le filtre array_merge

      si vous êtes en PHP5

      1. dans vos boucles remplayer les array_merge par array_merge5
      2. ajouter dans mes_fonctions.php :
      function array_merge5 ($array1, $array2) {
          return array_merge ((array)$array1, (array)$array2);
      }

      il faudrait écrire un filtre générique indépendant de la version de PHP ....

    • Merci de ta réponse,

      Je n’ai plus de message d’erreur mais au niveau fonctionnement, ce n’est pas encore ça.
      Voici le résultat obtenu :

      B A C L G H O J P M D E S F Y N T R W Z (lettres cliquables (non classées alphabétiquement) qui correspondent bien à toutes les initiales des noms de mes auteurs quoi que je n’ai aucun nom commençant par W)

      A (en lettrine)

      Alfred AGROUN
      Boussad ABGRES
      Jean-Stéphane ANTINIOLLE
      Rezak AYAUDA
      Tchao ASLIHA

      Cet affichage correspond bien aux cinq auteurs dont le nom commence par la lettre A. Par contre, ce sont toujours les mêmes 5 noms qui apparaissent quelque soit la lettre sur laquelle je clique dans la liste proposée.

      Au fait, dans l’article cité plus haut (Tri alphabétique tout en SPIP) il est spécifié qu’il faut être en SPIP 1.9.2 en UTF8 or, je suis bien en SPIP 1.9.2 mais ma base est en Latin1. Le problème viendrait-il de là ? Si oui, comment y remédier ?

      Bruno.

      1. bien vérifier que vous n’avez pas oublié le tri {par nom} dans vos boucles
      2. essayer de passer votre base en UTF-8
    • J’ai bien le tri par nom dans mes boucles car j’ai fait un simple copier coller de votre code fourni. J’ai essayé de passer ma table auteur et le champ nom en UTF8 mais ça donne le même résultat (je n’imagine même pas devoir passer toute ma base en UTF8 et gérer tous les problèmes d’accents qui en découlerait dans tous mes articles, mes brèves, etc.)

    • Si cela peut donner une piste à faire en mode SPIP et à adapter ce code est plus court mais moitié en php... il était destiné aux pétitions, merci et pardon à l’auteur de ce code dont j’ai oublié l’adresse...

      <BOUCLE_tot(SIGNATURES){id_article}>
      </BOUCLE_tot>
       
      <h3>Liste par ordre alphab&eacute;tique des #GRAND_TOTAL signataires</h3> 
       
      	<?php
      			$l=[(#ENV{lettre,a})];
      			//liste des lettres de l'alphabet...
      			for ($i=ord("a");$i<=ord("z");$i++){
      				echo (($l !=chr($i)) ? ('<a href="#URL_PAGE{signatures,id_article=#ID_ARTICLE&lettre='.chr($i).'"><strong>'.mb_strtoupper(chr($i)).'</strong></a>') : mb_strtoupper(chr($i)));
      				echo " &nbsp; ";
      			}
      	?>
      </B_tot>
       
      <BOUCLE_signatures_alpha(SIGNATURES) {nom_email==^#ENV{lettre,a}} {par nom_email}>#NOM
      </BOUCLE_signatures_alpha>
    • Il me semble qu’il suffit de déclarer l’array avant la boucle,

      #SET{arrayarts,#ARRAY}

    Répondre à ce message

  • 2

    Pour obtenir quelque chose de similaire la méthode donnée par Améliorer l’affichage public et la gestion des auteurs et traducteurs d’un site SPIP est plus complète, et à mon sens plus pratique. Elle impose de saisir les noms sous la forme « Nom*Prénom », et explique à SPIP comment remettre les choses à l’endroit au moment de l’affichage.

    • Bonjour,

      Je pense qu’il faut vraiment éviter au maximum tout ce qui est codage arbitraire. Genre ajouter des étoiles au milieu des noms.

      Pour un informaticien, comme toi et moi, c’est facile, mais cela parait souvent trop ardu pour des gens qui n’y sont pas habitués et qui ont pourtant tout à fait vocation à gérer l’ espace privé.

      A mon sens, le mieux serait qu’il y ait en standard dans une prochaine version de SPIP les champs « Prénom » et « Nom », et d’utiliser un plugin attendant.

      Martinus

    • Bof pour la mise en standard : perso je préfère un seul champ « nom ou pseudo ». Mais ce qui serait bien c’est qu’on puisse ajouter des champs de façon simple. Tu ajouterais prénom, et ça marcherait directement.

    Répondre à ce message

Ajouter un commentaire

Qui êtes-vous ?

Pour afficher votre trombine avec votre message, enregistrez-la d’abord sur gravatar.com (gratuit et indolore) et n’oubliez pas d’indiquer votre adresse e-mail ici.

Ajoutez votre commentaire ici

Ce champ accepte les raccourcis SPIP {{gras}} {italique} -*liste [texte->url] <quote> <code> et le code HTML <q> <del> <ins>. Pour créer des paragraphes, laissez simplement des lignes vides.

Ajouter un document

Suivre les commentaires : RSS 2.0 | Atom