ADD Bannissement (sans le slap)
authorGreg Burri <greg.burri@gmail.com>
Sat, 10 May 2008 22:23:05 +0000 (22:23 +0000)
committerGreg Burri <greg.burri@gmail.com>
Sat, 10 May 2008 22:23:05 +0000 (22:23 +0000)
13 files changed:
css/1/pageMinichat.css
doc/graphiques/icones/ban.xcf [new file with mode: 0644]
doc/graphiques/icones/kick.xcf [new file with mode: 0644]
doc/graphiques/icones/slap.xcf [new file with mode: 0644]
doc/graphiques/path2383.png [new file with mode: 0644]
doc/protocole3.txt
img/ban.gif [new file with mode: 0644]
img/kick.gif [new file with mode: 0644]
img/slap.gif [new file with mode: 0644]
js/euphorik.js
js/pageMinichat.js
modules/erl/euphorik_bd.erl
modules/erl/euphorik_protocole.erl

index d7cd580..234f9f5 100755 (executable)
        opacity: 0.5;
 }
 
        opacity: 0.5;
 }
 
+#outilsBan {
+       border-width: 1px 1px 1px 1px;
+       border-color: #253f18;
+       border-style: solid;
+       padding: 2px;
+       position: absolute;
+       display: none;
+}
+#outilsBan img {
+       float: right;
+       vertical-align: middle;
+       margin: 0px 0px 0px 0px;
+}
+
 #page.minichat .titreSmiles:hover {
        background-color: #2d8800;
 }
 #page.minichat .titreSmiles:hover {
        background-color: #2d8800;
 }
diff --git a/doc/graphiques/icones/ban.xcf b/doc/graphiques/icones/ban.xcf
new file mode 100644 (file)
index 0000000..5819921
Binary files /dev/null and b/doc/graphiques/icones/ban.xcf differ
diff --git a/doc/graphiques/icones/kick.xcf b/doc/graphiques/icones/kick.xcf
new file mode 100644 (file)
index 0000000..569ef77
Binary files /dev/null and b/doc/graphiques/icones/kick.xcf differ
diff --git a/doc/graphiques/icones/slap.xcf b/doc/graphiques/icones/slap.xcf
new file mode 100644 (file)
index 0000000..ef1af5d
Binary files /dev/null and b/doc/graphiques/icones/slap.xcf differ
diff --git a/doc/graphiques/path2383.png b/doc/graphiques/path2383.png
new file mode 100644 (file)
index 0000000..6313230
Binary files /dev/null and b/doc/graphiques/path2383.png differ
index f0b0b38..6c1032e 100644 (file)
@@ -161,7 +161,8 @@ Le format de la date n'est pas formel.
             "last_page" : true | false,\r
             "messages" : [
                {\r
             "last_page" : true | false,\r
             "messages" : [
                {\r
-                  "id" : 54,\r
+                  "id" : 54,
+                  "user_id" : 344,\r
                   "date" : "Hier 17:26:54",\r
                   "system" : true | false,\r
                   "owner" : true | false,\r
                   "date" : "Hier 17:26:54",\r
                   "system" : true | false,\r
                   "owner" : true | false,\r
diff --git a/img/ban.gif b/img/ban.gif
new file mode 100644 (file)
index 0000000..9a7d412
Binary files /dev/null and b/img/ban.gif differ
diff --git a/img/kick.gif b/img/kick.gif
new file mode 100644 (file)
index 0000000..ce127dd
Binary files /dev/null and b/img/kick.gif differ
diff --git a/img/slap.gif b/img/slap.gif
new file mode 100644 (file)
index 0000000..2265fda
Binary files /dev/null and b/img/slap.gif differ
index 2d64fe2..6ed1963 100755 (executable)
@@ -673,10 +673,10 @@ Client.prototype.connexion = function(messageJson)
 \r
 Client.prototype.deconnexion = function()\r
 {
 \r
 Client.prototype.deconnexion = function()\r
 {
-   this.flush()\r
+   this.flush()
+   this.delCookie()\r
    this.setStatut(statutType.deconnected) // deconnexion\r
    this.resetDonneesPersonnelles()\r
    this.setStatut(statutType.deconnected) // deconnexion\r
    this.resetDonneesPersonnelles()\r
-   this.delCookie ()\r
 }
 
 Client.prototype.chargerDonnees = function(data)
 }
 
 Client.prototype.chargerDonnees = function(data)
@@ -771,6 +771,39 @@ Client.prototype.majMenu = function()
    }
 }
 
    }
 }
 
