Tri alphabétique tout en SPIP

EDIT Mars 2016

Pour afficher une liste alphabétique en SPIP3 (ou bien en SPIP2 avec le plugin iterateurs) il suffit de faire

  1. <BOUCLE_alpha(DATA){enum A,Z}>[<a href='##VALEUR'>(#VALEUR)</a>]</BOUCLE_alpha>

Ensuite remplir les tableaux de chaque lettre avec les titres et identifiants commençant par cette lettre, par exemple sur des mots :

<BOUCLE_tableaux(MOTS){id_groupe=4}{par titre}>
[(#SET{premierelettre,[(#TITRE|substr{0,1})]})]
[(#SET{beginby[(#GET{premierelettre}|unique)],#ARRAY})]
[(#SET{beginby[(#GET{premierelettre})],[(#GET{beginby[(#GET{premierelettre})]}|array_merge{#ARRAY{#TITRE,#ID_MOT}})]})]
</BOUCLE_tableaux>

Il ne reste plus qu’à utiliser une boucle DATA à l’intérieur d’une boucle alphabétique pour y rappeler chaque tableau

<BOUCLE_alpha2(DATA){enum A,Z}>
<B_data>
[<h1 id="#VALEUR">(#VALEUR)</h1>]
	<ul> 
      <BOUCLE_data(DATA){source tableau, #GET{beginby#VALEUR}}{par cle}>
      <li><a href="#URL_MOT{#VALEUR}">#CLE</a></li>
      </BOUCLE_data>
      </ul>
</B_data>
</BOUCLE_alpha2>

Vous êtes en SPIP 1.9.2 ou supérieur et en UTF8, vous avez une liste d’articles ou d’auteurs importante ? près de 1000 ? et vous souhaitez la voir s’afficher par ordre alphabétique ainsi
A|B|C|D|E|F...

Pire, vous voulez la trier en ne prenant en compte que ce qui suit l’article du titre et donc exclure le la les ou l’

Quoi ! vous voudriez en plus qu’à la demande du visiteur les articles s’affichent suivant la lettre demandée comme ici pour A :
-  animal(l’)
-  animaleries(les)
-  animaux(les)

Alors... suivez le guide !

On commence par la fonction qui permet d’extraire la première lettre.
On met cette fonction dans le fichier squelettes/mes_fonctions.php
qui est du pur php, et commençera donc par <?php pour finir par ?>

//squelettes/mes_fonctions.php
//extrait la première lettre et la passe en majuscules
function onelettre($texte) {
	$texte = $texte{0}; // première lettre
	// remplacement des caractères accentués
	// exemple trouvé là: 
	// 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;
}

Vous pouvez déjà voir ce que ça donne dans une boucle !
[(#TITRE|onelettre)] devrait retourner A ou B ou Z

Ajout du 4 septembre 2011

Essayez la solution de marcimat avec le code php |substr{0,1}

<B_alpha>
  <p class="pagination">
<BOUCLE_alpha(ARTICLES) {par titre} {fusion LEFT(titre,1)}>
  <a href='[(#URL_PAGE{titi}|parametre_url{lettre, [(#TITRE|substr{0,1})]})]'>
  [(#TITRE|substr{0,1})]
  </a>                
</BOUCLE_alpha>                        
  </p>
</B_alpha>

Utiliser #ARRAY

Avec SPIP 1.9.2 on va s’amuser sur la possibilité de stocker un array pour l’envoyer dans une boucle, on a donc deux boucles distinctes :

-  BOUCLE_listalpha n’affiche qu’une seule fois la première lettre et grace à SET{beginby[(#TITRE|onelettre)]...
stocke dans un tableau (array) la liste des articles de cette lettre,
pour tester le tableau de la lettre B on fera [(#GET{beginbyB}|print_r)]
-  BOUCLE_artlettre, la deuxième boucle, récupère ensuite le tableau ainsi crée, grace à #ENV et si il n’y a pas de lettre demandé on commence par A #GET{beginby#ENV{lettre,A}simple non ?

On colle donc le code suivant dans le squelette de son choix, par exemple dans rubrique-5.html qui pourrait correspondre à un catalogue ou chaque livre est un article.

Index alphabetique des titres<br />
<div class="alphabetique">

<BOUCLE_listalpha(ARTICLES){id_secteur}{par titre}>[(#SET{beginby[(#TITRE|onelettre)],[(#GET{beginby[(#TITRE|onelettre)]}|array_merge{#ARRAY{#COMPTEUR_BOUCLE,#ID_ARTICLE}})]})][<a href="[(#SELF|parametre_url{lettre,[(#TITRE|onelettre)]})]" [(#TITRE|onelettre|=={[(#ENV{lettre,A})]}|?{class='expose'})]>(#TITRE|onelettre|unique)</a>|]</BOUCLE_listalpha> 

<ul class="laliste"> 
<BOUCLE_artlettre(ARTICLES){id_article IN #GET{beginby#ENV{lettre,A}} }{id_secteur}>
<li><a href="#URL_ARTICLE"><span>[(#TITRE)][ ((#TITRE))]</span> </a></li> 
</BOUCLE_artlettre>
</ul>

</div>

Toujours plus loin !

Ah ! mais... il y a l’animal et les animaux qui se sont mis sous le L !
ça ne va pas du tout ! et peut être aussi les lettres ne sont pas l’ordre alphabétique...

On reprend mes_fonctions.php et changer le critère de tri dans la première BOUCLE (merci hubombing)

function critere_parsansL($idb, &$boucles, $crit) { // retire el " le los l' la le les the et tri après FROM TRIM(LEADING \'l’ «\'  FROM TRIM(LEADING \'Le « \' FROM TRIM(LEADING \' \'  FROM TRIM(LEADING \'&amp\'
    $not = $crit->not;
    $boucle = &$boucles[$idb];
    $id = $boucle->primary;
 	$boucle->order[] = "'TRIM(LEADING \'Le «\' FROM TRIM(LEADING \'le \'FROM TRIM(LEADING \'los \'  FROM TRIM(LEADING \'la \' FROM TRIM(LEADING \'l’ «\'  FROM TRIM(LEADING \'Le « \'   FROM TRIM(LEADING \'La « \' FROM TRIM(LEADING \'l’\' FROM TRIM(LEADING \'«\' FROM TRIM(LEADING \'&nbsp;\' FROM TRIM(LEADING \'les \' FROM LOWER(titre))))))))))))'";
}

On remplace donc le critère par titre de la première BOUCLE par parsansL

Les Hics

Quelques hics subsistent, les mots avec accents ou avec par exemple Œ, ne sont pas classés dans la base, ainsi L’Œuvre ou L’écureuil se retrouve en bas de page des A. Ajout du 4 septembre 2011 [1]

On commence donc par une boucle simple qui va seulement chercher chaque lettre de l’alphabet, évidemment il vous faut suffisamment d’articles !

Pour résoudre l’affichage des accents dans leur bonne lettre, sachant que les accents du genre éèô et le cher e dans l’o sont stockés dans le tableau array du A, il faut empêcher l’affichage des articles commençant par L’Œuvre et réinterroger le tableau A en mettant une 4em boucle doublons alphabet

Eh mais, en plus mes articles ne sont pas du tout rangés là !!?
Psst, Il manque 3 fonctions encore !

//squelettes/mes_fonctions.php
/* on retire les le la etc */
function sansle($texte) {
	$pattern[0]  = "#^Les |La |Le |Lo |The[[:space:]]?#";   
	$pattern[1]  = "#^L’?#"; //apostrophe utf8
	$pattern[3] = "#^&\#171;?#"; //guillemet
	$pattern[5] = "#^&?#"; //&
	$pattern[2]  = "#^&nbsp;?#"; //espace
	$pattern[6] = "#^[[:space:]]?#"; //&
	$pattern[4]  = "#«#"; //guillemet La «  
	$texte = preg_replace($pattern, '', $texte); 
	return $texte;
}


/* on ne garde que le la l'*/
function quele($texte){
$txtsanse=sansle(trim($texte));
if ($txtsanse!=trim($texte))
		$texte= str_replace("$txtsanse","","$texte");
		else $texte='';
return trim($texte); 
}

function onelettrebis($chaine) {
$chaine=sansle($chaine); 
$chaine=filtrer_entites($chaine); // si il y a des fois des accents en dur qui trainent
$chaine = unicode2charset(utf_8_to_unicode($chaine), 'iso-8859-1'); // on code en html ISO
    $a = "ÀÁÂÃÄÅàáâãäåÒÓÔÕÖØòóôõöøÈÉÊËèéêëÇçÌÍÎÏìíîïÙÚÛÜùúûüÿÑñ";
    $b = "AAAAAAaaaaaaOOOOOOooooooEEEEeeeeCcIIIIiiiiUUUUuuuuyNn";
$chaine=strtr($chaine, $a, $b); // on retire les accents
$chaine=strtoupper($chaine); // on passe en majuscules 
/// si débute par le fameux Œ ou œ
$pattern = "^(Œ|œ)";
if (eregi($pattern,$chaine,$regs)) 
$chaine='O';//$chaine= $regs[0]; //on va renvoyer la lettre O
else 
$chaine = $chaine{0}; 
    return $chaine ;
}

Au final

Voici donc le code final, avec le nouveau critère de tri et les filtres ajoutés

<div id="alphabetique">
<h3>Index alphab&eacute;tique des titres</h3>
[(#REM) <!-- on met dans un tableau les articles qui commencent par la même lettre -->]
<div class="alpha">
[<!--(#REM)La première boucle affiche seulement chaque lettre de l'alphabet avec son lien-->]
<BOUCLE_listalphaone(ARTICLES){id_secteur}{par titre}>[<a href="[(#SELF|parametre_url{lettre,[(#TITRE|onelettre)]})]" [(#TITRE|onelettre|=={[(#ENV{lettre,A})]}|?{id='expose'})]>(#TITRE|onelettre|unique)</a>]</BOUCLE_listalphaone>
[<!--(#REM)La deuxième boucle stocke dans chaque Arraylettre les titres des articles +ou- alphabétiquement sans le l' la les..-->]
<BOUCLE_listalphadeux(ARTICLES){id_secteur}{parsansL}>[(#SET{beginby[(#TITRE|sansle|onelettre)],[(#GET{beginby[(#TITRE|sansle|onelettre)]}|array_merge{#ARRAY{#COMPTEUR_BOUCLE,#ID_ARTICLE}})]})]</BOUCLE_listalphadeux> 
<br class="nettoyeur" />
</div>
[<h1>(#ENV{lettre,A})</h1>]
<ul class="laliste"> 
[(#REM) test du tableau faire  [(#GET{beginbyB}|print_r)]]  

[(#REM)<!--La troisième boucle affiche les articles stockés dans Arraylettre si seulement ils correspondent bien à la lettre-->]
<BOUCLE_artlettre(ARTICLES){id_article IN #GET{beginby#ENV{lettre,A}} }{id_secteur}{doublons alphabet}>
[<li>
(#TITRE|onelettrebis|=={#ENV{lettre,A}}|?{' ',''}) 
<a href="#URL_ARTICLE"><span>[(#TITRE|sansle)][ ((#TITRE|quele))]</span> #SOUSTITRE</a>
</li>]
</BOUCLE_artlettre>

[(#REM)<!--La quatrième boucle réinterroge les articles stockés de ArrayA dans lequel il y a les accents, ils correspondent bien à la lettre-->] 
<BOUCLE_artlettreb(ARTICLES){id_article IN #GET{beginbyA} }{id_secteur}{doublons alphabet}> 
[<li>
(#TITRE|onelettrebis|=={#ENV{lettre,A}}|?{' ',''}) 
<a href="#URL_ARTICLE"><span>[(#TITRE|sansle)][ ((#TITRE|quele))]</span> #SOUSTITRE</a>
</li>]
</BOUCLE_artlettreb>
 
</ul>
</div>

Avec du style s’il vous plait !

<style>
#alphabetique{ font-size:0.7em; color:#555} 
#expose{text-decoration:none; background-color:#FFCC00}
.laliste a{color:#555;}
.laliste a span{color:#000;}
</style>

Et voila le résultat

À l’épreuve du site Quilombo rubrique catalogue.

Vous pouvez adapter le même principe aux auteurs ou aux mots clefs.

Notes

[1Et pour passer en UTF-8 votre base MYSQL et éviter ce souci sur un vrai tri alphabétique correct essayez aussi /ecrire/?exec=convert_sql_utf8

Dernière modification de cette page le 9 février 2019

Discussion

18 discussions

  • 19

    Salut,

    J’utilise parfaitement cette contribution et je suis ravi du résultat. Merci pour ce travail !

    Cependant j’aimerais pouvoir classer chaque mots du titre par ordre alphabétique

    CAD :

    pour un article ayant pour titre : Adaptation Tiller pour montage et entretien des planches

    J’aimerais pouvoir le retrouver dans ’A’ pour Adaptation, mais aussi dans ’T’ pour Tiller et aussi dans ’M’ pour montage etc ...

    Une idée là dessus ?

    • Bonjour,
      merci de tes remerciements, à mon avis la boucle DATA est plus simple à mettre en œuvre.
      Pour ton problème, tu pourrais construire un array pour chaque titre avec un explode sur l’espace vide, où chaque valeur sera un mot.
      http://www.spip.net/fr_article5671.html
      Sauf que tu devras constituer un array pour retirer les mots comme pour,de,avec,quoi,qui,donc,etc …

      Pour faire simple, à moins que tu aies envie de te décarcasser en php, je suppose (car je ne sais pas ce que tu souhaites faire exactement) qu’un système de mots clefs serait plus adapté.
      ++

    • Merci pour ta réponse rapide
      Je débute totalement avec spip et son environnement et encore plus avec Php ... Je ne suis pas très à l’aise avec tout ça.
      J’ai pensé à utiliser les mots clés au départ mais les articles sont générés depuis un site syndiqué et j’ai plus d’un millier d’articles à traiter, jouer avec les mots clés me prendrait un temps fou
      La solution Tri alphabétique tout en spip me va bien du coup mais il me manque la partie énoncée plus haut.

      Un petit coup de main là dessus ne serait pas de refus.

      Merci d’avance

    • Hello,

      Voici une solution. Ça reprend le même principe que l’article de Touti, à savoir stocker les lettres utiisées et les identifiants des articles correspondants, afficher un menu alphabet, et enfin afficher les articles de la lettre sélectionnée, mais avec quelques différences formelles.

      L’affichage est minimaliste, c’est à adapter bien sûr.

      Dans le squelette :

      [(#REM) 1) On fait un tableau contenant les lettres et les articles correspondants.
              Ces boucles n'affichent rien, elles servent juste à stocker les données. ]
      #SET{lettres,#ARRAY}
      <BOUCLE_articles(ARTICLES)>
      [(#REM) on itère sur les 1ères lettres des mots du titre ]
      <BOUCLE_lettres_article(DATA){source table, #TITRE|premieres_lettres}>
      [(#REM) on récupère les identifiants des articles de la lettre, et on y ajoute celui de l'article ]
      #SET{ids,#GET{lettres/#VALEUR}|sinon{#ARRAY}|push{#ID_ARTICLE}}
      [(#REM) on met à jour la lettre et les ids associés dans le tableau ]
      #SET{lettres,#GET{lettres}|array_merge{#ARRAY{#VALEUR,#GET{ids}}}}
      </BOUCLE_lettres_article>
      </BOUCLE_articles>
       
      [(#REM) 2) Affichage du menu alphabet, il n'affiche que les lettres réellement utilisées dans les articles. ]
      <B_alphabet>
      <p class="pagination">
      <BOUCLE_alphabet(DATA){source table, #GET{lettres}|array_keys}{par valeur}>
      	[(#ENV{lettre}|non|et{#COMPTEUR_BOUCLE|=={1}}|oui)#SET{lettre,#VALEUR}]
      	<a href='[(#SELF|parametre_url{lettre,#VALEUR})]'>#VALEUR</a>
      </BOUCLE_alphabet>
      </p>
      </B_alphabet>
       
      [(#REM) 3) Affichage des articles correspondant à la lettre active (= dans l'URL) ]
      #SET{lettre,#ENV{lettre,#GET{lettre}}}
      <B_articles_lettre>
      <ul>
      <BOUCLE_articles_lettre(ARTICLES){id_article IN #GET{lettres/#GET{lettre}}}>
      <li>#TITRE</li>
      </BOUCLE_articles_lettre>
      </ul>
      </B_articles_lettre>

      Dans mes_fonctions.php :

      /**
       * Retourne une liste de lettres correspondant aux 1ères lettres des mots d'un texte.
       *
       * @param string $texte
       * @return string|array
       */
      function premieres_lettres($texte) {
       
      	// liste des mots
      	if ($mots = str_word_count($texte, 1)) {
       
      		// on stocke la 1ère lettre de chaque mot dans un tableau
      		$lettres = array();
      		foreach($mots as $mot) {
      			// on ne prend en compte que les mots d'au moins 4 lettres
      			if (strlen($mot) > 3) {
      				$lettres = array_merge($lettres, array(strtolower(substr($mot,0,1))));
      			}
      		}
       
      		if (count($lettres)) {
      			$texte = $lettres;
      		}
      	}
       
      	return $texte;
      }
    • C’est carrément ce qu’il fallait, quelques retouches pour faire ça beau prop’ et c’est nickel !

      Merci beaucoup

    • Salut,

      Merci pour le coup de patte, j’ai effectué quelques corrections sur la fonction :

      function premieres_lettres($texte) {
       
      	// liste des mots
      	if ($mots = str_word_count($texte, 1, 'ÀÁÂÃÄÅàáâãäåÒÓÔÕÖØòóôõöøÈÉÊËèéêëÇçÌÍÎÏìíîïÙÚÛÜùúûüÿÑñ')) {
      		// on stocke la 1ère lettre de chaque mot dans un tableau
      		$lettres = array();
       
      		foreach($mots as $mot) {
      			// on ne prend en compte que les mots d'au moins 4 octets
      			if (strlen($mot) > 3) {
      				$lettres = array_merge($lettres, array(strtoupper(substr($mot,0,1))));
       
      			}
      		}
       
      		if (count($lettres)) {
      			$texte = $lettres;
       
      		}
      	}
       
      	return $texte;
      }

      Il ne me manque plus que de pouvoir surligner ou souligner le mot retenu pour la lettre selectionnée.

    • Tiens, les envois de notification de ce forum sont très aléatoires je n’ai pas reçu les 2 derniers messages.
      Tu peux aussi constituer et ajouter un tableau des mots interdits

      function premieres_lettres($texte) {
       
      	//array des mots interdits à compléter
      	$mots_interdits = array('avec','sans','pour','etc');
       
          	// liste des mots
          	if ($mots = str_word_count($texte, 1, 'ÀÁÂÃÄÅàáâãäåÒÓÔÕÖØòóôõöøÈÉÊËèéêëÇçÌÍÎÏìíîïÙÚÛÜùúûüÿÑñ')) {
          		// on stocke la 1ère lettre de chaque mot dans un tableau
          		$lettres = array();
       
       
          		foreach($mots as $mot) {
          			// on ne prend en compte que les mots d'au moins 4 octets
          			if (strlen($mot) > 3 AND !in_array($mot,$mots_interdits)) {
          				$lettres = array_merge($lettres, array(strtoupper(substr($mot,0,1))));
       
          			}
          		}
       
          		if (count($lettres)) {
          			$texte = $lettres;
       
          		}
          	}
       
          	return $texte;
          }
    • Génial tout ça !!

      Par contre une idée pour surligner les mots de la liste associés à ceux de la lettre en cours ?

      En tous cas Merci beaucoup !!

    • Salut,

      Finalement le tri ne fonctionnait pas tout à fait comme il faut dans le sens ou je me suis retrouvé avec des tris un peu bizarre.
      J’ai bien fait le ménage dans ce que je voulais et ajouté quelques bricoles pour que ça tourne. Cette fois j’arrive à un beau résultat.
      à la seule exception c’est qu’il me colle les doubles point dans le classement de la lettre N ?!

      J’ai tourné le truc dans tous les sens mais rien y fait. Je n’arrive pas à me débarrasser de ce fichu double point.
      Je ne sais vraiment pas quoi faire

      Voici la fonction que j’ai adapté :

      /**
       * Retourne une liste de lettres correspondant aux 1ères lettres des mots d'un texte.
       *
       * @param string $texte
       * @return string|array
       */
      function premieres_lettres($texte) {
       //print_r($texte);
       	$a = "ÀÁÂÃÄÅàáâãäåÒÓÔÕÖØòóôõöøÈÉÊËèéêëÇçÌÍÎÏìíîïÙÚÛÜùúûüÿÑñ"; 
       	$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"));
       
         //array des mots interdits à compléter
      	$mots_interdits = array('&#58;','quot','sous','avec','sans','pour','etc','dans','general','autoconstruire','autoconstruction','autoconstruites','autoconstruits','autoconstruite','Autoconstruction');
       
       
      	// liste des mots
      	if ($mots = str_word_count($texte, 1, $a)) {
      		// on stocke la 1ère lettre de chaque mot dans un tableau
      		$lettres = array();
       
      		foreach($mots as $mot) {
      			// on ne prend en compte que les mots d'au moins 4 octets
      			if (strlen($mot) > 3 AND !in_array($mot,$mots_interdits)){
      				$lettres = array_merge($lettres, array(strtoupper(substr($mot,0,1))));
       
      				//print_r($lettres);
      			}
      		}
       
      		if (count($lettres)) {
      			$texte = $lettres;
       
      			//print_r($texte);
      		}
      	}
       
      	return $texte;
      }

      aussi je reviens à la charge avec mon surlignage, j’aimerais bien que les mots ou la première lettre des mots listés pour la lettre selectionnée soient surlignés. Une idée ?

      Merci d’avance :)

    • Bonsoir,
      je ne vois pas bien pour les deux points de la lettre N …
      Concernant le surlignage, un javascript serait surement mieux, voici une fonction créée pour l’occasion en php / SPIP qui est le moins gourmande possible.
      Pour récupérer la lettre appelée dans l’url, il te faut vérifier que #ENV{lettre} fonctionne bien, à partir de là, il te faut tester la première lettre de chaque mot avec

      [(#TITRE|detect_lettrine{#ENV{lettre}})]

      /**
       * Retourne la première lettre d'une chaine encadré d'un strong 
       * seulement si cette lettre correspond au caractère demandé
       * 
       * @param string $chaine
       *    Chaine de recherche
       * @param string $lettre
       *    Caractère recherché
       * @return string
       *   La chaine modifiée
       * @example
       *    ```
       *    #SET{titre,'Écriture te voila !'} [(#GET{titre}|detect_lettrine{'e'})]
       *    ```
       *
      **/
      function detect_lettrine($chaine,$lettre){
      	//fonction SPIP pour le filtre url_nettoyer qui eliminera les accents
      	include_spip('action/editer_url');
      	//la première lettre de la chaine
      	$first_lettre = mb_substr($chaine,0,1,'UTF-8');
      	$length = strlen($first_lettre);
      	//si elle correspond à la lettre de l'url
      	if(strtolower(url_nettoyer($first_lettre,2)) == $lettre){
      		$lettre_avec_encadrement = '<strong class="lettrine">'.$first_lettre.'</strong>';
      		$chaine = substr_replace($chaine, $lettre_avec_encadrement, 0, $length);
      	}
      return $chaine;
      }
    • Merci pour cette réponse, mais je n’arrive pas à implémenter cette solution ...
      y’aurait il possibilité de m’envoyer un exemple dans une boucle ?

      Merci d’avance

    • Ah erratum, j’ai reussi à le faire fonctionner mais il ne me fait la selection que sur le premier mot de la chaine alors qu’il serait plus pertinent de faire le surlignement sur tous les mots de la chaine qui commencent par la lettre, je continue à creuser ...

      En tous cas merci ça se présente plutot bien :) :) :)

    • Qu’est-ce que tu n’arrives pas à faire ?
      Quelle est la balise qui affiche ta liste des items dans ton squelette ? si je suppose que c’est un #TITRE tu mets
      [(#TITRE|detect_lettrine{#ENV{lettre}})]
      si c’est le #CHAPO tu mets
      [(#CHAPO|detect_lettrine{#ENV{lettre}})]
      Dans l’url tu ajoutes le parametre lettre=a (ou lettre=b …)
      et sur la liste des titres, chaque titre commençant par a comme « Ah que la vie est belle » ou « Anticipation planétaire » verra sa première lettre mise en gras.
      Tu peux aussi jouer sur la class CSS .lettrine{color:red} pour colorer cette première lettre trouvée.
      ++

    • comme dit plus haut j’ai réussi à le faire fonctionner, j’ai remplacé strtolower par strtoupper et ça marche. par contre, toujours comme dit plus haut c’est à chaque mots qui commencent par la lettre selectionnée qu’il serait souhaitable de surligner.
      Je me demande si intégrer la partie surlignement

      1. $lettre_avec_encadrement = '<strong class="lettrine">'.$first_lettre.'</strong>';

      ne serait pas plus pratique.

    • Avec la fonction « premiere_lettres » utilisée plus haut j’obtiens un array :

      Array ( [0] => S ) Array ( [0] => S [1] => C ) 

      Pour un titre : Semis • Canne à semer ou Semis et semer devraient êtres surlignés.

      [mode boulet]

    • Oui, tu dois adapter le code, mais il n’aurait pas été aussi lisible, il faut récupérer les mots du titre dans un array $mots et faire un

      foreach($mots as $mot){
      // if ici la fonction de traitement de chaque mot
      // reconstruire le nouveau titre
      $titre_nouveau.=$mot;
      }
      return $titre_nouveau;
    • Je m’enmele les crayons là dessus ... je surligne ARRAY et mon tableau ne s’affiche plus ...

      Help

    • Bon bon bon,
      c’est à la fin, je suppose que c’est bien au moment de l’affichage des titres qu’il faut la fonction detect_lettrine ?

      [(#REM) 3) Affichage des articles correspondant à la lettre active (= dans l'URL) ]
      #SET{lettre,#ENV{lettre,#GET{lettre}}}
      <B_articles_lettre>
      <ul>
      <BOUCLE_articles_lettre(ARTICLES){id_article IN #GET{lettres/#GET{lettre}}}>
      <li>[(#TITRE|detect_lettrine{#ENV{lettre}})]</li>
      </BOUCLE_articles_lettre>
      </ul>
      </B_articles_lettre>

      Je disais que tu peux modifier cette fonction avec à son début la construction d’un array des mots de la chaine de manière à ce que chaque mot soit testé (donc avec un foreach) avant d’être retourné modifié ou pas, enfin, si j’ai bien compris ce que tu veux faire.
      Peux-tu mettre une URL de test pour que l’on puisse voir ce que tu veux faire ?

    • Pas de soucis pour la mise en place, dans le squelette, ça ça roule.
      C’est justement l’array que je n’arrive pas à construire et à tester.
      Je bosse en local, du coup pas possible de mettre une URL de test :/
      J’aimerais pouvoir surligner chaque lettre dans chaque titre comme sur le l’image jointe.

      Merci pour le coup de main !

    • Je crois que je suis bon pour une cure d’aspirine et surtout de PHP ...
      Je n’arrive résolument pas à reconstituer la chaine de caractères après l’avoir passée dans un tableau ...

      J’ai tenté un array sur la fonction detect_lettrine mais je n’arrive résolument pas à reconstituer la chaine à partir de cet array.

      Exemple :

      function detect_lettrine($texte,$lettre){
       
      	//fonction SPIP pour le filtre url_nettoyer qui eliminera les accents
      	include_spip('action/editer_url');
       
      	print_r($texte);
       
      	//compte les mots de chaque chaine et les place dans un tableau
       
      	if ($mots = str_word_count($texte, 1)) {
       
      		print_r($mots);
       
      			// on stocke la 1ère lettre de chaque mot dans un tableau
      			$first_lettre = array();
       
      			foreach($mots as $mot) {
       
      			//print_r($mot);
       
      			//je récupère la première lettre de la chaine
       
      			$first_lettre = mb_substr($mot,0,1,'UTF-8');
      			$first_lettre = strtoupper($first_lettre);			
      			$first_lettre = str_replace($lettre, '<strong class="lettrine">'.$lettre.'</strong>', $first_lettre);
       
       
       
       
      			//je remplace tous les caractère correspondants à la lettre correspondant à l'url 
       
      	print_r($first_lettre);
      	//print_r($length);
      	//si elle correspond à la lettre de l'url
       
      	if(strtoupper(url_nettoyer($first_lettre,2)) == $lettre){
       
      		$lettre_avec_encadrement = '<strong class="lettrine">'.$first_lettre.'</strong>';
       
      		//print_r($lettre_avec_encadrement);
       
      		$mots = substr_replace($mots, $lettre_avec_encadrement, 0, $length);
       
      		//print_r($mots);
      		}
      	}
      }
      return $texte;
       
      }

      J’ai aussi tenté une adaptation de la fonction premiere_lettre indiquée pour l’affichage du menu alphabet mais là encore je n’arrive pas à reconstituer la chaine.

      Exemple :

      function premieres_lettres_sur($texte, $lettre) {
       //print_r($texte);
       $texte = sansle($texte);
       $texte = unicode2charset(utf_8_to_unicode($texte), 'iso-8859-1');
       
       	 $a = "ÀÁÂÃÄÅàáâãäåÒÓÔÕÖØòóôõöøÈÉÊËèéêëÇçÌÍÎÏìíîïÙÚÛÜùúûüÿÑñ";
          $b = "AAAAAAaaaaaaOOOOOOooooooEEEEeeeeCcIIIIiiiiUUUUuuuuyNn";
      	$texte=strtr($texte, $a, $b); // on retire les accents
       	$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"));
       
         //array des mots interdits à compléter
      	$mots_interdits = array('guide','','Autres','autre','&#58;','quot','sous','avec','sans','pour','etc','dans','general','autoconstruit','autoconstruire','autoconstruction','autoconstruites','autoconstruits','autoconstruite','Autoconstruction');
      //print_r($mots_interdits);
       
      	// liste des mots
      	if ($mots = str_word_count($texte, 1, $a)) {
      		// on stocke la 1ère lettre de chaque mot dans un tableau
      		$lettres = array();
       
      		foreach($mots as $mot) {
      			//print_r($mots);
      			// on ne prend en compte que les mots d'au moins 4 octets
      			if (iconv_strlen($mot) > 3 AND !in_array($mot,$mots_interdits)){
      				$lettres = array_merge($lettres, array(strtoupper(substr($mot,0,1))));
      				$lettres = str_replace($lettre, '<strong class="lettrine">'.$lettre.'</strong>', $lettres);
      				//$texte = implode($lettres);
       
      				//print_r($texte);
      			}
      		}
       
      		if (count($lettres)) {
      			//print_r($lettres);
      			$texte = $lettres;
       
      			print_r($texte);
      		}
       }
      	return $texte;
      }

      Bref, dans les deux cas j’arrive à selectionner les bonnes lettres et les surligner c’est une bonne chose, mais impossible de reconstituer la chaine proprement.
      C’est vraiment la dernière partie fonctionnelle qu’il me manque avant de pouvoir lacher le kraken !

      Si quelqu’un à une idée j’en serais très reconnaissant :)

      Merci d’avance

    Répondre à ce message

  • Tangerine

    Bonjour,

    Est-ce que le tri alphabétique tout en Spip fonctionne avec SPIP 3 ?
    Si oui, y a-t-il des particularités dans la mise en place ?

    Bien cordialement,

    Guillaume.

    Répondre à ce message

  • Le nouveau critère de boucle {collecte} à partir de SPIP 2 permet de se faciliter la vie pour trier proprement avec des caractères un peu exotiques comme l’E dans l’O.

    http://www.spip.net/fr_article4028.html

    Je l’ai utilisé avec utf8_swedish_ci et ça marche impec, les articles commençant par un Œ se retrouvent après ceux commençant par un O, au lieu de finir tristement après le Z...

    #SET{collation, utf8_swedish_ci}
    <BOUCLE_tri(ARTICLES) {par titre} {collecte #GET{collation}}> ...

    Pour un affichage avec l’initiale, pour un sommaire par lettre par exemple, il faut jouer avec la translittération pour récupérer un O et pas un Œ :

    [(#SET{initiale,#TITRE|translitteration|mb_substr{0,1}|unique})]

    Sinon on se retrouve avec :
    O
    Occidental

    Œ
    Œuvre
    Oriental

    P
    ...

    Répondre à ce message

  • Pas certain d’avoir une réponse vu que le tuto date un peu, je tente quand même. Super guide en tout cas que ce tri tout en SPIP.

    Je l’ai adapté à une revue en ligne utilisant le squelette RevueLigne. Ca marche nickel. Mais...
    J’aimerais aller plus loin. Imaginons que j’aie deux auteurs, chacun apparaissant dans ma table articles sous les champs auteur1 et auteur2, quelqu’un a-t-il une piste pour adapter ce tri à cette exigence ? Me faut-il règler le problème des deux champs dans une fonction PHP ?
    Merci par avance :-)

    Répondre à ce message

  • 1

    Bonjour,
    j’utilise le code suivant :

    1. <BOUCLE_listalpha(ARTICLES){id_secteur}{par titre}>[(#SET{beginby[(#TITRE|onelettre)],[(#GET{beginby[(#TITRE|onelettre)]}|array_merge{#ARRAY{#COMPTEUR_BOUCLE,#ID_ARTICLE}})]})][<a href="[(#SELF|parametre_url{lettre,[(#TITRE|onelettre)]})]" [(#TITRE|onelettre|=={[(#ENV{lettre,A})]}|?{class='expose'})]>(#TITRE|onelettre|unique)</a>|]</BOUCLE_listalpha>

    Tout marche super bien seulement je souhaiterais afficher les lettres (inactives) pour lesquelles il n’y a pas d’article. Comment faire ?

    Merci d’avance,
    Mathieu.

    • Déjà il y a plus simple pour extraire la première lettre avec le filtre php |substr{0,1} mieux vaut d’ailleurs se référer à l’article de marcimat sur son site voir http://marcimat.magraine.net/Tri-al...

      Mais dans ton cas, si tu veux toutes les lettres de l’alphabet, il faudrait écrire en dur les lettres que tu souhaites, quelque chose du style

      <ul>
      <li><a href="[(#SELF|parametre_url{lettre,a})]">A</a></li>
      <li><a href="[(#SELF|parametre_url{lettre,b})]">B</a></li>
      ...etc
      </ul>

      Enfin, c’est ce qui me semble le plus simple.

    Répondre à ce message

  • merci c’est excellent !! bon ca maffiche 0-9 dans le menu même sil ny a rien.
    Autrement question concernant la reécriture d’url. Actuellement ca sort ainsi /glossaire.html&lettre=a , comment faire pour que les urls sortent propres du style : /glossaire-a.html ??

    Répondre à ce message

  • Mathieu Doublet

    Bonjour,

    je ne me sers pas de tout (simplement des deux premières étapes) et ça marche du feu de dieu pour l’instant.

    Un grand merci !

    Répondre à ce message

  • 2

    Bonjour,
    quelqu’un utilise cette contrib sous spip 2.1 ?
    si oui quelles modifications sont nécessaires pour que ça marche ?

    Répondre à ce message

  • 1

    Attention, ce plugin ne fonctionne pas en 2.0.10. En 2.0.07, no problem.
    En 2.0.10, j’ai des erreurs sur array_merge() [function.array-merge] : Argument #1 is not an array in composer.php(51) : eval()’d code on line 190. A suivre...

    • si ca fonctionne en 2.0.10 (avec PHP5), il suffit de rajouter ce qu’a dit marc dans un commentaire précédent dans mes_fonctions.php

      pour mémoire sa solution

      function array_merge5 ($array1, $array2) {
      return array_merge ((array)$array1, (array)$array2);
      }

      un exemple qui marche http://reflexes.samizdat.net/spip.php?rubrique16

    Répondre à ce message

  • finalement cela fonctionne t’il avec spip 2 ?
    ou existe t’il un plugin quelque part ?
    merci

    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