+Client.prototype.ban = function(userId, minutes)
+{
+   var thisClient = this
+
+   // par défaut un ban correspond à 3 jours
+   if (typeof(minutes) == "undefined")
+      minutes = 60 * 24 * 3
+      
+   jQuery.ajax({
+      type: "POST",
+      url: "request",
+      dataType: "json",
+      data: this.util.jsonVersAction(
+         {
+            "action" : "ban",
+            "cookie" : thisClient.cookie,
+            "duration" : minutes,
+            "user_id" : userId
+         }),
+      success: 
+         function(data)
+         {
+            if (data["reply"] == "error")
+               thisClient.util.messageDialogue(data["error_message"])
+         }
+   })
+}
+
+Client.prototype.kick = function(userId)
+{
+   this.ban(userId, 15)
+}
+
 ///////////////////////////////////////////////////////////////////////////////////////////////////
 
 function initialiserListeStyles(client)
 ///////////////////////////////////////////////////////////////////////////////////////////////////
 
 function initialiserListeStyles(client)
index 2be7ca5..94f6e7c 100755 (executable)
@@ -40,6 +40,21 @@ PageMinichat.prototype.charger = function()
    \r
    this.util.setCaretToEnd(jQuery("form input.message")[0])
 
    \r
    this.util.setCaretToEnd(jQuery("form input.message")[0])
 
+   // les outils de bannissement (uniquement pour les ekMaster)
+   if (this.client.ekMaster)
+   {    
+      jQuery("body").append(
+         "<div id=\"outilsBan\"><img id=\"ban\" alt=\"Ban de 3 jours\" src=\"img/ban.gif\" /><img id=\"kick\" alt=\"Ban de 15min\" src=\"img/kick.gif\" /><img id=\"slap\" alt=\"Avertissement\" src=\"img/slap.gif\" /></div>"
+      )
+      jQuery("#outilsBan").hover(
+         function(){},
+         function()
+         {
+            jQuery("#outilsBan").hide()
+         }
+      )
+   }
+
    // <smiles>
    jQuery("body").append("<div id=\"smiles\"></div>")
    // affichage des smiles
    // <smiles>
    jQuery("body").append("<div id=\"smiles\"></div>")
    // affichage des smiles
@@ -134,6 +149,7 @@ PageMinichat.prototype.decharger = function()
    this.messages.stopAttenteCourante()
    
    jQuery("body #smiles").remove()
    this.messages.stopAttenteCourante()
    
    jQuery("body #smiles").remove()
+   jQuery("body #outilsBan").remove()
 }
 
 PageMinichat.prototype.getJSONMessage = function(pseudo, message, repondA)
 }
 
 PageMinichat.prototype.getJSONMessage = function(pseudo, message, repondA)
@@ -209,7 +225,11 @@ PageMinichat.prototype.envoyerMessage = function(pseudo, message)
                         thisPageMinichat.messages.conversations[c].messages[m].clientARepondu = true
                   */
                }
                         thisPageMinichat.messages.conversations[c].messages[m].clientARepondu = true
                   */
                }
-            }\r
+            }
+            else if (data["reply"] == "error")
+            {
+               thisPageMinichat.util.messageDialogue(data["error_message"])
+            }  \r
          }\r
       }\r
    )
          }\r
       }\r
    )
@@ -239,9 +259,10 @@ function Reponse(id, pseudo, login)
   * @param pseudo
   * @param contenu
   */
   * @param pseudo
   * @param contenu
   */
-function Message(id, date, pseudo, login, contenu)
+function Message(id, auteurId, date, pseudo, login, contenu)
 {
    this.id = id
 {
    this.id = id
+   this.auteurId = auteurId
    this.date = date
    this.pseudo = pseudo
    this.login = login
    this.date = date
    this.pseudo = pseudo
    this.login = login
@@ -484,7 +505,7 @@ Conversation.prototype.flush = function(funClickOuvrirConv)
             "\">" +
                "<div class=\"extraire\">&gt;</div>" +
                "[<span class=\"date\">" + message.date + "</span>]" +
             "\">" +
                "<div class=\"extraire\">&gt;</div>" +
                "[<span class=\"date\">" + message.date + "</span>]" +
-               "<span class=\"pseudo\">" + identifiant + "</span>:" +
+               "<span class=\"pseudo\" id=\"user" + message.auteurId + "\">" + identifiant + "</span>:" +
                XHTMLrepondA +
                "<span class=\"contenu\">" + (message.systeme ? this.formateur.remplacerBalisesHTML(message.contenu) : this.formateur.traitementComplet(message.contenu, message.pseudo)) + "</span>" +
             "</div>"
                XHTMLrepondA +
                "<span class=\"contenu\">" + (message.systeme ? this.formateur.remplacerBalisesHTML(message.contenu) : this.formateur.traitementComplet(message.contenu, message.pseudo)) + "</span>" +
             "</div>"
@@ -518,13 +539,40 @@ Conversation.prototype.flush = function(funClickOuvrirConv)
             }
          )
          
             }
          )
          
+         // les outils de bannissement (uniquement pour les ekMaster)
+         if (thisConversation.client.ekMaster)
+            jQuery(".pseudo", this).hover(
+               function(e)
+               {     
+                  var userId =  parseInt(jQuery(this).attr("id").substr(4))
+                  var element = jQuery(e.target)
+                  var h = element.height()
+                  var offset = element.offset()
+                  var outils = jQuery("#outilsBan").css("top", offset.top - 2).css("left", offset.left - 2).height(h < 16 ? 16 : h).width(element.width() + 16 * 3 + 4).show()
+                  jQuery("img", outils).unbind()
+                  jQuery("#kick", outils).click(
+                     function(e)
+                     {
+                        thisConversation.client.kick(userId)
+                     }
+                  )
+                  jQuery("#ban", outils).click(
+                     function(e)
+                     {
+                        thisConversation.client.ban(userId)
+                     }
+                  )
+               },
+               function(){}
+            )
+         
          jQuery(this).click(
             function(event)
             {
                if (jQuery(event.target).is("a")) return
                
                // l'id du message
          jQuery(this).click(
             function(event)
             {
                if (jQuery(event.target).is("a")) return
                
                // l'id du message
-               idMess = jQuery(this).attr("id")
+               var idMess = jQuery(this).attr("id")
                
                // extraction d'une conversation
                if (jQuery(event.target).is(".extraire"))
                
                // extraction d'une conversation
                if (jQuery(event.target).is(".extraire"))
@@ -611,9 +659,6 @@ function Messages(client, formateur, util)
    
    this.conversations = new Array() // les conversations, la première représente la conversation principale
    this.nouvelleConversation(0)
    
    this.conversations = new Array() // les conversations, la première représente la conversation principale
    this.nouvelleConversation(0)
-
-// Obsolète
-   //this.idDernierMessage = null // l'id du dernier message connu
    
    // l'objet JSONHttpRequest représentant la connexion d'attente
    this.attenteCourante = null
    
    // l'objet JSONHttpRequest représentant la connexion d'attente
    this.attenteCourante = null
@@ -690,6 +735,7 @@ Messages.prototype.ajouterMessage = function(element, numConversation)
    
    var message = new Message(
       id,      
    
    var message = new Message(
       id,      
+      element["user_id"],
       element["date"],
       element["nick"],
       element["login"],
       element["date"],
       element["nick"],
       element["login"],
@@ -849,8 +895,6 @@ Messages.prototype.rafraichirMessages = function(vider)
       
    this.stopAttenteCourante()
    
       
    this.stopAttenteCourante()
    
-   /*if (vider)
-      this.idDernierMessage = null*/
    if (vider)
       for (var i = 0; i < this.conversations.length; i++)
          this.conversations[i].idDernierMessageAffiche = 0\r
    if (vider)
       for (var i = 0; i < this.conversations.length; i++)
          this.conversations[i].idDernierMessageAffiche = 0\r
index 1b37b4c..87dab2f 100755 (executable)
@@ -29,6 +29,7 @@
    
    % messages :
    nouveau_message/3,
    
    % messages :
    nouveau_message/3,
+   nouveau_message_sys/1,
    messages/1,
    messages/2,\r
    messages/3,
    messages/1,
    messages/2,\r
    messages/3,
@@ -44,6 +45,8 @@
    % ip :
    ip_table/0,
    ban/2,
    % ip :
    ip_table/0,
    ban/2,
+   deban/1,
+   est_banni/1,
    can_register/1,
    
    % versions :
    can_register/1,
    
    % versions :
@@ -114,10 +117,11 @@ reset() ->
    mnesia:clear_table(counter),\r
    mnesia:clear_table(user),\r
    mnesia:clear_table(reponse_minichat),\r
    mnesia:clear_table(counter),\r
    mnesia:clear_table(user),\r
    mnesia:clear_table(reponse_minichat),\r
-   mnesia:clear_table(minichat),\r
+   mnesia:clear_table(minichat),
+   mnesia:clear_table(ip_table),\r
    % crée l'utilisateur root\r
    mnesia:transaction(fun() ->\r
    % crée l'utilisateur root\r
    mnesia:transaction(fun() ->\r
-      User = #user{id = 0, pseudo = "Sys", login = "Sys", date_creation = now(), date_derniere_connexion = now()},\r
+      User = #user{id = 0, pseudo = "Sys", login = "Sys", date_creation = now(), date_derniere_connexion = now(), ek_master = true},\r
       mnesia:write(User),\r
       User\r
    end).
       mnesia:write(User),\r
       User\r
    end).
@@ -160,7 +164,7 @@ set_profile(Cookie, Login, Password, Pseudo, Email, Css, Nick_format, Page_princ
                               % TODO : pourquoi ne pas tester avec la valeur "undefined" plutôt qu'avec "is_list" ?
                               % TODO : validation plus strict des données (pas de page négative dans les conv par exemple)
                               login = if is_list(Login) -> Login; true -> User#user.login end,
                               % TODO : pourquoi ne pas tester avec la valeur "undefined" plutôt qu'avec "is_list" ?
                               % TODO : validation plus strict des données (pas de page négative dans les conv par exemple)
                               login = if is_list(Login) -> Login; true -> User#user.login end,
-                              password = if is_list(Password) -> Password; true -> User#user.password end,
+                              password = if is_list(Password) andalso Password =/= [] -> Password; true -> User#user.password end,
                               pseudo = if is_list(Pseudo) -> Pseudo; true -> User#user.pseudo end,
                               email = if is_list(Email) -> Email; true -> User#user.email end,
                               css = if is_list(Css) -> Css; true -> User#user.css end,
                               pseudo = if is_list(Pseudo) -> Pseudo; true -> User#user.pseudo end,
                               email = if is_list(Email) -> Email; true -> User#user.email end,
                               css = if is_list(Css) -> Css; true -> User#user.css end,
@@ -303,12 +307,6 @@ user_by_mess(Id) ->
       end
    )).\r
    \r
       end
    )).\r
    \r
-   \r
-% Renvoie l'utilisateur root\r
-root() ->\r
-   {ok, User} = user_by_id(0),\r
-   User.
-   
    
 % Ajoute un message. Repond_A est une liste d'id auquel le message répond
 % retourne soit l'id du message soit erreur.
    
 % Ajoute un message. Repond_A est une liste d'id auquel le message répond
 % retourne soit l'id du message soit erreur.
@@ -324,7 +322,6 @@ nouveau_message(Mess, Auteur_id, Repond_A) ->
       if Nb_id_trouve =/= length(Repond_A) -> throw("Un ou plusieurs messages introuvable");
          true -> ok
       end,
       if Nb_id_trouve =/= length(Repond_A) -> throw("Un ou plusieurs messages introuvable");
          true -> ok
       end,
-      Id = nouvel_id(minichat),
       % compare les dernière
       Delta = delta_date_ms(Auteur#user.date_derniere_connexion, now()),
       Nouvel_indice_flood = Auteur#user.indice_flood + if Delta =< ?DUREE_SPAM -> 2; true -> -1 end,
       % compare les dernière
       Delta = delta_date_ms(Auteur#user.date_derniere_connexion, now()),
       Nouvel_indice_flood = Auteur#user.indice_flood + if Delta =< ?DUREE_SPAM -> 2; true -> -1 end,
@@ -334,14 +331,13 @@ nouveau_message(Mess, Auteur_id, Repond_A) ->
       },
       % est-ce que l'auteur à trop floodé ?
       if Auteur#user.indice_flood =/= ?INDICE_SPAM_MAX, Auteur_maj#user.indice_flood =:= ?INDICE_SPAM_MAX, Delta =< ?DUREE_BLOCAGE_SPAM ->
       },
       % est-ce que l'auteur à trop floodé ?
       if Auteur#user.indice_flood =/= ?INDICE_SPAM_MAX, Auteur_maj#user.indice_flood =:= ?INDICE_SPAM_MAX, Delta =< ?DUREE_BLOCAGE_SPAM ->
-         Root = root(),
          mnesia:write(Auteur#user{indice_flood = Auteur_maj#user.indice_flood}),
          mnesia:write(Auteur#user{indice_flood = Auteur_maj#user.indice_flood}),
-         mnesia:write(#minichat{id=Id, auteur_id=Root#user.id, date=now(), pseudo=Root#user.pseudo, contenu=Auteur#user.pseudo ++ "(" ++ Auteur#user.login ++ ") est bloqué pour " ++ integer_to_list(trunc(?DUREE_BLOCAGE_SPAM / 1000)) ++ " secondes pour cause de flood.."}),
-         Id;
+         nouveau_message_sys(Auteur#user.pseudo ++ "(" ++ Auteur#user.login ++ ") est bloqué pour " ++ integer_to_list(trunc(?DUREE_BLOCAGE_SPAM / 1000)) ++ " secondes pour cause de flood..");
       Auteur#user.indice_flood =:= ?INDICE_SPAM_MAX, Delta =< ?DUREE_BLOCAGE_SPAM ->
          erreur;
       true ->     
          mnesia:write(Auteur_maj),
       Auteur#user.indice_flood =:= ?INDICE_SPAM_MAX, Delta =< ?DUREE_BLOCAGE_SPAM ->
          erreur;
       true ->     
          mnesia:write(Auteur_maj),
+         Id = nouvel_id(minichat),
          inserer_reponses(Id, Repond_A),
          mnesia:write(#minichat{id=Id, auteur_id=Auteur#user.id, date=now(), pseudo=Auteur#user.pseudo, contenu=Mess}),
          Id
          inserer_reponses(Id, Repond_A),
          mnesia:write(#minichat{id=Id, auteur_id=Auteur#user.id, date=now(), pseudo=Auteur#user.pseudo, contenu=Mess}),
          Id
@@ -356,6 +352,18 @@ inserer_reponses(Id_repondant, [Id_mess | Reste]) ->
 inserer_reponses(_, []) ->
    ok.
    
 inserer_reponses(_, []) ->
    ok.
    
+% Permet de créer un message système.
+% Renvoie l'id du message système
+nouveau_message_sys(Mess) ->
+   {ok, Root} = user_by_id(0),
+   mnesia:transaction(
+      fun() ->
+         Id = nouvel_id(minichat),
+         mnesia:write(#minichat{id=Id, auteur_id=0, date=now(), pseudo=Root#user.pseudo, contenu=Mess})
+      end
+   ).
+   
    
 % Renvoie N messages se trouvant sur la première page
 messages(N) ->  
    
 % Renvoie N messages se trouvant sur la première page
 messages(N) ->  
@@ -498,6 +506,47 @@ ban(IP, Duration) ->
    ).
    
 
    ).
    
 
+% Débanni une ip
+deban(IP) ->
+   ban(IP, 0).
+
+   
+% Renvoie soit {true, Temps} où Temps est le temps en minutes pendant lequel le user est encore banni
+% ou false.
+est_banni(User_id) ->
+   resultat_transaction(mnesia:transaction(
+      fun() ->
+         case qlc:e(qlc:q([
+            {IP#ip_table.ban, IP#ip_table.ban_duration} ||
+            U <- mnesia:table(user),
+            U#user.id =:= User_id,
+            IP <- mnesia:table(ip_table),
+            IP#ip_table.ip =:= U#user.last_ip
+         ])) of
+            [{Ban, Ban_duration}] ->
+               Echeance = date_plus_minutes(Ban, Ban_duration),
+               Now = now(),
+               if Echeance < Now -> % l'échéance est passée
+                     false;
+                  true ->
+                     {true, trunc(delta_date_ms(Echeance, Now) / 1000 / 60)}
+               end;
+            _ ->
+               false
+         end
+      end
+   )).
+   
+   
+% Ban est une date tel que retourner par now().
+% Ban_duration est un temps en minutes.
+% retourne une date.
+date_plus_minutes(Ban, Ban_duration) ->
+   Duration_sec = Ban_duration * 60,
+   {MegaSec, Sec, MicroSec} = Ban,
+   {MegaSec + if Sec + Duration_sec >= 1000000 -> 1; true -> 0 end,(Sec + Duration_sec) rem 1000000, MicroSec}.
+   
+
 % Si deux enregistrements consequtifs de la même IP sont fait en moins d'une seconde alors 
 % ip_table.nb_try_register est incrémenté de 1 sinon il est décrémenté de 1 (jusqu'a 0).
 % Si ip_table.nb_try_register vaut 5 alors l'ip ne peux plus s'enregistrer pour une heure.
 % Si deux enregistrements consequtifs de la même IP sont fait en moins d'une seconde alors 
 % ip_table.nb_try_register est incrémenté de 1 sinon il est décrémenté de 1 (jusqu'a 0).
 % Si ip_table.nb_try_register vaut 5 alors l'ip ne peux plus s'enregistrer pour une heure.
index 194961e..dc29bf5 100755 (executable)
@@ -150,22 +150,17 @@ wait_event(Data) ->
                            Est_proprietaire = User =/= inconnu andalso User#user.id =:= Mess#minichat.auteur_id,
                            A_repondu_a_message = User =/= inconnu andalso euphorik_bd:a_repondu_a_message(User#user.id, Mess#minichat.id),
                            Est_une_reponse_a_user = User =/= inconnu andalso euphorik_bd:est_une_reponse_a_user(User#user.id, Mess#minichat.id),
                            Est_proprietaire = User =/= inconnu andalso User#user.id =:= Mess#minichat.auteur_id,
                            A_repondu_a_message = User =/= inconnu andalso euphorik_bd:a_repondu_a_message(User#user.id, Mess#minichat.id),
                            Est_une_reponse_a_user = User =/= inconnu andalso euphorik_bd:est_une_reponse_a_user(User#user.id, Mess#minichat.id),
-                           User_mess =
-                              if Mess#minichat.auteur_id =:= 0 ->
-                                    inconnu;
-                                 true ->
-                                    {ok, U2} = euphorik_bd:user_by_id(Mess#minichat.auteur_id),
-                                    U2
-                              end,
+                           {ok, User_mess } = euphorik_bd:user_by_id(Mess#minichat.auteur_id),
                            {struct, [
                               {id, Mess#minichat.id},
                            {struct, [
                               {id, Mess#minichat.id},
+                              {user_id, User_mess#user.id},
                               {date, format_date(Mess#minichat.date)},
                               {system, Mess#minichat.auteur_id =:= 0},
                               {owner, Est_proprietaire},
                               {answered, A_repondu_a_message},
                               {is_a_reply, Est_une_reponse_a_user},
                               {nick, Mess#minichat.pseudo},
                               {date, format_date(Mess#minichat.date)},
                               {system, Mess#minichat.auteur_id =:= 0},
                               {owner, Est_proprietaire},
                               {answered, A_repondu_a_message},
                               {is_a_reply, Est_une_reponse_a_user},
                               {nick, Mess#minichat.pseudo},
-                              {login, if User_mess =:= inconnu -> Mess#minichat.pseudo; true -> User_mess#user.login end},
+                              {login, User_mess#user.login},
                               {content, Mess#minichat.contenu},
                               {answer_to, {array, lists:map(
                                  fun(Id_mess) ->                   
                               {content, Mess#minichat.contenu},
                               {answer_to, {array, lists:map(
                                  fun(Id_mess) ->                   
@@ -205,21 +200,37 @@ put_message(
 ) ->
    case euphorik_bd:user_by_cookie(Cookie) of
       {ok, User} ->
 ) ->
    case euphorik_bd:user_by_cookie(Cookie) of
       {ok, User} ->
-         Strip_content = string:strip(Content),
-         if (Strip_content =:= []) ->
-               erreur("Message vide");
-            true ->
-               % TODO : non-atomique (update_pseudo+nouveau_message)
-               euphorik_bd:update_pseudo_user(User#user.id, Nick),
-               case euphorik_bd:nouveau_message(Strip_content, User#user.id, Answer_to) of
-                     erreur -> erreur("Impossible d'ajouter un nouveau message");
-                  _ ->
-                     json_reponse_ok()
+         case euphorik_bd:est_banni(User#user.id) of
+            {true, Temps_restant} ->
+               erreur("Vous êtes banni pour encore " ++ format_minutes(Temps_restant));
+            _ ->
+               Strip_content = string:strip(Content),
+               if Strip_content =:= [] ->
+                     erreur("Message vide");
+                  true ->
+                     % TODO : non-atomique (update_pseudo+nouveau_message)
+                     euphorik_bd:update_pseudo_user(User#user.id, Nick),
+                     case euphorik_bd:nouveau_message(Strip_content, User#user.id, Answer_to) of
+                           erreur -> erreur("Impossible d'ajouter un nouveau message");
+                        _ ->
+                           json_reponse_ok()
+                     end
                end
          end;
                end
          end;
-      _ ->
-         erreur("Utilisateur inconnu")
-      end.
+   _ ->
+      erreur("Utilisateur inconnu")
+   end.
+   
+   
+% Formatage de minutes.
+% par exemple : "1min", "45min", "1h23min", "1jour 2h34min"
+format_minutes(Min) ->
+   Jours = Min div (60 * 24),
+   Heures = Min rem (60 * 24) div 60,
+   Minutes = Min rem (60),
+   if Jours =/= 0 -> integer_to_list(Jours) ++ "Jour" ++ if Jours > 1 -> "s"; true -> "" end ++ " "; true -> "" end ++
+   if Heures =/= 0 -> integer_to_list(Heures) ++ "h"; true -> "" end ++
+   lists:flatten(io_lib:format(if Jours =:= 0, Heures =:= 0 -> "~w"; true -> "~2.2.0w" end, [Minutes])) ++ "min".
 
 
 % bannissement d'un utilisateur (son ip est bannie)
 
 
 % bannissement d'un utilisateur (son ip est bannie)
@@ -229,18 +240,31 @@ ban(
       {duration, Duration},
       {user_id, User_id}
    ]) ->
       {duration, Duration},
       {user_id, User_id}
    ]) ->
-   % controle que l'utilisateur est un admin
-   case euphorik_bd:user_by_cookie(Cookie) of
-      {ok, User = #user{ek_master = true}} ->
-         case euphorik_bd:user_by_id(User_id) of
-            {ok, User} ->
-               euphorik_bd:ban(User#user.last_ip, Duration);
-            _ ->
-               erreur("Utilisateur à bannir inconnu")
-         end;
-      _ ->
-         erreur("Utilisateur inconnu ou non ek master")
-   end.
+      % controle que l'utilisateur est un admin
+      case euphorik_bd:user_by_cookie(Cookie) of
+         {ok, User1 = #user{ek_master = true}} ->
+            case euphorik_bd:user_by_id(User_id) of
+               {ok, User1} ->
+                  erreur("Il n'est pas possible de s'auto bannir");
+               {ok, User2 = #user{ek_master = false}} ->
+                  euphorik_bd:ban(User2#user.last_ip, Duration),
+                  euphorik_bd:nouveau_message_sys(lists:flatten(io_lib:format("~s ~s est ~s pour ~s",
+                     [
+                        User2#user.pseudo,
+                        if User2#user.login =:= [] -> ""; true -> "(" ++ User2#user.login ++ ")" end,
+                        if Duration =< 15 -> "kické"; true -> "banni" end,
+                        format_minutes(Duration)
+                     ]
+                  ))),
+                  json_reponse_ok();
+               {ok, _} ->
+                  erreur("L'utilisateur est lui même un ekMaster");
+               _ ->
+                  erreur("Utilisateur à bannir inconnu")
+            end;
+         _ ->
+            erreur("Utilisateur inconnu ou non ek master")
+      end.
       
 
 % Construit une erreur
       
 
 % Construit une erreur
@@ -273,7 +297,7 @@ format_date(Date) ->
       end ++\r
       io_lib:format("~2.10.0B:~2.10.0B:~2.10.0B", [Heure, Minute, Seconde])
    ).
       end ++\r
       io_lib:format("~2.10.0B:~2.10.0B:~2.10.0B", [Heure, Minute, Seconde])
    ).
-   
+
 
 json_reponse_ok() ->
    {struct, [{reply, "ok"}]}.
 
 json_reponse_ok() ->
    {struct, [{reply, "ok"}]}.